From a78bd8404a31ee31913574d3bce5c5ba23868673 Mon Sep 17 00:00:00 2001 From: philc Date: Fri, 12 May 2023 07:59:32 +0200 Subject: [PATCH] update auth openpgp.js --- .gitignore | 5 +- api/middlewares/checkHeaders.js | 171 +- api/middlewares/hasAccessrighton.js | 93 +- api/middlewares/isAuthenticated.js | 261 +- api/middlewares/lg/headers_en.json | 12 +- api/models/Notifications.js | 20 + api/models/Pagans.js | 204 +- api/models/Setup.js | 127 +- api/models/Wwws.js | 118 + api/models/lg/Pagans_en.json | 6 +- api/routes/nations.js | 4 + api/routes/odmdb.js | 25 +- api/routes/pagans.js | 448 +- api/routes/wwws.js | 45 + apxtrib.js | 225 +- electedtown.json | 1 + .../{nginx => ASUPnginx}/nginx.conf.mustache | 0 .../proxy_params copy.mustache} | 0 .../adminapx/ASUPnginx/proxy_params.mustache | 11 + nationchains/www/adminapx/index_en.html | 207 +- .../www/adminapx/static/img/logo/favicon.png | Bin 0 -> 3785 bytes .../adminapx/static/img/logo/logobgdark.png | Bin 0 -> 7188 bytes .../adminapx/static/img/logo/logobgdark.svg | 168 + .../adminapx/static/img/logo/logobglight.png | Bin 0 -> 10193 bytes .../adminapx/static/img/logo/logobglight.svg | 156 + .../static/img/logo/logocarredark.png | Bin 0 -> 4606 bytes .../static/img/logo/logocarrelight.png | Bin 0 -> 4255 bytes nationchains/www/adminapx/static/js/apxapp.js | 163 +- .../www/adminapx/static/js/apxpagans.js | 232 + .../www/adminapx/static/js/apxtowns.js | 0 .../www/adminapx/static/js/apxtribcli.js | 89 +- .../www/adminapx/static/js/apxtribes.js | 0 .../adminapx/static/style/fonts/logo/OFL.txt | 93 + .../logo/Quicksand-VariableFont_wght.ttf | Bin 0 -> 116832 bytes .../static/style/fonts/logo/Quicksand.zip | Bin 0 -> 489827 bytes .../static/style/fonts/logo/README.txt | 67 + .../fonts/logo/static/Quicksand-Bold.ttf | Bin 0 -> 72836 bytes .../fonts/logo/static/Quicksand-Light.ttf | Bin 0 -> 72964 bytes .../fonts/logo/static/Quicksand-Medium.ttf | Bin 0 -> 73176 bytes .../fonts/logo/static/Quicksand-Regular.ttf | Bin 0 -> 73200 bytes .../fonts/logo/static/Quicksand-SemiBold.ttf | Bin 0 -> 73084 bytes .../adminapx/static/style/fonts/texte/OFL.txt | 93 + .../style/fonts/texte/Questrial-Regular.ttf | Bin 0 -> 148120 bytes .../static/style/fonts/texte/Questrial.zip | Bin 0 -> 152858 bytes .../style/fonts/texte/generator_config.txt | 5 + .../fonts/texte/questrial-regular-demo.html | 706 + .../texte/questrial-regular-webfont.woff | Bin 0 -> 26760 bytes .../texte/questrial-regular-webfont.woff2 | Bin 0 -> 19964 bytes .../specimen_files/grid_12-825-55-15.css | 370 + .../specimen_files/specimen_stylesheet.css | 502 + .../static/style/fonts/texte/stylesheet.css | 12 + .../texte/webfontkit-20230503-055110.zip | Bin 0 -> 59824 bytes .../adminapx/static/tpl/apxmodal_en.mustache | 29 + .../static/tpl/articleeditor_en.mustache | 1 + ...e_en.mustache => articleslist_en.mustache} | 0 .../static/tpl/emailregisterhtml_en.mustache | 17 + .../static/tpl/emailregistertxt_en.mustache | 17 + .../adminapx/static/tpl/nginx.conf.mustache | 34 + .../tpl/nginxmodelwebsite.conf.mustache} | 0 .../static/tpl/nginxproxyparams.mustache | 11 + .../static/tpl/pagancreate_en.mustache | 173 +- ..._en.mustache => paganloginout_en.mustache} | 0 .../adminapx/static/tpl/townsetup_en.mustache | 50 + .../tpldata/ASUPtownconf.json} | 6 +- .../www/adminapx/static/tpldata/initconf.json | 31 + ...loginout_en.json => paganloginout_en.json} | 0 .../tpldata/{conf_en.json => setup_en.json} | 0 .../{ => static/tpldata}/tribeconf.json | 0 .../static => cdn/share}/js/axios.min.js | 0 .../static => cdn/share}/js/axios.min.map | 0 nationchains/www/cdn/share/js/dayjs.min.js | 1 + .../static => cdn/share}/js/editor.js | 0 .../static => cdn/share}/js/mustache.min.js | 0 nationchains/www/cdn/share/js/openpgp.js | 43562 ++++++++++++++++ .../static => cdn/share}/js/openpgp.min.js | 0 .../share}/js/openpgp.min.js.map | 0 nationchains/www/cdn/share/logo/favicon.png | Bin 0 -> 3785 bytes .../www/cdn/share/logo/logobgdark.png | Bin 0 -> 7188 bytes .../www/cdn/share/logo/logobgdark.svg | 168 + .../www/cdn/share/logo/logobglight.png | Bin 0 -> 10193 bytes .../www/cdn/share/logo/logobglight.svg | 156 + .../www/cdn/share/logo/logocarredark.png | Bin 0 -> 4606 bytes .../www/cdn/share/logo/logocarrelight.png | Bin 0 -> 4255 bytes nationchains/www/nginx_adminapx.conf | 36 +- tribes/devenv/person/itm/testerC.json | 1 + 85 files changed, 47890 insertions(+), 1042 deletions(-) create mode 100644 api/models/Notifications.js create mode 100644 api/models/Wwws.js create mode 100644 api/routes/wwws.js create mode 100644 electedtown.json rename nationchains/www/adminapx/{nginx => ASUPnginx}/nginx.conf.mustache (100%) rename nationchains/www/adminapx/{nginx/proxy_params.mustache => ASUPnginx/proxy_params copy.mustache} (100%) create mode 100644 nationchains/www/adminapx/ASUPnginx/proxy_params.mustache create mode 100644 nationchains/www/adminapx/static/img/logo/favicon.png create mode 100644 nationchains/www/adminapx/static/img/logo/logobgdark.png create mode 100644 nationchains/www/adminapx/static/img/logo/logobgdark.svg create mode 100644 nationchains/www/adminapx/static/img/logo/logobglight.png create mode 100644 nationchains/www/adminapx/static/img/logo/logobglight.svg create mode 100644 nationchains/www/adminapx/static/img/logo/logocarredark.png create mode 100644 nationchains/www/adminapx/static/img/logo/logocarrelight.png create mode 100644 nationchains/www/adminapx/static/js/apxpagans.js create mode 100644 nationchains/www/adminapx/static/js/apxtowns.js create mode 100644 nationchains/www/adminapx/static/js/apxtribes.js create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/OFL.txt create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/Quicksand-VariableFont_wght.ttf create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/Quicksand.zip create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/README.txt create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Bold.ttf create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Light.ttf create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Medium.ttf create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Regular.ttf create mode 100644 nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-SemiBold.ttf create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/OFL.txt create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/Questrial-Regular.ttf create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/Questrial.zip create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/generator_config.txt create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/questrial-regular-demo.html create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/questrial-regular-webfont.woff create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/questrial-regular-webfont.woff2 create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/specimen_files/grid_12-825-55-15.css create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/specimen_files/specimen_stylesheet.css create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/stylesheet.css create mode 100644 nationchains/www/adminapx/static/style/fonts/texte/webfontkit-20230503-055110.zip create mode 100644 nationchains/www/adminapx/static/tpl/apxmodal_en.mustache create mode 100644 nationchains/www/adminapx/static/tpl/articleeditor_en.mustache rename nationchains/www/adminapx/static/tpl/{listofarticle_en.mustache => articleslist_en.mustache} (100%) create mode 100644 nationchains/www/adminapx/static/tpl/emailregisterhtml_en.mustache create mode 100644 nationchains/www/adminapx/static/tpl/emailregistertxt_en.mustache create mode 100755 nationchains/www/adminapx/static/tpl/nginx.conf.mustache rename nationchains/www/adminapx/{nginx/modelwebsite.conf.mustache => static/tpl/nginxmodelwebsite.conf.mustache} (100%) create mode 100644 nationchains/www/adminapx/static/tpl/nginxproxyparams.mustache rename nationchains/www/adminapx/static/tpl/{loginout_en.mustache => paganloginout_en.mustache} (100%) create mode 100644 nationchains/www/adminapx/static/tpl/townsetup_en.mustache rename nationchains/www/adminapx/{townconf.json => static/tpldata/ASUPtownconf.json} (89%) create mode 100644 nationchains/www/adminapx/static/tpldata/initconf.json rename nationchains/www/adminapx/static/tpldata/{loginout_en.json => paganloginout_en.json} (100%) rename nationchains/www/adminapx/static/tpldata/{conf_en.json => setup_en.json} (100%) rename nationchains/www/adminapx/{ => static/tpldata}/tribeconf.json (100%) rename nationchains/www/{adminapx/static => cdn/share}/js/axios.min.js (100%) rename nationchains/www/{adminapx/static => cdn/share}/js/axios.min.map (100%) create mode 100644 nationchains/www/cdn/share/js/dayjs.min.js rename nationchains/www/{adminapx/static => cdn/share}/js/editor.js (100%) rename nationchains/www/{adminapx/static => cdn/share}/js/mustache.min.js (100%) create mode 100644 nationchains/www/cdn/share/js/openpgp.js rename nationchains/www/{adminapx/static => cdn/share}/js/openpgp.min.js (100%) rename nationchains/www/{adminapx/static => cdn/share}/js/openpgp.min.js.map (100%) create mode 100644 nationchains/www/cdn/share/logo/favicon.png create mode 100644 nationchains/www/cdn/share/logo/logobgdark.png create mode 100644 nationchains/www/cdn/share/logo/logobgdark.svg create mode 100644 nationchains/www/cdn/share/logo/logobglight.png create mode 100644 nationchains/www/cdn/share/logo/logobglight.svg create mode 100644 nationchains/www/cdn/share/logo/logocarredark.png create mode 100644 nationchains/www/cdn/share/logo/logocarrelight.png create mode 100644 tribes/devenv/person/itm/testerC.json diff --git a/.gitignore b/.gitignore index a4d52cd..08ae98b 100755 --- a/.gitignore +++ b/.gitignore @@ -5,9 +5,10 @@ /nationchains/blocks /nationchains/deals /nationchains/logs -/nationchains/towns +/nationchains/nations /nationchains/pagans -/nationchains/nations +/nationchains/towns /nationchains/tribes +/nationchains/www/nginx_adminapx.conf /nationchainssave /yarn* diff --git a/api/middlewares/checkHeaders.js b/api/middlewares/checkHeaders.js index 98a68a3..afff1d6 100755 --- a/api/middlewares/checkHeaders.js +++ b/api/middlewares/checkHeaders.js @@ -1,82 +1,95 @@ -const conf = require( '../../nationchains/tribes/conf.json' ); +const conf = require("../../nationchains/tribes/conf.json"); -const checkHeaders = ( req, res, next ) => { - /** - * @apiDefine apxHeader - * @apiGroup Middleware - * @apiDescription Header is mandatory to access apxtrib see tribes/townconf.json.exposedHeaders - * A turn around can be done with a simple get params has to be sent in the get url. Usefull to send simple get without header like ?xworkon=tribeName&xlang=en... priority is given to headers - * For performance, tokens are store globaly in req.app.locals.tokens={xpaganid:xauth} - * if xlang is not in conf.languagesAvailable - * - * @apiHeader {string} xjwt Pagans unique jwt token store in local town Pagans data or "noauth" - * @apiHeader {string} xpseudo Pagans unique Pagan id in uuid format or "nouuid" - * @apiHeader {string} xlang the 2 letter langage it request the api (if not exist the 2 first letter of Accept-Language header ) if lang does not exist in the town then en is set (as it always exist in en). - * @apiHeader {string} xtribe Tribes id where pseudo want to act - * @apiHeader {string} xapp Name of www/xapp folder that host app that send the request - * /tribeid/person/xpseudo.json have accessright on this app store in /tribe/tribeid/www/xapp - * - * @apiError missingexposedHeaders it miss an exposedHeaders - * - * @apiErrorExample {json} Error-Response: - * HTTP/1/1 404 Not Found - * { - * status:400, - * ref:"middleware" - * msg:"missingheaders", - * data: ["xpseudo","xjwt"] - * } - * - * @apiHeaderExample {json} Header-Exemple: - * { - * xtribe:"apache", - * xalias:"toto", - * xhash:"", - * xlang:"en", - * xapp:"popular" - * } - */ - req.session = {}; - const header = {}; - if (!req.header('xlang') && req.header('Content-Language')) req.params.xlang=req.header('Content-Language'); - let missingheader = []; - console.log('req.headers',req.headers) - for( const h of conf.api.exposedHeaders ) { - //console.log( h, req.header( h ) ) - if( req.params[ h ] ) { - header[ h ] = req.params[ h ] - } else if( req.header( h ) ) { - header[ h ] = req.header( h ) - } else { - missingheader.push(h); - } - }; - //console.log( 'header', header ) - // store in session the header information - req.session.header = header; - // Each header have to be declared - if( missingheader != "" ) { - // bad request - return res.status( 400 ) - .json( { - ref:"headers", - msg: "missingheader", - data: missingheader - } ); - }; - //console.log( req.app.locals.tribeids ) - if( !req.app.locals.tribeids.includes( header.xtribe ) ) { - return res.status( 400 ) - .json( { - ref:"headers", - msg: 'tribeiddoesnotexist', - moreinfo: header.xtribe - } ); - } - if( !conf.api.languages.includes( header.xlang ) ) { - console.log('warning language requested does not exist force to en glish') - header.xlang="en"; - } - next(); +const checkHeaders = (req, res, next) => { + /** + * @apiDefine apxHeader + * @apiGroup Middleware + * @apiDescription Header is mandatory to access apxtrib see tribes/townconf.json.exposedHeaders + * A turn around can be done with a simple get params has to be sent in the get url. Usefull to send simple get without header like ?xworkon=tribeName&xlang=en... priority is given to headers + * For performance, tokens are store globaly in req.app.locals.tokens={xpaganid:xauth} + * if xlang is not in conf.languagesAvailable + * + * @apiHeader {string} xjwt Pagans unique jwt token store in local town Pagans data or "noauth" + * @apiHeader {string} xpseudo Pagans unique Pagan id in uuid format or "nouuid" + * @apiHeader {string} xlang the 2 letter langage it request the api (if not exist the 2 first letter of Accept-Language header ) if lang does not exist in the town then en is set (as it always exist in en). + * @apiHeader {string} xtribe Tribes id where pseudo want to act + * @apiHeader {string} xapp Name of www/xapp folder that host app that send the request + * /tribeid/person/xpseudo.json have accessright on this app store in /tribe/tribeid/www/xapp + * + * @apiError missingexposedHeaders it miss an exposedHeaders + * + * @apiErrorExample {json} Error-Response: + * HTTP/1/1 400 Not Found + * { + * status:400, + * ref:"headers" + * msg:"missingheaders", + * data: ["headermissing1"] + * } + *@apiErrorExample {json} Error-Response: + * HTTP/1/1 404 Not Found + * { + * status:404, + * ref:"headers" + * msg:"tribeiddoesnotexist", + * data: {xalias} + * } + * + * @apiHeaderExample {json} Header-Exemple: + * { + * xtribe:"apache", + * xalias:"toto", + * xhash:"", + * xdays:"123" + * xlang:"en", + * xapp:"popular" + * } + */ + req.session = {}; + const header = {}; + if (!req.header("xlang") && req.header("Content-Language")) + req.params.xlang = req.header("Content-Language"); + let missingheader = []; + console.log("req.headers", req.headers); + for (const h of conf.api.exposedHeaders) { + //console.log( h, req.header( h ) ) + if (req.params[h]) { + header[h] = req.params[h]; + } else if (req.header(h)) { + header[h] = req.header(h); + } else { + missingheader.push(h); + } + } + //console.log( 'header', header ) + // store in session the header information + req.session.header = header; + // Each header have to be declared + if (missingheader != "") { + // bad request + return res.status(400).json({ + ref: "headers", + msg: "missingheader", + data: missingheader, + }); + } + //console.log( req.app.locals.tribeids ) + // xtribe == "town" is used during the setup process + if ( + !( + header.xtribe == "town" || req.app.locals.tribeids.includes(header.xtribe) + ) + ) { + return res.status(404).json({ + ref: "headers", + msg: "tribeiddoesnotexist", + data: { xtribe: header.xtribe }, + }); + } + if (!conf.api.languages.includes(header.xlang)) { + console.log("warning language requested does not exist force to english"); + header.xlang = "en"; + } + next(); }; module.exports = checkHeaders; diff --git a/api/middlewares/hasAccessrighton.js b/api/middlewares/hasAccessrighton.js index b383446..6b462b6 100755 --- a/api/middlewares/hasAccessrighton.js +++ b/api/middlewares/hasAccessrighton.js @@ -1,42 +1,69 @@ -const fs = require( 'fs-extra' ); -const glob = require( 'glob' ); -const path = require( 'path' ); +const fs = require("fs-extra"); +const glob = require("glob"); +const path = require("path"); -const config = require( '../../nationchains/tribes/conf.json' ); +const config = require("../../nationchains/tribes/conf.json"); -const hasAccessrighton = ( object, action, ownby ) => { - /* +const hasAccessrighton = (object, action, ownby) => { + /* @action (mandatory) : CRUDO @object (mandatory)= name of a folder object in /tribeid space can be a tree for example objects/items @ownby (option) = list des uuid propriétaire return next() if all action exist in req.app.local.tokens[UUID].ACCESSRIGHTS.data[object] OR if last action ="O" and uuid exist in ownBy Careffull if you have many action CRO let O at the end this will force req.right at true if the owner try an action on this object - */ - return ( req, res, next ) => { - //console.log( 'err.stack hasAccessrights', err.statck ) - //console.log( `test accessright on object:${object} for ${req.session.header.xworkon}:`, req.app.locals.tokens[ req.session.header.xpaganid ].ACCESSRIGHTS.data[ req.session.header.xworkon ] ) - req.right = false; - if( req.app.locals.tokens[ req.session.header.xpaganid ].ACCESSRIGHTS.data[ req.session.header.xworkon ] && req.app.locals.tokens[ req.session.header.xpaganid ].ACCESSRIGHTS.data[ req.session.header.xworkon ][ object ] ) { - req.right = true; - [ ...action ].forEach( a => { - if( a == "O" && ownby && ownby.includes( req.session.header.xpaganid ) ) { - req.right = true; - } else { - req.right = req.right && req.app.locals.tokens[ req.session.header.xpaganid ].ACCESSRIGHTS.data[ req.session.header.xworkon ][ object ].includes( a ) - } - } ) - } - //console.log( 'Access data autorise? ', req.right ) - if( !req.right ) { - return res.status( 403 ) - .json( { - info:'forbiddenAccessright', - ref: 'headers', - moreinfo: {xpaganid:req.session.header.xpaganid,object:object, xworkon:req.session.header.xworkon, action:action} - } ) - } - next(); - } -} + + need to check first a person exist with this alias in tribe + + const person = fs.readJsonSync( + `${conf.dirname}/nationchains/tribes/${req.session.header.xtribe}/persons/${req.session.header.xalias}.json` + ); + + + + + + */ + + return (req, res, next) => { + //console.log( 'err.stack hasAccessrights', err.statck ) + //console.log( `test accessright on object:${object} for ${req.session.header.xworkon}:`, req.app.locals.tokens[ req.session.header.xpaganid ].ACCESSRIGHTS.data[ req.session.header.xworkon ] ) + req.right = false; + if ( + req.app.locals.tokens[req.session.header.xpaganid].ACCESSRIGHTS.data[ + req.session.header.xworkon + ] && + req.app.locals.tokens[req.session.header.xpaganid].ACCESSRIGHTS.data[ + req.session.header.xworkon + ][object] + ) { + req.right = true; + [...action].forEach((a) => { + if (a == "O" && ownby && ownby.includes(req.session.header.xpaganid)) { + req.right = true; + } else { + req.right = + req.right && + req.app.locals.tokens[ + req.session.header.xpaganid + ].ACCESSRIGHTS.data[req.session.header.xworkon][object].includes(a); + } + }); + } + //console.log( 'Access data autorise? ', req.right ) + if (!req.right) { + return res.status(403).json({ + info: "forbiddenAccessright", + ref: "headers", + moreinfo: { + xpaganid: req.session.header.xpaganid, + object: object, + xworkon: req.session.header.xworkon, + action: action, + }, + }); + } + next(); + }; +}; module.exports = hasAccessrighton; diff --git a/api/middlewares/isAuthenticated.js b/api/middlewares/isAuthenticated.js index 6131c5b..ec68e7b 100755 --- a/api/middlewares/isAuthenticated.js +++ b/api/middlewares/isAuthenticated.js @@ -1,207 +1,106 @@ -const jwt = require("jwt-simple"); const fs = require("fs-extra"); -const moment = require("moment"); const dayjs = require("dayjs"); const glob = require("glob"); +const openpgp = require("openpgp"); const conf = require("../../nationchains/tribes/conf.json"); -const isAuthenticated = (req, res, next) => { - //once a day rm oldest tokens than 24hours +const isAuthenticated = async (req, res, next) => { + // once a day rm oldest tokens than 24hours tag job by adding tmp/tokensmenagedone{day} const currentday = dayjs().date(); - console.log("dayjs", currentday); console.log( - "test si menagedone" + currentday, - !fs.existsSync(`${conf.dirname}/tmp/tokensmenagedone${currentday}`) + "if menagedone" + currentday, + !fs.existsSync(`${__base}tmp/tokensmenagedone${currentday}`) ); - if (!fs.existsSync(`${conf.dirname}/tmp/tokensmenagedone${currentday}`)) { + if (!fs.existsSync(`${__base}/tmp/tokens`)) + fs.mkdirSync(`${__base}tmp/tokens`); + if (!fs.existsSync(`${__base}tmp/tokensmenagedone${currentday}`)) { // clean oldest - const tsday = dayjs().date(); - console.log("tsday", tsday); - glob.sync(`${conf.dirname}/tmp/tokensmenagedone*`).forEach((f) => { + const tsday = dayjs().valueOf(); // now in timestamp format + glob.sync(`${__base}tmp/tokensmenagedone*`).forEach((f) => { fs.removeSync(f); }); - glob.sync(`${conf.dirname}/tmp/tokens/*.json`).forEach((f) => { - fs.readJson(f, (err, data) => { - if (!err && tsday - data.timestamp > 86400000) fs.remove(f); - }); + glob.sync(`${__base}tmp/tokens/*.json`).forEach((f) => { + if (tsday - parseInt(f.split("_")[1]) > 86400000) fs.remove(f); }); } //Check register in tmp/tokens/ - console.log("isRegister?"); + console.log("isAuthenticate?"); const resnotauth = { ref: "headers", msg: "notauthenticated", data: { xalias: req.session.header.xalias, - xtribe: req.session.header.xtribe, + xaliasexists: true, }, }; - console.lolg(req.session.header) - if (req.session.header.xalias == "anonymous") res.status(401).json(resnotauth); - - const tmpfs = `${conf.dirname}/tmp/tokens/${req.session.header.xtribe}_${req.session.header.xalias}_${req.session.header.hash}.json`; - if (!fs.exists(tmpfs)) { - //check if pseudo exist as a pagan in pagans/ and as a person in xtribe/persons/ and check hash is coming from publickey - if ( - !fs.existsSync( - `${conf.dirname}/nationchains/tribes/${req.session.header.xtribe}/persons/${req.session.header.xalias}.json` - ) - ) { - console.log( - `pseudo:${req.session.header.xalias} does not exist for xtribe ${req.session.header.xtribe}` - ); - res.status(401).json(resnotauth); - } - if ( - !fs.existsSync( - `${conf.dirname}/nationchains/pagans/${req.session.header.xalias}.json` - ) - ) { - console.log( - `pseudo:${req.session.header.xalias} does not exist as a pagan` - ); - res.status(401).json(resnotauth); - } - const person = fs.readJsonSync( - `${conf.dirname}/nationchains/tribes/${req.session.header.xtribe}/persons/${req.session.header.xalias}.json` - ); - const pagan = fs.readJsonSync( - `${conf.dirname}/nationchains/pagans/${req.session.header.xalias}.json` - ); - //check hash with publickey pagan.publickey - // if good => create a /tmp/tokens/xtribe_xalias_xhash.json ={timestamp} - // if not good res.json(resnotauth) + console.log(req.session.header); + if (req.session.header.xalias == "anonymous") { + console.log("alias anonymous means not auth"); + return res.status(401).json(resnotauth); } + + const tmpfs = `${__base}tmp/tokens/${req.session.header.xalias}_${ + req.session.header.xdays + }_${req.session.header.xhash.substring(20, 200)}`; + console.log(tmpfs); + if (!fs.existsSync(tmpfs)) { + // need to check detached sign + let publickey; + if ( + fs.existsSync( + `${__base}nationchains/pagans/itm/${req.session.header.xalias}.json` + ) + ) { + const pagan = fs.readJsonSync( + `${__base}nationchains/pagans/itm/${req.session.header.xalias}.json` + ); + publickey = pagan.publicKey; + } else { + resnotauth.data.xaliasexists = false; + if (req.body.publickey) { + publickey = req.body.publickey; + } else { + console.log("alias unknown"); + return res.status(404).send(resnotauth); + } + } + console.log(publickey); + console.log(Buffer.from(req.session.header.xhash, "base64").toString()); + const publicKey = await openpgp.readKey({ armoredKey: publickey }); + const msg = await openpgp.createMessage({ + text: `${req.session.header.xalias}_${req.session.header.xdays}`, + }); + const signature = await openpgp.readSignature({ + armoredSignature: Buffer.from( + req.session.header.xhash, + "base64" + ).toString(), + }); + console.log(msg); + console.log(signature); + console.log(publicKey); + const checkauth = await openpgp.verify({ + message: msg, + signature: signature, + verificationKeys: publicKey, + }); + console.log(checkauth); + console.log(checkauth.signatures[0].keyID); + //console.log(await checkauth.signatures[0].signature); + //console.log(await checkauth.signatures[0].verified); + + const { check, keyID } = checkauth.signatures[0]; + try { + await check; // raise an error if necessary + fs.outputFileSync(tmpfs, req.session.header.xhash, "utf8"); + } catch (e) { + resnotauth.msg = "signaturefailed"; + console.log("not auth fail sign"); + return res.status(401).send(resnotauth); + } + } + console.log("Authenticated"); next(); }; -const isAuthenticatedold = (req, res, next) => { - /* - check if authenticated with valid token - if not => set req.session.header.xjwt=1 - if yes => set for xWorkon - req.session.header.accessrights={ - app:{'tribeid:website':[liste of menu]}, - data:{ "sitewebsrc": "RWCDO", - "contacts": "RWCDO"}} - Liste of menu is linked with the app tht h - ave to be consistent with accessrights.data - data, list of object accessright Read Write Create Delete Owner - a xuuid can read any objet if R - if O wner means that it can only read write its object create by himself - */ - console.log("isAuthenticated()?"); - //console.log( 'req.app.locals.tokens', req.app.locals.tokens ) - //console.log( 'req.session.header', req.session.header ); - // Check if token exist or not - req.session.header.accessrights = { app: "", data: {} }; - if ( - req.session.header.xalias == "1" || - !req.app.locals.tokens[req.session.header.xalias] - ) { - console.log( - `isAuthenticated no : uuid=1 (value=${req.session.header.xalias}) or locals.tokens[uuid] empty ` - ); - console.log( - "req.app.locals.tokens de xalias", - req.app.locals.tokens[req.session.header.xalias] - ); - console.log( - "list key uuid de req.app.locals.tokens", - Object.keys(req.app.locals.tokens) - ); - req.session.header.xjwt = "1"; - } else if ( - req.app.locals.tokens[req.session.header.xalias].TOKEN !== - req.session.header.xjwt - ) { - // console.log(req.session.header.xuuid); - // console.log(req.session.header.xjwt); - // update tokens from file in case recently logged - try { - console.log( - "token not in list of token (req.app.locals.tokens) try to refresh from file" - ); - req.app.locals.tokens = fs.readJsonSync(`${conf.tmp}/tokens.json`); - } catch (err) { - console.log( - `check isAuthenticated issue in reading ${conf.tmp}/tokens.json` - ); - } - if ( - req.app.locals.tokens[req.session.header.xalias].TOKEN !== - req.session.header.xjwt - ) { - // if still does not exist then out - console.log("isAuthenticated no, token outdated"); - req.session.header.xjwt = "1"; - req.session.header.xalias = "1"; - } - } - if (req.session.header.xjwt == "1") { - //return res.status( 403 ) - return res.status(403).json({ - info: ["forbiddenAccess"], - model: "Pagans", - moreinfo: "isAuthenticated faill", - }); - } else { - console.log("isAuthenticated yes"); - if (req.app.locals.tokens[req.session.header.xalias]) { - //console.log( `accessright pour ${req.session.header.xalias}`, req.app.locals.tokens[ req.session.header.xalias ].ACCESSRIGHTS ); - //set header.accessrights from tokens.json - req.session.header.accessrights = - req.app.locals.tokens[req.session.header.xalias].ACCESSRIGHTS; - } else { - // case of bypass no accessright available - req.session.header.accessrights = {}; - } - // Once per day, clean old token - const currentday = moment().date(); - console.log( - "test si menagedone" + currentday, - !fs.existsSync(`${conf.tmp}/menagedone${currentday}`) - ); - if (!fs.existsSync(`${conf.tmp}/menagedone${currentday}`)) { - glob.sync(`${conf.tmp}/menagedone*`).forEach((f) => { - fs.remove(f, (err) => { - if (err) { - console.log("err remove menagedone", err); - } - }); - }); - glob.sync(`${conf.tmp}/mdcreator*.log`).forEach((f) => { - fs.remove(f, (err) => { - if (err) { - console.log("err remove mdcreator log", err); - } - }); - }); - const newtokens = {}; - for (const k of Object.keys(req.app.locals.tokens)) { - try { - const decodedToken = jwt.decode( - req.app.locals.tokens[k].TOKEN, - conf.jwtSecret - ); - //console.log( moment( decodedToken.expiration ), moment() ) - //console.log( moment( decodedToken.expiration ) >= moment() ) - if (moment(decodedToken.expiration) >= moment()) { - newtokens[k] = req.app.locals.tokens[k]; - } - } catch (err) { - console.log("Check isAuthenticated cleaning token ", err); - } - } - req.app.locals.tokens = newtokens; - fs.outputJsonSync(`${conf.tmp}/tokens.json`, newtokens); - fs.writeFileSync( - `${conf.tmp}/menagedone${currentday}`, - "fichier semaphore to clean data each day can be deleted with no consequence", - "utf-8" - ); - } - next(); - } -}; module.exports = isAuthenticated; diff --git a/api/middlewares/lg/headers_en.json b/api/middlewares/lg/headers_en.json index 320dcbf..c6ea8ff 100644 --- a/api/middlewares/lg/headers_en.json +++ b/api/middlewares/lg/headers_en.json @@ -1,7 +1,7 @@ { - "missingheader":"Some header miss to have a valid request: {{#data}} {{.}} {{/data}}", - "tribeiddoesnotexist":"Header xtribe: {{data}} does not exist in this town", - "authenticated":"Your perso{{{xpseudo}}} is register for tribe {{{xtribe}}}", - "notauthenticated":"Your pseudo {{xpseudo}} are not register into tribe {{xtribe}} ", - "forbiddenAccessright":"Pagan {{data.xpseudo}} has not access right to act {{data.action}} onto object {{data.object}} for tribe {{mor.xworkon}}" -} \ No newline at end of file + "missingheader": "Some header miss to have a valid request: {{#data}} {{.}} {{/data}}", + "tribeiddoesnotexist": "Header xtribe: {{data.xtribe}} does not exist in this town", + "authenticated": "Your alias{{{data.xalias}}} is authenticated", + "notauthenticated": "Your alias: {{data.xalias}} is not authenticated {{^data.aliasexists}} and this alias does not exist !{{/data.aliasexists}}", + "forbiddenAccessright": "Pagan {{data.xalias}} has not access right to act {{data.action}} onto object {{data.object}} for tribe {{mor.xworkon}}" +} diff --git a/api/models/Notifications.js b/api/models/Notifications.js new file mode 100644 index 0000000..b9ad50c --- /dev/null +++ b/api/models/Notifications.js @@ -0,0 +1,20 @@ +const glob = require("glob"); +const path = require("path"); +const fs = require("fs-extra"); + +/** + * To manage any communication between Pagan + * mayor druid emailing/sms/paper from tribe register smtp, simcard, mail api to Person(s) / Pagan(s) + * volatile notification message from tribe activities to Pagans / person () + * + */ + +const Notifications = {}; + +Notifications.send = (data) => { + const ret = {}; + console.log("TODO dev notification emailing"); + return ret; +}; + +module.exports = Notifications; diff --git a/api/models/Pagans.js b/api/models/Pagans.js index e535583..d79357f 100644 --- a/api/models/Pagans.js +++ b/api/models/Pagans.js @@ -1,97 +1,125 @@ const glob = require("glob"); const path = require("path"); +const dayjs = require("dayjs"); const fs = require("fs-extra"); -const axios = require('axios'); -const openpgp = require('openpgp'); -const conf=require('../../nationchains/tribes/conf.json') - +const axios = require("axios"); +const openpgp = require("openpgp"); +var conf = {}; +if (fs.existsSync("../../nationchains/tribes/conf.json")) { + conf = require("../../nationchains/tribes/conf.json"); +} +console.log(conf); /** - * Pagan Management numeric Identity - * - * - * + * Pagan Management numeric Identity and Person (Person = Pagan Id + tribe) + * + * + * */ -const Pagans= {} +const Pagans = {}; -Pagans.createId = async (alias,passphrase) =>{ - /** - * @param {string} alias a unique alias that identify an identity - * @param {string} passphrase a string to cipher the publicKey (can be empty, less secure but simpler) - * @return {publicKey,privateKey} with userIds = [{alias}] - */ - let apxpagans={}; - if (fs.existsSync(`${conf.dirname}/nationchains/pagans/idx/alias_all.json`)){ - apxpagans = fs.readJsonSync( - `${conf.dirname}/nationchains/pagans/idx/alias_all.json` - ); - } - if (Object.keys(apxpagans).includes(alias)){ - return {status:409,ref:"pagans",msg:"aliasalreadyexist"} - }; - const {privateKey,publicKey} = await openpgp.generateKey({ - type: "ecc", // Type of the key, defaults to ECC - curve: "curve25519", // ECC curve name, defaults to curve25519 - userIDs: [{ alias: alias }], // you can pass multiple user IDs - passphrase: passphrase, // protects the private key - format: "armored", // output key format, defaults to 'armored' (other options: 'binary' or 'object') - }); - apxpagans[alias]={alias,publicKey}; - fs.outputJsonSync(`${conf.dirname}/nationchains/pagans/idx/alias_all.json`,apxpagans); - fs.outputJsonSync(`${conf.dirname}/nationchains/pagans/itm/${alias}.json`,{alias,publicKey}); - return {status:200, data:{alias,privateKey,publicKey}} -} - - //console.log( Pagans.generateKey('toto','')) - Pagans.detachedSignature = async (pubK, privK, passphrase, message) => { - /** - * @pubK {string} a text public key - * @privK {string} a test priv key - * @passphrase {string} used to read privK - * @message {string} message to sign - * @Return a detached Signature of the message - */ - const publicKey = await openpgp.readKey({ armoredKey: pubK }); - const privateKey = await openpgp.decryptKey({ - privateKey: await openpgp.readPrivateKey({ armoredKey: privK }), - passphrase, - }); - const msg = await openpgp.createMessage({ text: message }); - return await openpgp.sign({ msg, signinKeys: privK, detached: true }); - }; - Pagans.checkdetachedSignature = async ( +Pagans.create = (alias, publicKey) => { + /** + * @param {string} alias a unique alias that identify an identity + * @param {string} publicKey a publicKey + * @return {object} { status: 200, data: { alias, publicKey } } + * xhash was checked by isauthenticated + * @todo use Odmdb to add a pagan + */ + let apxpagans = {}; + if (fs.existsSync(`${__base}nationchains/pagans/idx/alias_all.json`)) { + apxpagans = fs.readJsonSync( + `${__base}nationchains/pagans/idx/alias_all.json` + ); + } + apxpagans[alias] = { alias, publicKey }; + fs.outputJsonSync( + `${__base}nationchains/pagans/idx/alias_all.json`, + apxpagans + ); + fs.outputJsonSync(`${__base}nationchains/pagans/itm/${alias}.json`, { alias, - pubK, - detachedSignature, - message - ) => { - /** - * @alias {string} alias link to the publicKey - * @pubK {string} publiKey text format - * @detachedSignature {string} a detachedsignatured get from apx.detachedSignature - * @message {string} the message signed - * @return {boolean} true the message was signed by alias - * false the message was not signed by alias - */ - const publicKey = await openpgp.readKey({ armoredKey: pubK }); - const msg = await openpgp.createMessage({ text: message }); - const signature = await openpgp.readSignature({ - armoredSignature: detachedSignature, // parse detached signature - }); - const verificationResult = await openpgp.verify({ - msg, // Message object - signature, - verificationKeys: publicKey - }); - const { verified, keyID } = verificationResult.signatures[0]; - try { - await verified; // throws on invalid signature - console.log("Signed by key id " + keyID.toHex()); - return KeyId.toHex().alias == alias; - } catch (e) { - console.log("Signature could not be verified: " + e.message); - return false; - } + publicKey, + }); + return { status: 200, data: { alias, publicKey } }; +}; + +Pagans.personupdate = (alias, tribe, persondata) => { + //later use Odmdb ans schema person to manage this + /** + * @Param {string} alias pagan unique id + * @Param {string} tribe tribe id in this town + * @Param {object} persondata that respect /nationchains/schema/person.json + nationchains/tribe/tribeid/schema/personextented.json + * @return create or update a person /tribe/tribeid/person/alias.json + */ + let person = { + alias: alias, + dt_create: dayjs(), + accessrights: { profil: "user" }, }; - -module.exports=Pagans; \ No newline at end of file + if (fs.existsSync(`${__base}tribes/${tribe}/person/itm/${alias}.json`)) { + person = fs.readJsonSync( + `${__base}tribes/${tribe}/person/itm/${alias}.json` + ); + person.dt_update = dayjs(); + } + Object.keys(persondata).forEach((d) => { + person[d] = persondata[d]; + }); + //const checkjson= Checkjson.schema.data = (fs.readJsonSync(`${__base}}nationchains/schema/person.json`, person, false) + // if checkjson.status==200 create /update with odmdb to update index data + // see odmdb that did all and return standard message + fs.outputJSONSync( + `${__base}tribes/${tribe}/person/itm/${alias}.json`, + person, + { + space: 2, + } + ); + return { + status: 200, + ref: "Pagans", + msg: "successfullupdate", + data: { tribe: tribe }, + }; +}; + +Pagans.authenticatedetachedSignature = async ( + alias, + pubK, + detachedSignature, + message +) => { + /** + * Check that a message was signed with a privateKey from a publicKey + * This is not necessary if isAuthenticated, but can be usefull to double check + * @TODO finish it and implement it also in /apxpagan.js for browser + * @alias {string} alias link to the publicKey + * @pubK {string} publiKey text format + * @detachedSignature {string} a detachedsignatured get from apx.detachedSignature + * @message {string} the message signed + * @return {boolean} true the message was signed by alias + * false the message was not signed by alias + */ + const publicKey = await openpgp.readKey({ armoredKey: pubK }); + const msg = await openpgp.createMessage({ text: message }); + const signature = await openpgp.readSignature({ + armoredSignature: detachedSignature, // parse detached signature + }); + const verificationResult = await openpgp.verify({ + msg, // Message object + signature, + verificationKeys: publicKey, + }); + const { verified, keyID } = verificationResult.signatures[0]; + try { + await verified; // throws on invalid signature + console.log("Signed by key id " + keyID.toHex()); + return KeyId.toHex().alias == alias; + } catch (e) { + console.log("Signature could not be verified: " + e.message); + return false; + } +}; + +module.exports = Pagans; diff --git a/api/models/Setup.js b/api/models/Setup.js index bf0e812..a865466 100644 --- a/api/models/Setup.js +++ b/api/models/Setup.js @@ -4,6 +4,7 @@ const dnsSync = require("dns-sync"); const mustache = require("mustache"); const readlineSync = require("readline-sync"); +const Wwws = require("../models/Wwws.js"); /** * This Setup is run at the first installation * This is not an exportable module @@ -28,9 +29,9 @@ Setup.check = () => { ); process.exit(); } - if (fs.existsSync("./nationchains/tribes/conf.json")) { + if (fs.existsSync("./nationchains/www/nginx_adminapx.conf")) { console.log( - "\x1b[31m Be carefull you already have a town set in ./nationchains/tribes/index.conf.json, check and remove it if you want to setup this town." + "\x1b[31m Be carefull you already have a town set, check http://adminapx or remove ./nationchains/www/nginx_adminapx.conf to reset a sync with the last nationchains" ); process.exit(); } @@ -38,19 +39,51 @@ Setup.check = () => { }; Setup.init = async () => { + /** + * create empty nationchains + * rsync all subfolder nationchains except the tribes/ and /www/nginx_adminapx.conf + * + * Then to send new version we fix a master production + * + */ + const initconf = fs.readJSONSync( + "./nationchains/www/adminapx/static/tpldata/initconf.json" + ); + initconf.sudoerUser = process.env.USER; + initconf.dirname = path.resolve(`${__dirname}/../../`); + // To allow to serve the nation website until the end + initconf.nginx.include.push( + `${townconf.dirname}/nationchains/www/nginx_*.conf` + ); + // To allow to serve tribes web site + initconf.nginx.include.push( + `${townconf.dirname}/nationchains/tribes/*/www/nginx_*.conf` + ); + initconf.nginx.logs = `${townconf.dirname}/nationchains/logs/nginx`; + initconf.nginx.website = "adminapx"; + initconf.nginx.fswww = "nationchains/"; //for a local tribe nationchains/tribes/tribeid + initconf.nginx.tribeid = "town"; + initconf.nginx.pageindex = "index_en.html"; + const nginxconf = Wwws.apxtribinstall(initconf); + if (nginxconf.status == 200) { + } +}; +if (Setup.check()) Setup.init(); + +// After testing remove all stuff after this line + +Setup.initold = async () => { // Get standard conf and current data const townconf = fs.readJsonSync("./nationchains/www/adminapx/townconf.json"); const apxnations = fs.readJsonSync( `./nationchains/nations/idx/nationId_all.json` ); const apxtowns = fs.readJsonSync(`./nationchains/towns/idx/townId_all.json`); - - let apxpagans={} - if (fs.existsSync(`./nationchains/pagans/idx/alias_all.json`)){ - apxpagans = fs.readJsonSync( - `./nationchains/pagans/idx/alias_all.json` - ); - } + + let apxpagans = {}; + if (fs.existsSync(`./nationchains/pagans/idx/alias_all.json`)) { + apxpagans = fs.readJsonSync(`./nationchains/pagans/idx/alias_all.json`); + } if (!Object.keys(apxnations).includes(townconf.nationId)) { console.log( @@ -95,7 +128,7 @@ Setup.init = async () => { ) ) process.exit(); - + // saved and change nginx conf if (!fs.existsSync("/etc/nginx/nginxconf.saved")) { fs.moveSync("/etc/nginx/nginx.conf", "/etc/nginx/nginxconf.saved"); @@ -112,6 +145,16 @@ Setup.init = async () => { mustache.render(tplnginxconf, townconf), "utf8" ); + //proxyparam + const proxy_params = fs.readFileSync( + "./nationchains/www/adminapx/nginx/proxy_params.mustache", + "utf8" + ); + fs.outputFileSync( + "/etc/nginx/proxy_params", + mustache.render(proxy_params, townconf), + "utf8" + ); const tplnginxwww = fs.readFileSync( "./nationchains/www/adminapx/nginx/modelwebsite.conf.mustache", "utf8" @@ -124,8 +167,10 @@ Setup.init = async () => { fs.outputJsonSync("./nationchains/tribes/conf.json", townconf, { spaces: 2, }); + // Integrer cette partie du setup en inteactif. + // l'objectif du setup est de rendere accessible adminapx en local (IP local) ou production IP public //CREATE A TOWN setup local voir utiliser towns.create - townconf.town = { + /* townconf.town = { townId: townconf.townId, nationId: townconf.nationId, url: `http://${townconf.dns[0]}`, @@ -133,9 +178,13 @@ Setup.init = async () => { mayorid: townconf.mayorId, status: "unchain", }; - apxtowns[townconf.townId]=townconf.town; - fs.outputJsonSync(`./nationchains/towns/idx/townId_all.json`,apxtowns); - fs.outputJsonSync(`./nationchains/towns/itm/${townconf.townId}.json`,townconf.town,{spaces:2}); + apxtowns[townconf.townId] = townconf.town; + fs.outputJsonSync(`./nationchains/towns/idx/townId_all.json`, apxtowns); + fs.outputJsonSync( + `./nationchains/towns/itm/${townconf.townId}.json`, + townconf.town, + { spaces: 2 } + ); // Create tribe id voir a utiliser tribes.create() townconf.tribe = { tribeId: townconf.tribeId, @@ -145,27 +194,47 @@ Setup.init = async () => { townId: townconf.townId, }; //tribe does not exist in a new town - apxtribes={} - apxtribes[townconf.tribeId]=townconf.tribe; - fs.outputJsonSync(`./nationchains/tribes/idx/tribeId_all.json`,apxtribes); - fs.outputJsonSync(`./nationchains/tribes/itm/${townconf.tribeId}.json`,townconf.tribe,{spaces:2}); + apxtribes = {}; + apxtribes[townconf.tribeId] = townconf.tribe; + fs.outputJsonSync(`./nationchains/tribes/idx/tribeId_all.json`, apxtribes); + fs.outputJsonSync( + `./nationchains/tribes/itm/${townconf.tribeId}.json`, + townconf.tribe, + { spaces: 2 } + ); fs.ensureDirSync(`./nationchains/tribes/${townconf.tribeId}/logs/nginx`); //CREATE a mayorId pagans if it does not exist - if (!apxpagans[townconf.mayorId]){ - const Pagans=require('./Pagans'); - const createPagans=await Pagans.createId(townconf.mayorId,townconf.passphrase); - if (createPagans.status==200){ - fs.outputFileSync(`./${townconf.mayorId}_PrivateKey.txt`,createPagans.data.privateKey,"utf8"); - fs.outputFileSync(`./${townconf.mayorId}_PublicKey.txt`,createPagans.data.publicKey,"utf8"); - console.log(`\x1b[43mCut paste your keys /${townconf.mayorId}_PrivateKey.txt /${townconf.mayorId}_PublicKey.txt \x1b[0m`) - }else{ - console.log('Error at Pagan creation '); + if (!apxpagans[townconf.mayorId]) { + const Pagans = require("./Pagans"); + const createPagans = await Pagans.createId( + townconf.mayorId, + townconf.passphrase + ); + if (createPagans.status == 200) { + fs.outputFileSync( + `./${townconf.mayorId}_PrivateKey.txt`, + createPagans.data.privateKey, + "utf8" + ); + fs.outputFileSync( + `./${townconf.mayorId}_PublicKey.txt`, + createPagans.data.publicKey, + "utf8" + ); + console.log( + `\x1b[43mCut paste your keys /${townconf.mayorId}_PrivateKey.txt /${townconf.mayorId}_PublicKey.txt \x1b[0m` + ); + } else { + console.log("Error at Pagan creation "); console.log(createPagans); process.exit(); } - } - //restart nginx + }*/ + + //fin de partie à integer dans l'interface graphique adminapx + + //restart nginx const { exec } = require("child_process"); exec(townconf.nginx.restart, (error, stdout, stderr) => { if (error) { diff --git a/api/models/Wwws.js b/api/models/Wwws.js new file mode 100644 index 0000000..f61978b --- /dev/null +++ b/api/models/Wwws.js @@ -0,0 +1,118 @@ +const fs = require("fs-extra"); +const path = require("path"); +const dnsSync = require("dns-sync"); +const mustache = require("mustache"); +const readlineSync = require("readline-sync"); + +const conf = fs.existsSync("../../nationchains/tribes/conf.json") + ? require("../../nationchains/tribes/conf.json") + : {}; +const Wwws = {}; + +Wwws.apxtribinstall = (paramconf) => { + /** + * First install for a setup + * + */ + if (fs.existsSync("../../nationchains/www/nginx_adminapx.conf")) { + console.log("You already have a conf on this town"); + process.exit(); + } + //first install + const nginxconf = fs.readFileSync( + "../../nationchains/www/adminapx/static/tpl/nginx.conf.mustache", + "utf8" + ); + const proxyparams = fs.readFileSync( + "../../nationchains/www/adminapx/static/tpl/nginxproxy_params.mustache", + "utf8" + ); + // saved and change nginx conf + if (!fs.existsSync("/etc/nginx/nginxconf.saved")) { + fs.moveSync("/etc/nginx/nginx.conf", "/etc/nginx/nginxconf.saved"); + console.log( + "your previous /etc/nginx/nginx.conf was backup in /etc/nginx/nginxconf.saved" + ); + } + fs.outputFileSync( + "/etc/nginx/nginx.conf", + mustache.render(nginxconf, paramconf), + "utf8" + ); + fs.outputFileSync( + "/etc/nginx/proxy_params", + mustache.render(proxyparams, paramconf), + "utf8" + ); + if (!fs.existsSync(paramconf.nginx.logs)) fs.mkdirSync(paramconf.nginx.logs); + paramconf.nginx.firstinstall = true; + fs.outputJsonSync("../../nationchains/tribes/conf.json", paramconf, { + space: 2, + }); + + return Www.create(paramconf.nginx); +}; + +Wwws.create = (paramnginx) => { + /** + * Create an nginx conf to make available https://adminapx on a local network + * paramconf nginx.fswww place where the www folder is /tribeId/ + */ + const res = { + status: 200, + ref: "Www", + msg: "successfulwww", + data: { website: paramnginx.website }, + }; + const nginxwebsite = fs.readFileSync( + "../../nationchains/www/adminapx/static/tpl/nginxmodelwebsite.conf.mustache", + "utf8" + ); + fs.outputFileSync( + `./${paramnginx.fswww}www/nginx_${paramnginx.website}.conf`, + mustache.render(nginxwebsite, paramnginx), + "utf8" + ); + if (!fs.existsSync(`./${paramnginx.fswww}www/${paramnginx.website}`)) { + //See later how to generate specific template of webapp + fs.mkdirSync(`./${paramnginx.fswww}www/${paramnginx.website}`); + } + if (!fs.existsSync(`./${paramnginx.fswww}www/cdn`)) { + //See later how to generate specific template of webapp + fs.mkdirSync(`./${paramnginx.fswww}www/cdn`); + } + //restart nginx + const { exec } = require("child_process"); + exec(paramnginx.restart, (error, stdout, stderr) => { + if (error) { + if (paramnginx.firstinstall) { + console.log("\x1b[42m", error, stdout, stderr, "x1b[0m"); + } + //@todo supprimer la derniere config et relancer + res.status = 500; + res.msg = "nginxError"; + res.data = { msg: `${error}
${stdout}
${stderr}` }; + } else { + if (paramnginx.firstinstall) { + // the tribes/conf.json is saved in apxtribinstall + console.log( + `\x1b[42m###########################################################################################\x1b[0m\n\x1b[42mWellcome into apxtrib, you can now 'yarn dev' for dev or 'yarn startpm2' for prod or \n'yarn unittest' for testing purpose. Access to your town here \x1b[0m\x1b[32mhttp://adminapx\x1b[0m \x1b[42m \nto finish your town setup. Don't forget to set your localhost /etc/hosts by adding 127.0.0.1 adminapx or {LAN IP} adminapx . Check README's project to learn more. \x1b[0m\n\x1b[42m###########################################################################################\x1b[0m` + ); + } else { + // add website to tribe conf + } + } + }); + return res; +}; +Wwws.setssl = () => { + // Run process to change nginx conf to get a ssl +}; + +Wwws.configlist = (tribeId) => { + //if accessright R return list of conf parameter {webapp:{conf parameter}} + const res = { status: 200, data: {} }; + return res; +}; + +module.exports = Wwws; diff --git a/api/models/lg/Pagans_en.json b/api/models/lg/Pagans_en.json index 0e0dcd2..e848f10 100644 --- a/api/models/lg/Pagans_en.json +++ b/api/models/lg/Pagans_en.json @@ -1,3 +1,5 @@ { - -} \ No newline at end of file + "successfullcreate": "Alias creation for {{alias}} successfull. {{#withemail}} An email was sent to {{email}}, if you do not receive it, please download your keys before living this page.{{/withemail}}", + "successfulluppdate": "Your alias as a Person is now update into {{tribe}}", + "tribedoesnotexist": "Your tribe {{tribe}} does not exist in this town" +} diff --git a/api/routes/nations.js b/api/routes/nations.js index 4f8bbfc..849a107 100755 --- a/api/routes/nations.js +++ b/api/routes/nations.js @@ -33,5 +33,9 @@ router.post( '/push', checkHeaders, ( req, res ) => { res.send( { status: 200, payload: { moreinfo: fs.readFileSync( `${config.tribes}/${config.mayorId}/nationchains/nodes/${config.rootURL}`, 'utf-8' ) } } ) } ) +router.get('/synchro',checkHeaders, isAuthenticated,(req,res)=>{ + // run a sync from a list of electedtown to update nationchains folder + console.log('TODO') +}) module.exports = router; diff --git a/api/routes/odmdb.js b/api/routes/odmdb.js index 0f18c84..1d69c5d 100644 --- a/api/routes/odmdb.js +++ b/api/routes/odmdb.js @@ -32,6 +32,7 @@ router.get( * @apiSuccess (200) {object} data contains indexfile requested * */ + console.log("pzasse"); // indexname = objectname_key_value.json let objectLocation = "../../nationchains/"; if (!conf.api.nationObjects.includes(req.params.objectname)) { @@ -42,13 +43,11 @@ router.get( if (fs.existsSync(indexpath)) { res.status(200).json({ data: fs.readJsonSync(indexpath) }); } else { - res - .status(404) - .json({ - ref: "Odmdb", - msg: "objectfiledoesnotexist", - data: { indexpath }, - }); + res.status(404).json({ + ref: "Odmdb", + msg: "objectfiledoesnotexist", + data: { indexpath }, + }); } } ); @@ -88,13 +87,11 @@ router.get( if (fs.existsSync(objectpath)) { res.status(200).json({ data: fs.readJsonSync(objectpath) }); } else { - res - .status(404) - .json({ - ref: "Odmdb", - msg: "objectfiledoesnotexist", - data: { objectpath }, - }); + res.status(404).json({ + ref: "Odmdb", + msg: "objectfiledoesnotexist", + data: { objectpath }, + }); } } ); diff --git a/api/routes/pagans.js b/api/routes/pagans.js index 119148f..f28ef89 100755 --- a/api/routes/pagans.js +++ b/api/routes/pagans.js @@ -1,12 +1,13 @@ -const express = require( 'express' ); -const path = require( 'path' ); +const express = require("express"); +const path = require("path"); // Classes -const Pagans = require( '../models/Pagans.js' ); +const Pagans = require("../models/Pagans.js"); +const Notifications = require("../models/Notifications.js"); // Middlewares -const checkHeaders = require( '../middlewares/checkHeaders' ); -const isAuthenticated = require( '../middlewares/isAuthenticated' ); -const hasAccessrighton = require( '../middlewares/hasAccessrighton' ); +const checkHeaders = require("../middlewares/checkHeaders"); +const isAuthenticated = require("../middlewares/isAuthenticated"); +const hasAccessrighton = require("../middlewares/hasAccessrighton"); const router = express.Router(); /* models/Pagans.js @@ -47,53 +48,107 @@ Owner means it can be Write/Delete if field OWNER contain the UUID that try to a */ -router.get('/isregister', checkHeaders, isAuthenticated,(req,res)=>{ -/** - * @api {get} /pagans/isregister - * @apiName Is register check xalias and xhash - * @apiGroup Odmdb - * - * @apiUse apxHeader - * - * @apiParam {String} indexname Mandatory if in conf.nationObjects then file is into nationchains/ else in /nationchains/tribes/xtribe/objectname/idx/indexname indexname contains the ObjectName .*_ (before the first _) - * - * @apiError (404) {string} status the file does not exist - * @apiError (404) {string} ref objectmodel to get in the right language - * @apiError (404) {string} msg key to get template from objectmodel - * @apiError (404) {object} data use to pagansmodel: 'Pagans' } );render lg/objectmodel_lg.json - * - * @apiSuccess (200) {object} data contains indexfile requested - * - */ - res.send(Pagans.checkdetachedSignature(req.session.header.xalias,req.session.header.xhash)); +router.get("/isauth", checkHeaders, isAuthenticated, (req, res) => { + /** + * @api {get} /pagans/isregister + * @apiName Is register check xalias and xhash + * @apiGroup Pagans + * + * @apiUse apxHeader + * + * @apiError (400) {object} status missingheaders / xalias does not exist / signaturefailled + * @apiError (401) {object} alias anonymous (not authenticated) + * @apiError (404) {string} tribe does not exist + * + * + * + * @apiSuccess (200) {object} data contains indexfile requested + * + */ + res.send({ + status: 200, + ref: "headers", + msg: "authenticated", + data: { + xalias: req.session.header.xalias, + }, + }); +}); +router.post("/", checkHeaders, isAuthenticated, (req, res) => { + /** + * Create a pagan account from alias, publickey, if trusted recovery => + * Create a person in xtribe/person/xalias.json with profil.auth={email,privatekey, passphrase} + * Middleware isAuthenticated check that: + * - xhash is well signed from private key linked to the publickey of alias + * - check that alias does not already exist (if yes then verifiedsigne would be false) + * Need to wait next block chain to be sure that alias is register in the blokchain + */ + console.log("pass ici", req.body); + const feedback = { alias: req.body.alias, publickey: req.body.publickey }; + const newpagan = Pagans.create(req.body.alias, req.body.publickey); + if (newpagan.status == 200) { + if (req.body.email) { + feedback.withemail = true; + feedback.email = req.body.email; + feedback.privatekey = req.body.privatekey; + feedback.passphrase = req.body.passphrase; + Notifications.send({ + type: "email", + from: "", + dest: [req.body.email], + tpl: "registeremail", + tribe: req.session.header.xtribe, + data: feedback, + }); + } + if (req.body.trustedtribe) { + if (req.app.locals.tribeids.includes(req.body.trustedtribe)) { + delete feedback.withemail; + const persondata = { recovery: feedback }; + res.send( + Pagans.personupdate(req.body.alias, req.body.trustedtribe, persondata) + ); + } else { + res.send({ + status: 404, + ref: "Pagans", + msg: "tribedoesnotexist", + data: { tribe: req.body.trustedtribe }, + }); + } + } else { + newpagan.data = feedback; + res.send(newpagan); + } + } else { + //error to create pagan + res.send(newpagan); + } +}); +router.put("/person", checkHeaders, isAuthenticated, (req, res) => { + /** + * add/update a person = alias + tribe with specific accessright and specific schema link to tribe + * @todo add tribe/schema/person.json + */ + console.log(req.body); + res.send( + Pagans.personupdate(req.body.alias, req.session.header.xtribe, req.body) + ); +}); +router.delete("/:alias", checkHeaders, isAuthenticated, (req, res) => { + console.log(`DELETE pagans nationchains/pagans/${req.params.alias}.json`); + const result = Pagans.delete(req.params.id, req.session.header); + res.status(result.status).send(result.data); +}); - res.send({status:200,ref:"headers",msg:"authenticated",data:{xalias:req.session.header.xalias,xtribe:req.session.header.xtribe}}) -}) -router.post('/', checkHeaders, (req,res)=>{ - // create a pagan account from alias, publickey, if trusted recovery={email,privatekey} - console.log(req.body) - -} ) -router.delete( '/:alias', checkHeaders, isAuthenticated, ( req, res ) => { - console.log( `DELETE pagans nationchains/pagans/${req.params.alias}.json` ); - const result = Pagans.delete( req.params.id, req.session.header ); - res.status( result.status ) - .send( result.data ); -} ); - - - - -router.get( '/isauth', checkHeaders, isAuthenticated, ( req, res ) => { - if( req.session.header.xpseudo == "1" ) { - return res.status( 401 ) - .send( { info: "not authenticate" } ); - } else return res.status( 200 ) - .send( { info: "well authenticated" } ) -} ) -router.post( '/login', checkHeaders, async ( req, res ) => { - // console.log('POST /users/login with: ', req.app.locals.header); - /* +/*router.get("/isauth", checkHeaders, isAuthenticated, (req, res) => { + if (req.session.header.xpseudo == "1") { + return res.status(401).send({ info: "not authenticate" }); + } else return res.status(200).send({ info: "well authenticated" }); +});*/ +router.post("/login", checkHeaders, async (req, res) => { + // console.log('POST /users/login with: ', req.app.locals.header); + /* Check un mot de passe pour un login pour obtenir un token d'authentification valable 1 hour, 1 day @header @@ -104,21 +159,22 @@ router.post( '/login', checkHeaders, async ( req, res ) => { utile le temps de reinitialisé son mot de passe. @return */ - console.log( 'login for ', req.body, "in", req.session.header ) - const log = await Pagans.loginUser( req.session.header, req.body, true ); - console.log( "log user login", log ); - if( log.status == 200 ) { - // update req.app.locals.tokens for this uuid just after login success then next isAuth will be valid - req.app.locals.tokens[ log.data.user.UUID ] = { TOKEN: log.data.user.TOKEN, ACCESSRIGHTS: log.data.user.ACCESSRIGHTS } - console.log( req.app.locals ) - } - return res.status( log.status ) - .send( log.data ); -} ); + console.log("login for ", req.body, "in", req.session.header); + const log = await Pagans.loginUser(req.session.header, req.body, true); + console.log("log user login", log); + if (log.status == 200) { + // update req.app.locals.tokens for this uuid just after login success then next isAuth will be valid + req.app.locals.tokens[log.data.user.UUID] = { + TOKEN: log.data.user.TOKEN, + ACCESSRIGHTS: log.data.user.ACCESSRIGHTS, + }; + console.log(req.app.locals); + } + return res.status(log.status).send(log.data); +}); - -router.get( '/getlinkwithoutpsw/:email', checkHeaders, async ( req, res ) => { - /* +router.get("/getlinkwithoutpsw/:email", checkHeaders, async (req, res) => { + /* Permet pour un email existant de renvoyer un email avec un lien valable 1h @email est le compte pour lequel on demande un accès Réponse: @@ -134,109 +190,161 @@ router.get( '/getlinkwithoutpsw/:email', checkHeaders, async ( req, res ) => { } } */ - console.log( `GET /users/getlinkwithoutpsw for email: ${req.params.email} tribeid :${req.header('X-Client-Id')}` ); - if( !req.params.email ) { - return res.status( 404 ) - .send( { - info: [ 'emailmissing' ], - model: 'Pagans' - } ); - } else { - try { - const getlink = await Pagans.getlinkwithoutpsw( req.params.email, req.session.header ); - console.log( 'getlink', getlink ) - //met à jour le token créer pour le uuid - req.app.locals.tokens[ getlink.data.info.xuuid ] = getlink.data.info.token; - // attention si on relance le serveur le token temporaire est perdu - return res.status( getlink.status ) - .send( getlink.data ); - } catch ( err ) { - console.log( err ) - } - } -} ); -router.post( '/register', checkHeaders, async ( req, res ) => { - console.log( `POST /users for ${req.session.header.xtribe}` ); - if( req.session.header.xjwt == '123123' ) { - // Creation d'un utilisateur avec information de base aucun droit - // On modifie le contenu du form pour n egarder que login/email et psw - // pour le client_id permet de traiter un user en attente de validation - console.log( 'req du post', req ); - } -} ); -router.get( '/info/:listindex', checkHeaders, isAuthenticated, hasAccessrighton( 'users', 'R' ), async ( req, res ) => { - console.log( `get users info on tribeid ${req.session.header.xworkon} for ${req.params.listindex} with accessright`, req.session.header.accessrights.data ); - const result = await Pagans.getinfoPagans( req.session.header.xpresworkon, req.session.header.accessrights, req.params.listindex ); - res.status( result.status ) - .send( result.data ); -} ); -router.get( '/list/:filter/:field', checkHeaders, isAuthenticated, hasAccessrighton( 'users', 'R' ), async ( req, res ) => { - console.log( 'GET /users/list/filtre/champs list for ' + req.session.header.xworkon ); - if( - [ 'admin', 'manager' ].includes( req.session.header.decodetoken[ 'apps' + req.session.header.xworkon + 'profil' ] ) ) { - try { - const userslist = await Pagans.getUserlist( req.session.header, req.params.filter, req.params.field ); - console.log( 'userslist', userslist ); - if( userslist.status == 200 ) { - return res.status( userslist.status ) - .send( userslist.data ); - } - } catch ( err ) { - console.log( err ); - return res.status( 400 ) - .send( { info: 'erreur' } ); - } - } else { - res.status( 403 ) - .send( { - info: [ 'forbiddenAccess' ], - model: 'Pagans' - } ); - } -} ); -router.get( '/uuid/:id', checkHeaders, isAuthenticated, hasAccessrighton( 'users', 'R' ), async ( req, res ) => { - console.log( `GET /users/uuid/${req.params.id}` ); - //console.log('req.app.locals: ', req.app.locals); - //console.log('req.session', req.session); - const result = await Pagans.getUser( req.params.id, req.session.header.xworkon, req.session.header.accessrights ); - res.status( result.status ) - .send( result.data ); -} ); -router.put( '/chgpsw/:id', checkHeaders, isAuthenticated, async ( req, res ) => { - console.log( `PUT update /users/chgpsw/${req.params.id}` ); - try { - const majpsw = await Pagans.updateUserpassword( req.params.id, req.session.header, req.body ); - res.status( majpsw.status ) - .send( majpsw.data ); - } catch ( { - status, - data - } ) { - res.status( status ) - .send( data ); - } -} ); -router.post( '/uuid', checkHeaders, isAuthenticated, hasAccessrighton( 'users', 'C' ), async ( req, res ) => { - console.log( 'POST /users create for ' + req.session.header.xworkon, req.body ); - const usercreate = await Pagans.createUser( req.session.header, req.body ); - return res.status( usercreate.status ) - .send( usercreate.data ); -} ); -router.put( '/uuid/:id', checkHeaders, isAuthenticated, hasAccessrighton( 'users', 'U' ), async ( req, res ) => { - console.log( `PUT update /users/${req.params.id}` ); - // console.log('req.app.locals: ', req.app.locals); - // console.log('req.session', req.session); - try { - const majUser = await Pagans.updateUser( req.params.id, req.session.header, req.body ); - res.status( majUser.status ) - .send( majUser.data ); - } catch ( { - status, - data - } ) { - res.status( status ) - .send( data ); - } -} ); + console.log( + `GET /users/getlinkwithoutpsw for email: ${ + req.params.email + } tribeid :${req.header("X-Client-Id")}` + ); + if (!req.params.email) { + return res.status(404).send({ + info: ["emailmissing"], + model: "Pagans", + }); + } else { + try { + const getlink = await Pagans.getlinkwithoutpsw( + req.params.email, + req.session.header + ); + console.log("getlink", getlink); + //met à jour le token créer pour le uuid + req.app.locals.tokens[getlink.data.info.xuuid] = getlink.data.info.token; + // attention si on relance le serveur le token temporaire est perdu + return res.status(getlink.status).send(getlink.data); + } catch (err) { + console.log(err); + } + } +}); +router.post("/register", checkHeaders, async (req, res) => { + console.log(`POST /users for ${req.session.header.xtribe}`); + if (req.session.header.xjwt == "123123") { + // Creation d'un utilisateur avec information de base aucun droit + // On modifie le contenu du form pour n egarder que login/email et psw + // pour le client_id permet de traiter un user en attente de validation + console.log("req du post", req); + } +}); +router.get( + "/info/:listindex", + checkHeaders, + isAuthenticated, + hasAccessrighton("users", "R"), + async (req, res) => { + console.log( + `get users info on tribeid ${req.session.header.xworkon} for ${req.params.listindex} with accessright`, + req.session.header.accessrights.data + ); + const result = await Pagans.getinfoPagans( + req.session.header.xpresworkon, + req.session.header.accessrights, + req.params.listindex + ); + res.status(result.status).send(result.data); + } +); +router.get( + "/list/:filter/:field", + checkHeaders, + isAuthenticated, + hasAccessrighton("users", "R"), + async (req, res) => { + console.log( + "GET /users/list/filtre/champs list for " + req.session.header.xworkon + ); + if ( + ["admin", "manager"].includes( + req.session.header.decodetoken[ + "apps" + req.session.header.xworkon + "profil" + ] + ) + ) { + try { + const userslist = await Pagans.getUserlist( + req.session.header, + req.params.filter, + req.params.field + ); + console.log("userslist", userslist); + if (userslist.status == 200) { + return res.status(userslist.status).send(userslist.data); + } + } catch (err) { + console.log(err); + return res.status(400).send({ info: "erreur" }); + } + } else { + res.status(403).send({ + info: ["forbiddenAccess"], + model: "Pagans", + }); + } + } +); +router.get( + "/uuid/:id", + checkHeaders, + isAuthenticated, + hasAccessrighton("users", "R"), + async (req, res) => { + console.log(`GET /users/uuid/${req.params.id}`); + //console.log('req.app.locals: ', req.app.locals); + //console.log('req.session', req.session); + const result = await Pagans.getUser( + req.params.id, + req.session.header.xworkon, + req.session.header.accessrights + ); + res.status(result.status).send(result.data); + } +); +router.put("/chgpsw/:id", checkHeaders, isAuthenticated, async (req, res) => { + console.log(`PUT update /users/chgpsw/${req.params.id}`); + try { + const majpsw = await Pagans.updateUserpassword( + req.params.id, + req.session.header, + req.body + ); + res.status(majpsw.status).send(majpsw.data); + } catch ({ status, data }) { + res.status(status).send(data); + } +}); +router.post( + "/uuid", + checkHeaders, + isAuthenticated, + hasAccessrighton("users", "C"), + async (req, res) => { + console.log( + "POST /users create for " + req.session.header.xworkon, + req.body + ); + const usercreate = await Pagans.createUser(req.session.header, req.body); + return res.status(usercreate.status).send(usercreate.data); + } +); +router.put( + "/uuid/:id", + checkHeaders, + isAuthenticated, + hasAccessrighton("users", "U"), + async (req, res) => { + console.log(`PUT update /users/${req.params.id}`); + // console.log('req.app.locals: ', req.app.locals); + // console.log('req.session', req.session); + try { + const majUser = await Pagans.updateUser( + req.params.id, + req.session.header, + req.body + ); + res.status(majUser.status).send(majUser.data); + } catch ({ status, data }) { + res.status(status).send(data); + } + } +); module.exports = router; diff --git a/api/routes/wwws.js b/api/routes/wwws.js new file mode 100644 index 0000000..304f21f --- /dev/null +++ b/api/routes/wwws.js @@ -0,0 +1,45 @@ +const express = require("express"); +const path = require("path"); + +// Classes +const Wwws = require("../models/Wwws.js"); +// Middlewares +const checkHeaders = require("../middlewares/checkHeaders"); +const isAuthenticated = require("../middlewares/isAuthenticated"); +const hasAccessrighton = require("../middlewares/hasAccessrighton"); +const router = express.Router(); +/** + * To manage an nginx conf + * + */ + +router.get("/tribes/:tribeId", checkHeaders, isAuthenticated, (req, res) => { + /** + * @api {get} /www/tribes/:tribeId + * @apiName Get list of application Name + * @apiGroup Www + * + * @apiUse apxHeader + * + * @apiParam {String} tribeId Mandatory that have to exist in current town and + * + * @apiError (404) {string} status the folder does not exist + * @apiError (404) {string} ref objectmodel to get in the right language + * @apiError (404) {string} msg key to get template from objectmodel + * @apiError (404) {object} data use to www's model: 'Www' } );render lg/objectmodel_lg.json + * + * @apiSuccess (200) {object} data contains liste of www conf of a tribe + * + */ + + res.send(Www.configlist(req.params.tribeId)); +}); +router.post("/:webappname", checkHeaders, isAuthenticated, (req, res) => { + /** + * Create a space web /tribes/www/webappname + * + * + * */ + res.send(Wwws.create(req.params.tribeId)); +}); +module.exports = router; diff --git a/apxtrib.js b/apxtrib.js index 73c9700..ba3395a 100755 --- a/apxtrib.js +++ b/apxtrib.js @@ -1,137 +1,146 @@ -const fs = require( 'fs-extra' ); -const bodyParser = require( 'body-parser' ); -const glob = require( 'glob' ); -const path = require( 'path' ); -const cors = require( 'cors' ); -const express = require( 'express' ); -const process = require('process'); +const fs = require("fs-extra"); +const bodyParser = require("body-parser"); +const glob = require("glob"); +const path = require("path"); +const cors = require("cors"); +const express = require("express"); +const process = require("process"); /******************************************* SEE https://gitea.ndda.fr/apxtrib/apxtrib/wiki/Devrules To have a quick understanding and convention before doing deeply in source code */ +// Global data : add here globale variable that take care between RAM space anf fs access // to make absolute path with `${__base}relativepath` -global.__base = __dirname +'/'; +global.__base = __dirname + "/"; +// app.locals.tribeids is defined later in apixtrib.js and allow express app to always have in memory a dynamic of tribeId available in req.app.locals.tribeids + // check setup -if( !fs.existsSync( '/etc/nginx/nginx.conf' ) ) { - console.log( '\x1b[31m Check documentation, nginx have to be installed on this server first, no /etc/nginx/nginx.conf available, install then rerun yarn command.' ); - process.exit(); +if (!fs.existsSync("/etc/nginx/nginx.conf")) { + console.log( + "\x1b[31m Check documentation, nginx have to be installed on this server first, no /etc/nginx/nginx.conf available, install then rerun yarn command." + ); + process.exit(); } -if( !fs.existsSync( './nationchains/tribes/conf.json' ) ) { - // this town is not set - console.log( `\x1b[42m############################################################################################\x1b[0m\n\x1b[42mWellcome into apxtrib, you must first init your town and tribes by a 'yarn setup'. \x1b[0m \n\x1b[42mThen 'yarn dev' or 'yarn startpm2' or 'yarn unittest'. Check README's project to learn more.\x1b[0m\n\x1b[42m############################################################################################\x1b[0m` ); - process.exit(); +if (!fs.existsSync(`${__base}nationchains/tribes/conf.json`)) { + // this town is not set + console.log( + `\x1b[42m############################################################################################\x1b[0m\n\x1b[42mWellcome into apxtrib, you must first init your town and tribes by a 'yarn setup'. \x1b[0m \n\x1b[42mThen 'yarn dev' or 'yarn startpm2' or 'yarn unittest'. Check README's project to learn more.\x1b[0m\n\x1b[42m############################################################################################\x1b[0m` + ); + process.exit(); } -const conf = require( './nationchains/tribes/conf.json' ); +const conf = require(`${__base}nationchains/tribes/conf.json`); // To make public this conf, careffull this localconf will be public, this the only difference between all apxtrib node -const localconf={nationId:conf.nationId,townId:conf.townId,tribeId:conf.tribeId,comment:"Generate by apxtrib.js with minimum of information"}; -fs.outputJsonSync('./nationchains/www/adminapx/static/tpldata/conf_en.json',localconf); -// Tribes allow to get local apxtrib instance context +const localconf = { + nationId: conf.nationId, + townId: conf.townId, + tribeId: conf.tribeId, + comment: "Generate by apxtrib.js with minimum of information", +}; +fs.outputJsonSync( + `${__base}nationchains/www/adminapx/static/tpldata/setup_en.json`, + localconf +); +// Run main express process + +// Each tribe has a context (domain, plugins route, website ) are all describe into idx tribeId_all.json +// {"tribename":{"tribeId":"tribename","dns":[array of domain],"status":"unchain","nationId":"ants","townId":"usbfarm"}} + // dataclient .tribeids [] .DOMs [] .routes (plugins {url:name route:path}) .appname {tribeid:[website]} //const dataclient = require( './api/models/Tribes' ).init(); -const tribelist=fs.readJsonSync(`./nationchains/tribes/idx/tribeId_all.json`); -let doms=conf.dns -let tribeIds=[] -let routes = glob.sync( './api/routes/*.js' ) - .map( f => { - return { url: `/${path.basename(f,'.js')}`, route: f } - } ); -//routes={url,route} check how to add plugin tribe route -Object.keys(tribelist).forEach(t=>{ - tribelist[t].dns.forEach(d=>{ - const dm=d.split('.').slice(-2).join('.') - if (!doms.includes(dm)) doms.push(dm) - }) - tribeIds.push(t) -}) -console.log('Allowed DOMs to access to this apxtrib server: ', doms ) +const tribelist = fs.readJsonSync( + `${__base}/nationchains/tribes/idx/tribeId_all.json` +); +let doms = conf.dns; // only dns of town during the init process +let tribeIds = []; +let routes = glob.sync(`${__base}/api/routes/*.js`).map((f) => { + return { url: `/${path.basename(f, ".js")}`, route: f }; +}); +//routes={url,route} check how to add plugin tribe route later +// keep only the 2 last part (.) of domain name to validate cors with it (generic domain) +Object.keys(tribelist).forEach((t) => { + tribelist[t].dns.forEach((d) => { + const dm = d.split(".").slice(-2).join("."); + if (!doms.includes(dm)) doms.push(dm); + }); + tribeIds.push(t); +}); +console.log("Allowed DOMs to access to this apxtrib server: ", doms); const app = express(); -Object.keys(conf.api.appset).forEach(p=>{ - app.set(p,conf.api.appset[p]) -}) +// load express parameter from conf +Object.keys(conf.api.appset).forEach((p) => { + app.set(p, conf.api.appset[p]); +}); // To set depending of data form or get size to send -app.use( bodyParser.urlencoded( conf.api.bodyparse.urlencoded ) ); +app.use(bodyParser.urlencoded(conf.api.bodyparse.urlencoded)); // To set depending of post put json data size to send -app.use( express.json() ) -app.use( bodyParser.json( conf.api.bodyparse.json ) ); +app.use(express.json()); +app.use(bodyParser.json(conf.api.bodyparse.json)); app.locals.tribeids = tribeIds; -console.log( 'app.locals.tribeids', app.locals.tribeids ); -// token will be store in /tmp/token/pseudo_token to check if isauthenticated -// User token authentification and user init user search -/*const datauser = require( './api/models/Pagans' ) - .init( dataclient.tribeids ); -app.locals.tokens = datauser.tokens; -console.log( 'app.locals.tokens key ', Object.keys( app.locals.tokens ) ) -*/ +console.log("app.locals.tribeids", app.locals.tribeids); // Cors management const corsOptions = { - origin: ( origin, callback ) => { - if( origin === undefined ) { - callback( null, true ); - } else if( origin.indexOf( 'chrome-extension' ) > -1 ) { - callback( null, true ); - } else { - //console.log( 'origin', origin ) - //marchais avant modif eslint const rematch = ( /^https?\:\/\/(.*)\:.*/g ).exec( origin ) - const rematch = ( /^https?:\/\/(.*):.*/g ) - .exec( origin ) - //console.log( rematch ) - let tmp = origin.replace( /http.?:\/\//g, '' ) - .split( '.' ) + origin: (origin, callback) => { + if (origin === undefined) { + callback(null, true); + } else if (origin.indexOf("chrome-extension") > -1) { + callback(null, true); + } else { + //console.log( 'origin', origin ) + //marchais avant modif eslint const rematch = ( /^https?\:\/\/(.*)\:.*/g ).exec( origin ) + const rematch = /^https?:\/\/(.*):.*/g.exec(origin); + //console.log( rematch ) + let tmp = origin.replace(/http.?:\/\//g, "").split("."); - if( rematch && rematch.length > 1 ) tmp = rematch[ 1 ].split( '.' ); - //console.log( 'tmp', tmp ) - let dom = tmp[ tmp.length - 1 ]; - if( tmp.length > 1 ) { - dom = `${tmp[tmp.length-2]}.${tmp[tmp.length-1]}` - } - console.log( `origin: ${origin}, dom:${dom}, CORS allowed? : ${dataclient.DOMs.includes( dom )}` ); - if( dataclient.DOMs.includes( dom ) ) { - callback( null, true ) - } else { - console.log( `Origin is not allowed by CORS` ); - callback( new Error( 'Not allowed by CORS' ) ); - } - } - }, - exposedHeaders: Object.keys( conf.api.exposedHeaders ) + if (rematch && rematch.length > 1) tmp = rematch[1].split("."); + //console.log( 'tmp', tmp ) + let dom = tmp[tmp.length - 1]; + if (tmp.length > 1) { + dom = `${tmp[tmp.length - 2]}.${tmp[tmp.length - 1]}`; + } + console.log( + `origin: ${origin}, dom:${dom}, CORS allowed? : ${doms.includes(dom)}` + ); + if (doms.includes(dom)) { + callback(null, true); + } else { + console.log(`Origin is not allowed by CORS`); + callback(new Error("Not allowed by CORS")); + } + } + }, + exposedHeaders: Object.keys(conf.api.exposedHeaders), }; // CORS -app.use( cors( corsOptions ) ); -// Static Routes +app.use(cors(corsOptions)); +// Static Routes // try to use nginx route instead in comments /*app.use( express.static( `${__dirname}/nationchains/tribes/${conf.mayorId}/www/cdn/public`, { dotfiles: 'allow' } ) ); */ -//Allow to public access a space dev delivered by apxtrib -// this is just a static open route for dev purpose, -// for production, we'll use a nginx static set to /www/app/appname -/*console.log( `${conf.dnsapxtrib}/space/tribeid/website`, dataclient.appname ); -Object.keys( dataclient.appname ) - .forEach( cid => { - dataclient.appname[ cid ].forEach( website => { - app.use( `/space/${cid}/${website}`, express.static( `${conf.tribes}/${cid}/spacedev/${website}` ) ); - } ) - } ); -*/ // Routers add any routes from /routes and /plugins -console.log( 'Routes available on this apxtrib instance' ); -console.log( routes ); -// prefix only use for dev purpose in production a proxy nginx redirect /app/ to node apxtrib +console.log("Routes available on this apxtrib instance"); +console.log(routes); +routes.forEach((r) => { + try { + app.use(r.url, require(r.route)); + } catch (err) { + console.log( + `\x1b[31m!!! WARNING issue with route ${r.route} from ${r.url} check err if route is key then solve err, if not just be aware that this route won't work on your server. If you are not the maintainer and no turn around please contact the email maintainer.\x1b[0m` + ); + console.log("raise err-:", err); + } +}); -routes.forEach( r => { - try { - app.use( r.url, require( r.route ) ); - } catch ( err ) { - console.log( `\x1b[31m!!! WARNING issue with route ${r.route} from ${r.url} check err if route is key then solve err, if not just be aware that this route won't work on your server. If you are not the maintainer and no turn around please contact the email maintainer.\x1b[0m` ) - console.log( 'raise err-:', err ); - } -} ) - -app.listen( conf.api.port, () => { - console.log( `check in your browser that api works http://${conf.dns}:${conf.api.port}` ); -} ); - -console.log( "\x1b[42m\x1b[37m", "Made with love for people's freedom, enjoy !!!", "\x1b[0m" ); +app.listen(conf.api.port, () => { + console.log( + `check in your browser that api works http://${conf.dns}:${conf.api.port}` + ); +}); +console.log( + "\x1b[42m\x1b[37m", + "Made with love for people's freedom, enjoy !!!", + "\x1b[0m" +); diff --git a/electedtown.json b/electedtown.json new file mode 100644 index 0000000..d929626 --- /dev/null +++ b/electedtown.json @@ -0,0 +1 @@ +["https://wallants.ndda.fr/nations/synchro"] \ No newline at end of file diff --git a/nationchains/www/adminapx/nginx/nginx.conf.mustache b/nationchains/www/adminapx/ASUPnginx/nginx.conf.mustache similarity index 100% rename from nationchains/www/adminapx/nginx/nginx.conf.mustache rename to nationchains/www/adminapx/ASUPnginx/nginx.conf.mustache diff --git a/nationchains/www/adminapx/nginx/proxy_params.mustache b/nationchains/www/adminapx/ASUPnginx/proxy_params copy.mustache similarity index 100% rename from nationchains/www/adminapx/nginx/proxy_params.mustache rename to nationchains/www/adminapx/ASUPnginx/proxy_params copy.mustache diff --git a/nationchains/www/adminapx/ASUPnginx/proxy_params.mustache b/nationchains/www/adminapx/ASUPnginx/proxy_params.mustache new file mode 100644 index 0000000..1b73dde --- /dev/null +++ b/nationchains/www/adminapx/ASUPnginx/proxy_params.mustache @@ -0,0 +1,11 @@ + proxy_set_header Host $http_host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header X-Forwarded-Proto $scheme; + client_max_body_size 10m; + client_body_buffer_size 128k; + proxy_connect_timeout 90; + proxy_send_timeout 90; + proxy_read_timeout 90; + proxy_buffers 32 4k; + proxy_set_header X-NginX-Proxy true; \ No newline at end of file diff --git a/nationchains/www/adminapx/index_en.html b/nationchains/www/adminapx/index_en.html index 66a1a87..26b3456 100644 --- a/nationchains/www/adminapx/index_en.html +++ b/nationchains/www/adminapx/index_en.html @@ -3,33 +3,38 @@ setup apXtrib + - - + + + - - - - + + + + + + + + + -
-

apXtrib

-

Manage and understand apXtrib back-end

-
- +
+
+
+ Loading... +
- - -
-
+
+
+
- + + \ No newline at end of file diff --git a/nationchains/www/adminapx/static/img/logo/favicon.png b/nationchains/www/adminapx/static/img/logo/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..599a05ccc9c65c325d9e030e79eb0061d1d5dbb7 GIT binary patch literal 3785 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 Fe*kCWg7N?W literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/img/logo/logobgdark.png b/nationchains/www/adminapx/static/img/logo/logobgdark.png new file mode 100644 index 0000000000000000000000000000000000000000..89d274ce92173c621f499b2b956e56067f8859d3 GIT binary patch literal 7188 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 + + + + + + + + + + + + + + + + + + + + + + apXtrib + BLOCKCHAIN OF DEMOCRACY + + diff --git a/nationchains/www/adminapx/static/img/logo/logobglight.png b/nationchains/www/adminapx/static/img/logo/logobglight.png new file mode 100644 index 0000000000000000000000000000000000000000..882c9fc089c9cc0a7b5637fd31faa6271e163b75 GIT binary patch literal 10193 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 Kp+?Rk;y(cY8-5%B literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/img/logo/logobglight.svg b/nationchains/www/adminapx/static/img/logo/logobglight.svg new file mode 100644 index 0000000..1e9a259 --- /dev/null +++ b/nationchains/www/adminapx/static/img/logo/logobglight.svg @@ -0,0 +1,156 @@ + + + + + + + + + + + + + + + + + + + + + apXtrib + BLOCKCHAIN OF DEMOCRACY + + + diff --git a/nationchains/www/adminapx/static/img/logo/logocarredark.png b/nationchains/www/adminapx/static/img/logo/logocarredark.png new file mode 100644 index 0000000000000000000000000000000000000000..e27c2179eeabf3a9f73a1d251646b04e5d846969 GIT binary patch literal 4606 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 literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/img/logo/logocarrelight.png b/nationchains/www/adminapx/static/img/logo/logocarrelight.png new file mode 100644 index 0000000000000000000000000000000000000000..c49780384e77270b7ef2abfe81b0969ccf5dc579 GIT binary patch literal 4255 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 - if (alias.length<3 || apx.data.object['index_pagans_alias_all'].includes(alias) ){ - alert('Please chose another alias') - return; - } - const keys = apx.generateKey(alias,passphrase) - for(let tag of ['inputalias','inputpassphrase']){ - document.getElementById(tag).classList.add('disabled') - } - document.getElementById('privatekey').setAttribute("key",keys.privateKey); - document.getElementById('publickey').setAttribute("key",keys.publicKey); - document.getElementById('generatekeys').classList.add('d-none'); - document.getElementById('downloadkeys').classList.remove('d-none'); - document.getElementById('createId').classList.remove('d-none'); -} - -app.registerIdentity=()=>{ - const data={ - alias:document.getElementById('inputalias').value, - privateKey:document.getElementById('privatekey').getAttribute('key'), - publicKey:document.getElementById('publickey').getAttribute('key'), - passphrase:document.getElementById('inputpassphrase').value - } - axios.post('api/pagans',data,apx.data.headers) - .then(rep=>{ - // genere authentification headers.xalias, xhash store object.pagans_alias={privateKey, passphrase}; console.log(rep)}) - // if check trust add data email recovery axios.post('api/persons,data, apx.data.headers) +app.runapirequest = (destmodalid, axiosreq, modalcontent) => { + /** + * Axios option + * {method: GET POST,..., + * url:'/relativepath' + * data:{anydata to pass} + * responseType:'json'} + */ + if (!modalcontent) { + modalcontent = { + title: "An axios request", + body: "", + actions: [], + }; + } + modalcontent.body += JSON.stringify(axiosreq); + console.log(apx.data.headers); + axiosreq.headers = apx.data.headers; + axios(axiosreq) + .then((rep) => { + console.log(rep); + modalcontent.body += "
" + JSON.stringify(rep.data); + app.modalmanagement("reqaxios", apx.data.tpl.apxmodal, modalcontent); }) - .catch(err=>{ - console.log('sorry',err); - }) -} - - - - - + .catch((err, rep) => { + console.log(err); + modalcontent.title += " - Status :" + err.response.status; + modalcontent.body += JSON.stringify(err.response.data); + app.modalmanagement("reqaxios", apx.data.tpl.apxmodal, modalcontent); + }); +}; +app.modalmanagement = (modalid, tpl, data, optionmodal) => { + /** + * Pre-request in hatml + *
+ * data must be online with tpl + * optionmodal see https://getbootstrap.com/docs/5.0/components/modal/ + */ + if (!optionmodal) + optionmodal = { backdrop: true, keyboard: true, focus: true }; + if (document.getElementById(modalid)) { + document.getElementById(modalid).remove(); + } + data.modalid = modalid; + document.getElementById("apxmodal").innerHTML += Mustache.render(tpl, data); + var currentmodal = new bootstrap.Modal( + document.getElementById(modalid), + optionmodal + ); + //var currentmodal = bootstrap.Modal.getOrCreateInstance(modalid); + currentmodal.show(); +}; app.search = (elt) => { - //@todo get search string from input then return tpldata.listofarticles={"search":[]} + //@todo get search string from input then return tpldata.articleslist={"search":[]} console.log("A FAIRE"); }; app.downloadlink = (keys, object, prefix) => { @@ -81,50 +96,52 @@ app.downloadlink = (keys, object, prefix) => { }; app.setupdata = () => { - //todo generate tpldata.listofarticle get html from /static/html - apx.data.tpldata.listofarticle = { + //todo generate tpldata.articleslist get html from /static/html + apx.data.tpldata.articleslist = { mostread: [{}], news: [{}], search: [], articles: "html", }; - const list = {} + const list = {}; document.querySelectorAll("[add2data]").forEach((e) => { /** Collect from any tag with add2data url attribute (tpl tpldata object) * name is the filename : * for tpl and tpldata (relativepath/filename_xx.ext) name=filename * For public object (pagans, towns, nations, block) not tribes object are gettable (only) * with /nationchains/objectName/idx/existingindex.json - * or for an item /itm//primarykey_value_of_item.json - * - * For tribe object get or for modification you need to use api and to have accessright setup properly: + * or for an item /itm//primarykey_value_of_item.json + * + * For tribe object get or for modification you need to use api and to have accessright setup properly: * for object index (/api/odmdb/objectname/idx/filename.json) name =objectname_filename * for object item (/api/odmdb/objectname/itm/primarykey.json) name =objectname_primarykey */ - if (e.getAttribute('object')){ - const url = e.getAttribute('object') - const spliturlobj=url.split("/").slice(-3); - const objectname=spliturlobj[0]; - const objecttype=spliturlobj[1]; - const objectkey=spliturlobj[2].substring(0,spliturlobj[2].length-5) - if (!list[objectname]) list[objectname]={}; - list[objectname][`${objecttype}${objectkey}`]= url; - }; - for (let k of ["tpl", "tpldata"]) { - if (e.getAttribute(k)){ - const url=e.getAttribute(k); - let localname=url.split('/').slice(-1)[0]; - if (url.includes('.mustache')) localname=localname.substring(0,localname.length-12); - if (url.includes('.html') || url.includes('.json') ) localname=localname.substring(0,localname.length-8); - if (!list[k]) list[k]={}; - list[k][localname]=url; - } + if (e.getAttribute("object")) { + const url = e.getAttribute("object"); + const spliturlobj = url.split("/").slice(-3); + const objectname = spliturlobj[0]; + const objecttype = spliturlobj[1]; + const objectkey = spliturlobj[2].substring(0, spliturlobj[2].length - 5); + if (!list[objectname]) list[objectname] = {}; + list[objectname][`${objecttype}${objectkey}`] = url; } -}); + for (let k of ["tpl", "tpldata"]) { + if (e.getAttribute(k)) { + const url = e.getAttribute(k); + let localname = url.split("/").slice(-1)[0]; + if (url.includes(".mustache")) + localname = localname.substring(0, localname.length - 12); + if (url.includes(".html") || url.includes(".json")) + localname = localname.substring(0, localname.length - 8); + if (!list[k]) list[k] = {}; + list[k][localname] = url; + } + } + }); // load external template and data - - for (let k of Object.keys(list) ) { + + for (let k of Object.keys(list)) { console.log(k, list[k]); apx.loadfile(list[k], k); } @@ -132,21 +149,21 @@ app.setupdata = () => { if (apx.data.firsttimeload) { // Need to wait the first time apx.data fully load setTimeout(() => { - app.setuppage(); - }, 300); + app.setuppage(); + }, 500); delete apx.data.firsttimeload; apx.save(); - }else{ + } else { app.setuppage(); } }; app.load = (idtoload, tplname, tpldata) => { - const tpl = apx.data.tpl[tplname] - ? apx.data.tpl[tplname] - : `missing template ${tplname}`; - const data = typeof tpldata == "string" ? apx.data.tpldata[tpldata] : tpldata; - document.getElementById(idtoload).innerHTML = Mustache.render(tpl, data); - }; + const tpl = apx.data.tpl[tplname] + ? apx.data.tpl[tplname] + : `missing template ${tplname}`; + const data = typeof tpldata == "string" ? apx.data.tpldata[tpldata] : tpldata; + document.getElementById(idtoload).innerHTML = Mustache.render(tpl, data); +}; app.setuppage = () => { // load partial template document.querySelectorAll("[apptoload]").forEach((e) => { diff --git a/nationchains/www/adminapx/static/js/apxpagans.js b/nationchains/www/adminapx/static/js/apxpagans.js new file mode 100644 index 0000000..9dedab5 --- /dev/null +++ b/nationchains/www/adminapx/static/js/apxpagans.js @@ -0,0 +1,232 @@ +/*eslint no-undef:0*/ +/*eslint-env browser*/ + +"use strict"; +var pagans = pagans || {}; +/** + * pagans.generateKey(params) create a public/private key compatible with apXtrib backend and store it in apx.data + * pagans.detachedSignature(params) generate a detachedSignature for a Message that backend can check with the publicKey of the alias + * pagans.checkdetachedSignature(params) check a detachedSignature for a Message (used on the backend as well) + * + * + */ + +pagans.loadtpldata = () => { + // adapte tpldata to template tpl + // get list of tribes from nationchains/towns/idx/townId_all.json + const datacreatepagan = { tribes: [] }; + + apx.data.towns.idxtownId_all[apx.data.tpldata.setup.townId].tribes.forEach( + (t) => { + if (t == apx.data.tpldata.setup.tribeId) { + datacreatepagan.tribes.push({ selected: true, tribeId: t }); + } else { + datacreatepagan.tribes.push({ tribeId: t }); + } + } + ); + return datacreatepagan; +}; + +pagans.generateKey = async (alias, passphrase) => { + /** + * @param {string} alias a unique alias that identify an identity + * @param {string} passphrase a string to cipher the publicKey (can be empty, less secure but simpler) + * @return {publicKey,privateKey} with userIds = [{alias}] + */ + const pgpparam = { + type: "ecc", // Type of the key, defaults to ECC + curve: "curve25519", // ECC curve name, defaults to curve25519 + userIDs: [{ alias: alias }], // you can pass multiple user IDs + passphrase: passphrase, // protects the private key + format: "armored", // output key format, defaults to 'armored' (other options: 'binary' or 'object') + }; + const { privateKey, publicKey } = await openpgp.generateKey(pgpparam); + // key start by '-----BEGIN PGP PRIVATE KEY BLOCK ... ' + // get liste of alias:pubklickey await axios.get('api/v0/pagans') + // check alias does not exist + return { alias, privateKey, publicKey }; +}; +pagans.detachedSignature = async (pubK, privK, passphrase, message) => { + /** + * @pubK {string} a text public key + * @privK {string} a test priv key + * @passphrase {string} used to read privK + * @message {string} message to sign + * @Return a detached Signature of the message + */ + const publicKey = await openpgp.readKey({ armoredKey: pubK }); + let privateKey; + if (passphrase == "") { + privateKey = await openpgp.readPrivateKey({ armoredKey: privK }); + } else { + privateKey = await openpgp.decryptKey({ + privateKey: await openpgp.readPrivateKey({ armoredKey: privK }), + passphrase, + }); + } + console.log(message); + const msg = await openpgp.createMessage({ text: message }); + console.log(msg); + const sig = await openpgp.sign({ + message: msg, + signingKeys: privateKey, + detached: true, + }); + return btoa(sig); +}; +pagans.authenticatedetachedSignature = async ( + alias, + pubK, + detachedSignature, + message +) => { + /** + * Check that alias (pubkey) signe a message + * @alias {string} alias link to the publicKey + * @pubK {string} publiKey text format + * @detachedSignature {string} a detachedsignatured get from apx.detachedSignature + * @message {string} the message signed + * @return {boolean} true the message was signed by alias + * false the message was not signed by alias + */ + const publicKey = await openpgp.readKey({ armoredKey: pubK }); + const msg = await openpgp.createMessage({ text: message }); + const signature = await openpgp.readSignature({ + armoredSignature: atob(detachedSignature), // parse detached signature + }); + const verificationResult = await openpgp.verify({ + msg, // Message object + signature, + verificationKeys: publicKey, + }); + const { verified, keyID } = verificationResult.signatures[0]; + try { + await verified; // throws on invalid signature + console.log("Signed by key id " + keyID.toHex()); + return KeyId.toHex().alias == alias; + } catch (e) { + console.log("Signature could not be verified: " + e.message); + return false; + } +}; +pagans.createIdentity = async (alias, passphrase = "") => { + console.log(Object.keys(apx.data.pagans["idxalias_all"])); + if ( + alias.length < 3 || + Object.keys(apx.data.pagans["idxalias_all"]).includes(alias) + ) { + alert("Please chose another alias"); + return; + } + const keys = await pagans.generateKey(alias, passphrase); + for (let tag of ["inputalias", "inputpassphrase"]) { + document.getElementById(tag).classList.add("disabled"); + } + console.log(keys); + document.getElementById("privatekey").setAttribute("key", keys.privateKey); + document.getElementById("publickey").setAttribute("key", keys.publicKey); + document.getElementById("generatekeys").classList.add("d-none"); + document.getElementById("trustintribe").classList.remove("d-none"); + document.getElementById("downloadkeys").classList.remove("d-none"); + document.getElementById("createId").classList.remove("d-none"); +}; + +pagans.registerIdentity = async () => { + const emailregex = + /^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/; + const data = {}; + data.alias = document.getElementById("inputalias").value; + data.publickey = document.getElementById("publickey").getAttribute("key"); + + if (document.getElementById("inputemailrecovery").value != "") { + data.email = document.getElementById("inputemailrecovery").value; + } + if (data.email && !emailregex.test(data.email)) { + alert("Check your email"); + return false; + } + var select = document.getElementById("trustedtribe"); + if (document.getElementById("trustedcheck").checked) { + data.trustedtribe = select.options[select.selectedIndex].value; + data.privatekey = document.getElementById("privatekey").getAttribute("key"); + data.passphrase = document.getElementById("inputpassphrase").value; + } + if (document.getElementById("trustedcheck").checked && !data.email) { + alert("Please provide a valid email"); + return false; + } + console.log(data); + //store and create authentification by signing 'alias_timestamp' + /* const passp = data.passphrase ? data.passphrase : ""; + apx.data.auth = { + alias: data.alias, + publickey: data.publickey, + privatekey: document.getElementById("privatekey").getAttribute("key"), + passphrase: passp, + }; + apx.data.headers.xalias = data.alias; + apx.data.headers.xdays = dayjs().valueOf(); + apx.save(); + const msg = `${data.alias}_${apx.data.headers.xdays}`; + apx.data.headers.xhash = await pagans.detachedSignature( + apx.data.auth.publickey, + apx.data.auth.privatekey, + passp, + msg + ); + apx.save();*/ + await pagans.authentifyme( + data.alias, + data.passphrase, + document.getElementById("privatekey").getAttribute("key"), + document.getElementById("publickey").getAttribute("key") + ); + console.log("header", apx.data.headers); + axios + .post("api/pagans", data, { headers: apx.data.headers }) + .then((reppagan) => { + console.log(reppagan); + }) + .catch((err) => { + console.log("sorry", err); + }); +}; + +pagans.authentifyme = async ( + alias, + passphrase, + privatekey, + publickeycreate +) => { + /** + * Set apx.data.auth with pub, priv, passphrase alias that allow authentification + * set headers with xdays (timestamp) and xhash of message: {alias}_{timestamp} generate with pub & priv key + * + * @Param {key} publickeycreate optional when alias does not exist + */ + console.log(alias, passphrase); + console.log(privatekey); + const passp = passphrase ? passphrase : ""; + const publickey = publickeycreate + ? publickeycreate + : apx.data.pagans.idxalias_all[alias].publicKey; + apx.data.auth = { + alias: alias, + publickey: publickey, + privatekey: privatekey, + passphrase: passp, + }; + apx.data.headers.xalias = alias; + apx.data.headers.xdays = dayjs().valueOf(); + apx.save(); + const msg = `${alias}_${apx.data.headers.xdays}`; + + apx.data.headers.xhash = await pagans.detachedSignature( + apx.data.auth.publickey, + apx.data.auth.privatekey, + passp, + msg + ); + apx.save(); +}; diff --git a/nationchains/www/adminapx/static/js/apxtowns.js b/nationchains/www/adminapx/static/js/apxtowns.js new file mode 100644 index 0000000..e69de29 diff --git a/nationchains/www/adminapx/static/js/apxtribcli.js b/nationchains/www/adminapx/static/js/apxtribcli.js index a721649..f459e1b 100644 --- a/nationchains/www/adminapx/static/js/apxtribcli.js +++ b/nationchains/www/adminapx/static/js/apxtribcli.js @@ -21,93 +21,12 @@ var apx = apx || {}; }, objects:{} }; - * apx.generateKey(params) create a public/private key compatible with apXtrib backend and store it in apx.data - * apx.detachedSignature(params) generate a detachedSignature for a Message that backend can check with the publicKey of the alias - * apx.checkdetachedSignature(params) check a detachedSignature for a Message (used on the backend as well) - * apx.ready(callback) equivalent of jquery Document.ready() * apx.save() save apx.data as xapp value in localStorage * apx.update() get localStorage up to date * apx.loadfile({url:name},dest,axiosoptions) async that wait to load a list of relativ url and store it in a apx.data[dest] */ -apx.generateKey = async (alias, passphrase) => { - /** - * @param {string} alias a unique alias that identify an identity - * @param {string} passphrase a string to cipher the publicKey (can be empty, less secure but simpler) - * @return {publicKey,privateKey} with userIds = [{alias}] - */ - const { privateKey, publicKey } = await openpgp.generateKey({ - type: "ecc", // Type of the key, defaults to ECC - curve: "curve25519", // ECC curve name, defaults to curve25519 - userIDs: [{ alias: alias }], // you can pass multiple user IDs - passphrase: passphrase, // protects the private key - format: "armored", // output key format, defaults to 'armored' (other options: 'binary' or 'object') - }); - // key start by '-----BEGIN PGP PRIVATE KEY BLOCK ... ' - // get liste of alias:pubklickey await axios.get('api/v0/pagans') - // check alias does not exist - return { alias, privateKey, publicKey }; - /* document.getElementById('privatekey').innerHTML=privateKey; - document.getElementById('publickey').innerHTML=publicKey; - apx.data.pagans={} - apx.data.pagans.privateKey=privateKey; - apx.data.pagans.publicKey=publicKey; - apx.save(); - console.log(apx.data) - */ -}; -apx.detachedSignature = async (pubK, privK, passphrase, message) => { - /** - * @pubK {string} a text public key - * @privK {string} a test priv key - * @passphrase {string} used to read privK - * @message {string} message to sign - * @Return a detached Signature of the message - */ - const publicKey = await openpgp.readKey({ armoredKey: pubK }); - const privateKey = await openpgp.decryptKey({ - privateKey: await openpgp.readPrivateKey({ armoredKey: privK }), - passphrase, - }); - const msg = await openpgp.createMessage({ text: message }); - return await openpgp.sign({ msg, signinKeys: privK, detached: true }); -}; -apx.checkdetachedSignature = async ( - alias, - pubK, - detachedSignature, - message -) => { - /** - * @alias {string} alias link to the publicKey - * @pubK {string} publiKey text format - * @detachedSignature {string} a detachedsignatured get from apx.detachedSignature - * @message {string} the message signed - * @return {boolean} true the message was signed by alias - * false the message was not signed by alias - */ - const publicKey = await openpgp.readKey({ armoredKey: pubK }); - const msg = await openpgp.createMessage({ text: message }); - const signature = await openpgp.readSignature({ - armoredSignature: detachedSignature, // parse detached signature - }); - const verificationResult = await openpgp.verify({ - msg, // Message object - signature, - verificationKeys: publicKey - }); - const { verified, keyID } = verificationResult.signatures[0]; - try { - await verified; // throws on invalid signature - console.log("Signed by key id " + keyID.toHex()); - return KeyId.toHex().alias == alias; - } catch (e) { - console.log("Signature could not be verified: " + e.message); - return false; - } -}; - apx.ready = (callback) => { /** * Wait loading DOM, when ready run callback function @@ -156,13 +75,17 @@ apx.loadfile = async (list, dest, axiosoptions) => { }; const urls = Object.keys(invertlist); if (!axiosoptions) { + // force no cache browser + apx.data.headers["Cache-Control"] = "no-cache"; + apx.data.headers["Pragma"] = "no-cache"; + apx.data.headers["Expires"] = 0; axiosoptions = { headers: apx.data.headers }; } - console.log('axiosiptions',axiosoptions) + console.log("axiosiptions", axiosoptions); const promiseArray = urls.map((r) => fetchURL(r, axiosoptions)); await Promise.all(promiseArray) .then((data) => { - console.log(data) + console.log(data); for (let pos = 0; pos < urls.length; pos++) { //console.log( 'url', urls[ pos ] ) //console.log( data[ pos ] ); diff --git a/nationchains/www/adminapx/static/js/apxtribes.js b/nationchains/www/adminapx/static/js/apxtribes.js new file mode 100644 index 0000000..e69de29 diff --git a/nationchains/www/adminapx/static/style/fonts/logo/OFL.txt b/nationchains/www/adminapx/static/style/fonts/logo/OFL.txt new file mode 100644 index 0000000..3ee9cb5 --- /dev/null +++ b/nationchains/www/adminapx/static/style/fonts/logo/OFL.txt @@ -0,0 +1,93 @@ +Copyright 2011 The Quicksand Project Authors (https://github.com/andrew-paglinawan/QuicksandFamily), with Reserved Font Name “Quicksand”. + +This Font Software is licensed under the SIL Open Font License, Version 1.1. +This license is copied below, and is also available with a FAQ at: +http://scripts.sil.org/OFL + + +----------------------------------------------------------- +SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007 +----------------------------------------------------------- + +PREAMBLE +The goals of the Open Font License (OFL) are to stimulate worldwide +development of collaborative font projects, to support the font creation +efforts of academic and linguistic communities, and to provide a free and +open framework in which fonts may be shared and improved in partnership +with others. + +The OFL allows the licensed fonts to be used, studied, modified and +redistributed freely as long as they are not sold by themselves. The +fonts, including any derivative works, can be bundled, embedded, +redistributed and/or sold with any software provided that any reserved +names are not used by derivative works. The fonts and derivatives, +however, cannot be released under any other type of license. The +requirement for fonts to remain under this license does not apply +to any document created using the fonts or their derivatives. + +DEFINITIONS +"Font Software" refers to the set of files released by the Copyright +Holder(s) under this license and clearly marked as such. This may +include source files, build scripts and documentation. + +"Reserved Font Name" refers to any names specified as such after the +copyright statement(s). + +"Original Version" refers to the collection of Font Software components as +distributed by the Copyright Holder(s). + +"Modified Version" refers to any derivative made by adding to, deleting, +or substituting -- in part or in whole -- any of the components of the +Original Version, by changing formats or by porting the Font Software to a +new environment. + +"Author" refers to any designer, engineer, programmer, technical +writer or other person who contributed to the Font Software. + +PERMISSION & CONDITIONS +Permission is hereby granted, free of charge, to any person obtaining +a copy of the Font Software, to use, study, copy, merge, embed, modify, +redistribute, and sell modified and unmodified copies of the Font +Software, subject to the following conditions: + +1) Neither the Font Software nor any of its individual components, +in Original or Modified Versions, may be sold by itself. + +2) Original or Modified Versions of the Font Software may be bundled, +redistributed and/or sold with any software, provided that each copy +contains the above copyright notice and this license. These can be +included either as stand-alone text files, human-readable headers or +in the appropriate machine-readable metadata fields within text or +binary files as long as those fields can be easily viewed by the user. + +3) No Modified Version of the Font Software may use the Reserved Font +Name(s) unless explicit written permission is granted by the corresponding +Copyright Holder. This restriction only applies to the primary font name as +presented to the users. + +4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font +Software shall not be used to promote, endorse or advertise any +Modified Version, except to acknowledge the contribution(s) of the +Copyright Holder(s) and the Author(s) or with their explicit written +permission. + +5) The Font Software, modified or unmodified, in part or in whole, +must be distributed entirely under this license, and must not be +distributed under any other license. The requirement for fonts to +remain under this license does not apply to any document created +using the Font Software. + +TERMINATION +This license becomes null and void if any of the above conditions are +not met. + +DISCLAIMER +THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT +OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE +COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL +DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM +OTHER DEALINGS IN THE FONT SOFTWARE. diff --git a/nationchains/www/adminapx/static/style/fonts/logo/Quicksand-VariableFont_wght.ttf b/nationchains/www/adminapx/static/style/fonts/logo/Quicksand-VariableFont_wght.ttf new file mode 100644 index 0000000000000000000000000000000000000000..c86807352380ee0dcbb5572aed915bd210bb9a71 GIT binary patch literal 116832 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 Qf6lyr4c4n`@ADw|AMg`Ah5!Hn literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/style/fonts/logo/Quicksand.zip b/nationchains/www/adminapx/static/style/fonts/logo/Quicksand.zip new file mode 100644 index 0000000000000000000000000000000000000000..cff48755ced0792d187cd53f403fe0f2d6742e52 GIT binary patch literal 489827 zcmcG%37lL-wLgBVZr`QXnVz1Ro_(gLXS!!!dZwpmpUjp?GD#-MWG4v;*+|#|Aq3DL zprE24pdjMTQ=ft?vI!zl0TFO{DEe3gWD`+wLr@+H)BXRRs(WwuBm?;W|K6Vu>FGYV zZq=z%r_NTVs#Z;_u0n|T}=wBIN2hZGb=DwZns}JqldD6ihr|xQ9 zzHje|XSSVr<{pLNaXvm=iZV4Zxp=5$FsdleQdQ4St(uvAa_p2g{QkS5r0$rST{rgC z%Z~g?Va3-eirz6Z+n)HqpEi^z%00hF`R%i7N7tTn#lE{0R`Dc$H|;!S$AR9i*?vVi zvP4l#&)yve4^V5U&xgO}-jmPS^T)rO5>ga~-`Cx8;_e;0N-uS-Qj~8#fag6Y0$?7{ zj-Y<}U3cOsXP*6o-+%UeJbznJybqpy+Rhz4fAW4_QLcYRQM8v&*>Uy()uawxk9U-s zQ+J%Qdpvl;4T|!>7DeG-J#gB=GoLu*2M!P6aoBy}jNJ!L)Gz%Ip05SKP*kM>NnTxO2=02q9d;`8KT&YD%d-(S-JXPqws_W6(SUi?u zJw08CVzg9LC?HHtMZ#=qj(c>EhcV9ls@pI<_{)55!^_ieeIvjoJp7|`7wLw_aOwPN zQ#16r{i^2n7&^Zwy`-T*$z)`E>KYysBW~~~m9)~}z(4B1pHZ$*obs+J3m)eJuE_WP zPVZ$7^F6BlUYx(Ayh=2bf9Idy06U`tRC*O#RO*53)}gf*NPb=3c9S;0l2mTb`n;rO~b{y0#4*W6YOA=1_^6xotL4T*7>^Kv> zv*Y|lr#$Ga{Lm@C+=0KST$9u8VhgVG=l)M8acyRbe;8cbrKB+9YGcXnVV(*bEW98C z8@gUpSlpFJ38n@|>kQn?ha;)H(yDxqJN;JP0*uYQ?(&w}qEf?U8iu-9yfJo#~7h@ejJCxHB2dh9gQhBz43Hh8c;EWVa9~k%SN@kyJ3g zz*iJYh9lRp@t{AwSoaz}l|ABeYwUE>=QGn^G<4Q2zWvzL*$(mT6Q(bizWP`B?7xcV zE_Oo>x%gF@%V)a1p7diL`s+^LEB^Y-8v-Ov1a?CDl*``m1Z1tM?1B{72&~JLdZiQH z5QyqVJW<7q3IiN+HwKB@)1^g$dsh#Ql`6i9VT-X3t=b-0)NaHl5|uT*%bL=EBK(*+ z=k>fjuxiQR93=A8_Lk-R$d;9jD-teWxNW?(W2!yuGx=4fuOXEFL69wiB>ZXRH6i&DnS+G21OFw1 zW)<}S?$i_K0)9+=B>7~0PoX}Nn6e#`JQDsmivtfX#_Wq2vmj|mCPoU{%O0svf<~#0 zM`@VY`=?KxsvWvuclvD_n15|KJy|(?!sIKu&X4r%Ja1zAmyVYsF?aIp$JY-X*pM*i zE(EVo8(Ks?U5SA&6eX^_%6~3s@U(hJ^c8c00|($n&vpL5%Nyid{EI>-5Y*sLE1wso zcV&8WN*nw>BUKJubFaeESGK`4O6V!Ss&< z^w*pIfkuBNeFgrYnI`z4T~OyqNN~fD@ld_|wXNTGTw4Qy%=J zd|T8v|2^U-`QB4_kNAn&nNdGy!Q))O1s`6ZoG$2l81>rkkp}*ws5ke0O?~>9_n%;` z%B@!Ynv3|A+RfJgyz&Efq5WL_-7(LfRW48tI?qW1OT1rDF3X{VU9Z6(16O6~nT0>D z{8_@mKNwf19?=h{e$vdc{wI`sa_CQ*S;B#@(9CjNNHa_Li^`{R+9l0QaA@XpJDu9? z-o*$?w8a>I_Z^=S2Hz!4gjNrM6Bp&^bbCI(8)rfU9Xv~HKg3@%R*XEa_d5T%viWSTov&eQolQWB26giL*tk`uOPL$EcLNqCkpkm zX?j>VUZc9$pqNYd>(-&|Cq{rK|NDJGWlGdiCX76Fcf%f%^%Mhtas! za*8Lpe7OL`Q;RU%jI!Z**fEpn9Eb31zF<{ z2%aHXen`w^34aV-L&08v6QqZ5`Y4*+g=_i9Y9EvfDle3-$*|Z@2fC zxhUj*2Rg)?nL`}?20d1aargtQa9d`aEM%O7Kc)Oc%0|eH-10Ax?v^@2xlHs*NjY>Q zR=P~m$?f(PWr+d-ix>)z`RJM*&(6N-P(z+FG8%`@Wt|LV$oo%%>UbteW(P4Fgi+yFL?TS2nY+=yOYQr3moc zl_6jG zdu0yP;4dpb&eol6lMjKqs{}5G=aqj+`H@w7Kfl%~A7`5gNAUEGywQ4oNzQvZf8`vK zGYNdu;;npPzY_nZQbLv#8sOZJ<7j5bYXU@iSl7IU|6#75Vz3h&>OsGGpZkuX97w+&ul~d>SkdKLQ7F9lOEHnb({zG7IO_zr`v4I4icwAD&nKC4ws~^OsB?!4>dM zSSM|h8s^rFl~l0wjR>oNN&(g7@p+Bnwi<^VMwU$ZWzpD!7#qYsUByii{$dVQa%nbH z)>IP~k%nEFe)S}WPQ=Zili+5=u*EF6WFFqHAP$zLBS}X|muHph<@myP{1EDEvTS4U zIP_*Jyx<6nzKJ;3;~r?s{{BLyr{hqpZm-|e5Wjs&r|*^14}0BiZ&fm~I>Jv<=e{f% zjfQ>hDwFS~pK~8xA72791~U!P@)1~^AqA0nMo|&R?yTZQ_dJ8Mhd5heU-W z><(G}dB%jVDB&#*{5R|)%7adMyPh}~a6xZ-T#4THd)em;~>?Y{~w-h#4))smEE=+l{(U{Mkk@nwtGt+i~^I1s) z{9LY?USYZMusu`lzd18jVE!jy{s^LR^-9vRB&}5>GMV6Rz6ZXW43EYqbQtqqeePbD zm!D+gXaAjtXCYRV41Tw%nx<*Ytz`~Z!Us@_P|iclxr=SQk%oeHgC{MNDfc9 zJkD4OxHXo~+hge}Jmxvgy>Hvk)$}pXi6{ObEf@6;p&<|}&)Rju0=M18qeA6T1k_~! z7=Dv_t&q=m?E1h^?bNB$d^!D0U#r=1@}KH@h2gq?@`RD9o`c5;tMK?QZJ#)AXCKLa zSNb`G7u;SweV^gs@7EI>4h*gT_$+yD5cc)JtB6)DSVBY;bPA(dCJtOR8C-?L7s1sN z`_{F$uj_l(C%wAw|Dd9~2{Lfm=gkH6pFBEMY5mcy>tu)B?6~Uj! zsg;;nb%>Z?6<8xXTBiSENQS`+^-u&x5SE!RZ0FEWF>_#3A~Fzb zYgp3e&hx;L;ckx}C@-jN?J9cR?c26!r=`|Z?`Z~Kx3@o)^ z4ytC&*id_KmB*L4B>ZLNbs=wMJ){pM`~~Go2}islxBN?3GqA0w6NLXO%fG06)+&F=yz;LoUj|-G ziDv%6UI%v!firYOC&*yzO4!kfHc(NuR_nq< zh8u!`s^-y-&YAYc#(-f2^pa?(W}vxkAfE59Zd;seTS;*VfvuqGZIY_64oE9Y_+LOJ zg2xec1soFVambrbI?u^UqUSn)`mAhiZ(Gn9LB%`YQ?3>793;wEPYp!bFAuUnq@otO1KCA@rv;?)+P5wf|^?&S=&_ASzelGDs7$hgklxkqnUn} z7L8*>oBJAqg-s(JkyKNeT2xTtkB_fyP0vyl?557`t2(Mn176*9F(a0&sq3iC&x^E- zv|yCcN3s>Z51vHClOnuv5k(9~svsC?77GT1iwkBIMuA+ZWM_{a+T6`CUAOYNJJt4? zUA=vKX4=|j_Vlegr(Fverm81Dx^3dD&0Sra&+;UXfB#g_bT;n-ic#S;j zS+{rbnNw4T7JEHzU;oa5!4n3&v<{YD1=!+4Q|uO@mR1e!92nR+NWX-X!U*pb^Y|gr z<0o0Ql#T*UUXFx6gD7^kJk4JTe;Sb!q7co7y~4|p@TUN$*(l*8H6{E7ntcwOW}}2Z zrrc}64|DX`fxpCpIpxWEm*t;Op3A|LyZ{M*UiqC|F#9Eh=x+T+4tw;MV8+D=g%r%&FLq30TSqO&qAq?d@eVj{m2b6J53OIpp# zVu6XDUsjVJb%#7z-WxulkG%TGP*Z+K9r9On&21Q_wYXIg(#jZLkRyp(J2x#386}Oi zRS}OV7r3(ga;?+PUn{@J>Ay`+vYd11C@g15X92gy`FZre4l|u%Go_ejWQUP^fZijO z&C{Z5nMjp}?d`vU=5s&H8=T6ZP;^S9vgs;5$bC)rGrl005 zB3mgAhm4R+a}!_Gt6}i`mU}g%v)yCfqk0X^gWq{@Nca=R zw>caRFaM7F3GU(Fb9>PK$-05a?vv>T{L%SUx{APwK6fQ!j}l_(7RJxe4WggGNbvph z);zyf>TAMO%v%Y63b8z)MFpJZr-VPoMg_f(uICB1U1YCRW#)K0zZdc9QfkZUbv)Fg zCF~47&=!I^MB+LXLV^)5H+|Y)i+`u5YkV5}o#r>aTDk-V9Rl`$Eqqzy_rk3*(s$Le zAd2*c}C~=ip9tD_bSt$5z1qG5gS!$Sl#JIIcud%4AF|9;meYSi2RVv zBjIF+5M1X^9U=-!C{a*CA3K@GVLp`XqVEd27Q%`|53JIZX<@;sXE|*?0oy0rCdoaC z%k!{CDThtgPx3&v`rI;#2}mf-5(#~F6;W*vb+5)8z2dnSkN6Eko4df{GfXe+MU9{1qOWLG)Gu&<)uNZovXF4n zArfw9Q%E>bOv3FffvSv6S4=EdzefK$4Lx3OlVjoWVomz z8z&ngWUopuF2AOCdCT~|!J&O)&CC00^1VOm-Iz#h>`iU#N^E4WLnI}spO^V-#(E-) z&z@d>XuP_2x}_8$@pR+B-qGRR1AV(jNB0h(#}cFKfayO(k8ifLorFJX!F7K7Y0Y+N zb(IB;@!Q{@goL^oHpX@0>8a?Q{Z1U?^h3o1?s|uJ&*{OtH$k>v>(o4*t@)fN-n{G7 zGn{98v(MgFi#NZeoThxse#)~wxJ~(p;7{3uTdlVYeyegp<{x%>)#uc9>)EeTACMol zOFX4M*l*c=IQs44?Ar(E?VFT0l|4?~mD#87e;ZHlQs$Hi=V^uYG%hfD6}|k{-u?Dd zPSH^s37voYK-__T-GSb)9gBe&HNsN0pb+guX{Yop4*Vv|?~t^3%&FzBbKiDKKkPv7 zKJ5($`e6rp=gzp`+$r7w&dpP%DefYY`C^VUN0SaKpvJ;{Z0TZXYq=&WEN_~%I;+az z+<(c?8$X73;v>vJM^*n4Wqa74LP&bY0#%!*-0}SGq}h=&_IyL zAU%!wjKvhy&R~rgNz2EwEaDXAJ!y5zq!C%JRXi+2o0F2#lL`ko3)J+rmes}jDkI~m zJmif-Jrs?ssEkgwVV!{ew@2$+)m$~!8|~aNRXf;`XSxlKt9r-4^qH9R{Y|Sz;_!6R z(Palm>yo7{vm?IZf}xFE^;dS*UCa)*v;`}g!fvmj>S}pc&D`se(T=`l_3nb?lKS%0 z+TNi(!{Jylo*P=j;G#n_YtI@f?^`!u`YZZd>N-lAhe~V8M_MJFK#_0r7|WBAg&l|W zMy||421faB92qDWfmTK0DG?;d`U>%3HJnYyq4d!h8yG)v2y!nn)ZLu#_fq@_+~E&2 z^$b_{ZAc_G^!0A&>e|57@k7g}&t6U(2_F)98A_bwXU)kk^- zW6pk!(Q6V?Stt|?R6HwsWq3=BP(;=@ZNrO-M!iWOKg_#Rt=&4b_O zaOgq8)+zsenDT);`TAybSrx$Grmo7K*(%13R9i1#jc;5RByqO`58UUEjt zUawtXmcyNUM{QNzx4dO+AB|e$bbl?h6%CZoR$UwUOIR$M5ZbD^cCxP)G?_j;A+#0y zpy`|I?cXyxy1T!B_b4bw5y|dM2Kjj}!5Z#X7Rxw>mBd>dM!X3_M(RHiu5yulL4vqo z8(UC?C`{EeqKX{gJkBZ>?H{fiS=W(V*&PUan#PV_WVj(A__9-$wG@u*8%=KL?%L3k z+}PESsT=G&a0v zFm=LE-;Tk-J;U6k>L%W7^v7IC&dF12d~!tiCn#rt!K%kd-2%T}j#Mw?`%4b|KILM2 z1o>bN`~fz`ju-WiKFF3oqWsFz2bbi)b^e1>Ej~c5$xDiCm2z7*h}Lg|J!;Q}H!&NK znWx$E3KC7oqWHTAF4jGLa4bi}oT#%a-$598awSJ>Vc}wN?3;q4kx?Qz`39Q~>`k&X z7{Os-oM>4mfD-@9(zhuWSfvl*>-;sIjdC z)M&p$R>FP4(#d@nHiG@`K4~NT&9)IxGi(I==^<$&ykXl2cnTZAetM3u5Vl(u0$PQI zAfCZO*k@S?fWktspf_xi7Q!2V(wvsGxRG+FB&PsQvO~fJ<_@J{Iau(Uthf-#j5j$f z2dnnG4qBE2pbtCHyHB(%2S8ytSY_|r;^3$5w4gB#s5O!l>lXE6&0oG_Kc&|1QAjRN z5w^;5n-eL`Dl34=QWji^O%p<$<|s<3P8ah3uPbh<@D!BhyNVKJ^BH!J%0BKT?b)%e zCr5d5UeOJSzJ^gmaD>+E4@>-GJnfBu}SCxvfljRTPS6%YYQtbvyx?}>6gEo~@7Xm&L# zEZr_?*mV8X7b!)aM@6u(8d<<@@7NX!T>i!I`LLSq!Y6Wu>nrDO5}J5+xc@bZZ2FKY zBD72|FG0^cl~K~ZmhVHV7Oj!xBqe%rSZv?GZkK>+0egmo}{K zahslCvAepZ{PxFpEjneYe`_+iwZDH$PtO*v)^rb6*6$M5!I~8Z28(J6 zOi!^}^%#e??PSe^^+BnVVp0e7?H(WB-D}I}5+Q@Io&=7*9jn(;3Q>PZ3Q72F%I6%J zULs_Mgx{)MWoch3|GZQFJIc>9a*h$4X0R-Oqw*VT1{0iS8Nqda=ixZ`XBM=34l*6- zwD64WR3_x^ST;;{YVs5HfHL}|nw zeoS$P+03aqWyza)KxnTC)c8?Bfhp0`8w)bBr^@@lF9=)BU;BKY-2<_v$(|qI5_7(2pGc`B#uZ9k^6kptIGEV^dV(4=F}#tY6?m8OC|?zvCf~jDYw!d!uOajGll*c>hwg0dD`m*ZDI37LUhzD0nc`={ zC=)sK%Mg97tP9afu(z~Z zs7gV=3>1hD3)BC}8MkkJ4ZY0#Dbuf9Eqb{LJ<<7%v?w8<7qh?9iVSpzaLuG|W~ZiJ z+%{5ySesGYT1}FqzjYWJ7r1F9v6T^95p}5cNBH#^1zG#eoLSIU_-zjK)}Oxtt92t9 zegU)7kNE<<+Xad^1knwN(*McXxb8QN*Q)-eBll5G zLNc4#Kasqur=RO!d3xF~7%u~txIG@?hmDy@Y7>}j5u=(^uFu?Q?)SSIw-kg;kBivE`??O%-FGlXk`c=>rL`5He1(+j0asJ%5 zM-oo9u=t^^3~Pz6*lFX^s>YDv&C4$cftCoKYM@tsab->lM6|T65;i4L*cDN2>u}iaC8*PU$0I zn#~(A7q>g5X%_xiymceipQ+_%GnXiT%6aPtHzJalc?)Go-WMn})@})8@Y*&Hf(RkG za%*v!SNHhuRuTROx7_X0586~`+pzMctwjO@bTMDcc=}EnPB{)z)~fyB6``4n!v-Q4 znC0)HT^7_fMMB#&^llCMHUH@9*x<@~V(!YJPN*IFR^H#yG}CP(RaM)!l>KvP)E$a+ zS9fnnrGHI0vFetk&CN?&(qg~De61vz@RY9TSh5Bbg+3vy!TI|HC6BN(V4X`H`3Go@ ze@{nBtb0ag>It#Rk*=1tDI3Mo*f4@UiS@}e5`ckQo24UvA%2M|ae`2hgbld%4?;Gb zB(S+zxl4@ea~MWQbZCQZw?jIylY~A!y7X<P<-szQ!iQd3Of;) zo%&AWPu;wXQh{YOB?3(HXl=YcYueJJEj}g_II`%psl^9IM-MEXI&D!ya_iv0ww|7C z1A|+WZ!bGMIeB>5G<{Bw?j9IGY=l0E2JohML};J0End6Zi3>+Kc3(1r!(Totxj^aL zuxHYaB1iZ=R%wcTe?)MPs9hz$mTG}deEz>MDmUW{g;Pasr=o59E%J#U2X7tyc1`x} z!}Rt|N=7+gu2H`=*V*{|H_JS}kAxe>jvpWiN?Q!i;AN(zf!+VAo2IPja4 zPv+3_IR}21@&QG6%I|dG_prgtoC7ECf$EQQ0SA6JE8h{kPm%u!x8KV?KcYO5(;nG| z^WVS8%KefwA%A}U^ShJ}T3!cn(9!LK&L@%GDBr%*!TOu1HQSqCSa6-+eO8tmuW+Du z?srCo;_Ctfc>Mcmuj|do4xrUQIXZ`qdP>y%A?8pAGJPw>9D+osw|yHcf{xg_ScPz$ zMlrJ}3g*r6Q|ild3D(!zuvvU|x+rqz4)9x`C1;n%Pu3NdaiJ*u-G zzoSH`qTNv(F=Zql%)Mc_-LYt7XK(M$k&zR69h_imy(dn= z|5cpSiwLvyrxrkpvzFgnciUkfxaUqlZwF!DYla;A*8Z-uz*4s z)nb)Ggwqhclv$J@Ov+$}7SzjDry@h~*kB|&(oxa0JWNl2eh)uc%-k6$446@ zea)q4(!qI+_pUX&am?ipw>G#iy99SMVxlN zf3EY6h^5HZZ=IxQdk4lUTbAtDONAXP+a)`eN(+31CDMWM$}N8cc#`EJ;dXhlK>)Yk zgLJ8YeN=`FmT}5~B`@5{Y!y<*NK!Va&>k2d=!)DS6j4ap4M*n<$;?=xZ{*nkVC}Pz zr4wqMPNfz@g4A@bXdT=&P~4X9)-^5P?XL;7tV#B)Zt!5qGTK-$+=r9e^0;R;MM#Gz zRtgjOSJBl=TH`~}yv|myD=@Zwc*$vtbdQGhTU9sAu5H7MH+Hxwv&zSfXk%V%pss(s zCLPC{k5c*y{Ud3GHAGg!KL#&34$n|18*BlPR=o(SiVRE(4#7%xe*NzWhu;I6Q*a1L znGcJ~h*k&n>S&3N{f}Mym5>Mj6RCE)yoTG)f~_swYtkx~)gnd-jtn^J0h1&=K_~g2 zEhyHuV8N&a&m1KDoYLQ)CcY?uo!*uMy{}M0C+C$-ZlyUa+Ea<|s6B(Z@D;TR`E*2Jb~DZ|%fcT8#sr7`y_+J%^Goaew*QkY zOIE8ad+S*=Lu6T6`;pLZ?;*J*p|tiRp*O6zy*%{If>JzHW{1eqWE)axvb{f$gB{s~ zvX;9JQ2ZIC`HZkMW$Am^1;Rp=aFQPcCkqsC)Mi_#@V;~2%RaX)RQbM*MfSOEp~~kr zHreO4g}QM2wuOqP?0I3|NUXnU*|ic*vRwA@?%j~(7X2ueE}?gBa;U1FU_oJ_ZlV>d z8*uW#cLe853C3o7#>!5r>F zETp-ysIIu7G?3@g5EF&vd3Um5%0pTJd7lq6=HW>5Oivo@e!pqrJgsz{A7_$1jWv9EA`9x~51?*D^h6v~ zN1doNu7>TpW#~8@ii%dnU;a(g&)DaXssz}>mL58;AP_HX9};>I8ISy z22a=PhMyg$rN1GX#@QFhI87(ncKl3#iTzaU5SMsd1qwVSXD(9_;muIrf!`M#_^sF( z`ho+$(t+QAQ;h!Tz(4Q6Z&IGg{0ULlhRoaSJ1h-fpp-IsLks4Z;xIlL4Mr+#Jk>Mg z(b+y1l1a?;)z~Ix_*~yCxlU)>J#LkMv*YGcWXWDP7pEf!2 zZdF5lCHjn8aBMHuSKyC0vQ7-!JROH+HEj8SmYc`Kf{WQt+}`2+6KzYwhUs$U8Ntcs z{tcZx&^Hwuo)M9}k?Gde-CmBEetHZ`XvJUwcqE#qqG$WaveK%~%FgxutXGK3ho{!I zE{V|(DeDL`;>M>X)oEp3a4*401`=H7w;qx*kf0>(B=m-LQrZ!ebiSt(l^p+TLgy%1za8Y5#(W#wyXfZ zUx?M;81{}sN3cGIvu-3@t#{yY2}f*hiho{UaU+M#XO*iE(QJC(E@!agshoVyp2(?Z zit2}~xsh$R-)E0F@8do3oGYd5Bdi(wda9LX5$~o&VIhUY4j{{OPZUPv?UiWP9_#F2EVj|)<_8?JB@5n98QZFNa7A`LfwhhI$m8j4lkh} zpK^P(0m791fj$Q{>Xep#O#|NSe(_5}hDD+KH7$*A=))i^-+a*raA!8FN38B2lP1Mt zGbeU+oj9{sWTw>A06K=7y*X_ex)W=V1o?2hgR)IAT!(|we-e|}Afv&Tc{oO}%8h+h>A%hot z98=kom#oN>)q2X#I+x=H9K>R%6}pJQnZTf~cA0wda6Gjs*}t*PD9B^jP3-z_J^ei~ zP@Dax>iT72#FN}TI(EvU<~7x->f5XuB_$o}Q;QI6=017qM~8 z3#@2JPB+u%!y%ybcvEAy>PQ%+16I(&sKQ_i+({>>rRYQ|!LQDq&n#VHG)^`nM^Z-I zS~Ely4mlhiZS~o1~_7sLjW|}*ulNG5Qqg^wt_0xSr z>sp&mz$y18%nNJHD|9^?`wH*~eG?joMf zq#^z8q8VaE1Lv_0ZRz<2tVNdZOMgM|#aUjRr~kmci0-*`ejd(d#lbhG?wUJ*O)@IY zYGiI6!J2YAaVU#B9;}d0w2#x0lY$xR?cH}mW`*Q2tRfOoSj={*;QZ9_0|VPqwY@dG zyuGSxV-lEGwx|E6X)+$4Xqp`xpS^l|OWX3+mgQ~7vEyC7#Nw9j^<7=-yIUqZ{VspU zXe_z5vvX}Swy5Lkma+Q!v6j)*tLL6xJKBN@CR;|=iqZKyM(087x#YATtU#E6$+mEscYEwDyB4w>i(MUshFr#%~1s*Y_|D%Qys z!}B(o7pG4AkykV9rt%og8l7*Ho|E=6^ttjDu!I&Q84KeA!c{6eyJn#m1=f&6(ol~s z#{AaShMLOC3+oJqbH}_zZSl%@s5BCo)I^*S;>fKZU)ElhUs@8@!DHUWNO5hbvMCx` zW6}#GUs^M7C~4M0vmX?S%gJ9X<0Li`5~-Y%6T`)MJSz|{Ep89GQ9Y)AO|YsZTIMeE zdi-3)!Ft5;SC9#0ZZC;=!p#*mU7mcufgppSxx95{Wo0;H$&a{}dJ@7W*yn^r>us9q z$&1xiwZS=IN=fFQ&>{7-KZxd65-ZkSpoAU%0I7%`&_j0`K5Y-XAr!jbK-(X|?qi5U zwDhw15jW3AUV}?_-L7H-fBG$ost7%xWLU(?d0D9EG^}C`-Xb}Um5*nK;>vJG0e14j zipz`S*Tf=~p;C~+k7$(Jdx$!I&qCKkP4#~KNAi-yB4(r-bKL6Twi zDX+6GXs#xl^=aYbE2Nl85FDT$X9Ru}8sY$y%a6&2NmOJi>CZmIvrdQ3)MFrLozUoQN9ImMf7uUmDuFACGWUf=W!QO`T+zUuqCV@RY#{)8FK@!MJc}tW@TaeaF z8}0|#g)$inQj+P!Swma7!680GJqeV_D4PIb76ff!hw^?vP)L+%0OU1g3m|Gvnb(z* z0C73>wBodheSm0o8FmRP=0A}*vriGu>~iH2?1mFK3rGU1Pj87bz*#_A*(FMv-zRYv zkP>zou1xq2Y6i~MTbC#qekUNnSwO1r)^q$JKu}0P8rWsZYt$Cu%&s9CqRi{afD<@d z^?VGF2LS=jvL3FoVs%{N%;~JDGHxiBs5obW=qw-!?pNLtWq`AQw4#l)+9z=qkP@!r z{>!v?*d)(xOC9wCk8Slo? zyAq5=jMrf*8!YXd6#GxUa{hCel;{;(fl;RS|H&`KN!)R8m?qz!md`J-p1;AjSns!H z>*p@Y$Pw?q!FSJpKS}Sqkf~!oKb_u}BTe|Fz#k3-gMqNW0B=-hz@x|#fmnUFT(V0RP9gIzoHs_2 zgD??sANXCNqi|7=`klqiMCg zw;(C&&ri2e@7XW%?-64!>LzTla>wpLWTJ__h48+UZVy!oM@JeDuqMbng<=xRnJHmr z6nvIBH% zK_rWZPB)rI8dhYZ**;T>^GyiN6MB=I=cDOR4F^%FZA)7c%O#espy^<#ni9)gT9WUc zQO;v)Y5t{-8tda6-pl0njNx3SmspHFW4%L+OQ;Jw_8Yamwf6rn*}tDuFU7vl+xR@g^+A)bgkBH}R}U-eJG{M2#X z&l)3$>`(V*Ro(2?u!3i-fo#!nH;Db*47xO3KG!u6#Cp!I_PckeD(}@?rkU?*W-ivO z1x;Kp(aX5H!;KStcOc?Nk8x*&c@w%pi`qi_K)d#=Z(!G?ZyjR-v6i#9WuK8BohmcAW&O-q{_h>Yj)P!G+@!C2cLd zy)`^JUZ67f!>piZMPeqtqz%z2Pur6CO!s(WxmOElSXnA;8tR;0wA0IcCyY*aj5LL` z0+#^{JAmQm5YIroQjP?5A3aZnA%+QATnr{eSr-UFSP4@^gm;maM(P9^Arl_#(grr| z&AhEGJTXzAa-?***y=U&@D4REOEL9fuf%7#V|vjp5A*F@G~GGeT%Z-;6vJT3?ttN? z4%BxyHdc_ij8mk?*bc}Q%V(o?9>-I~^*qoJz-bpzNQ9Q3^XOs+&G&q!icJU<3^d-p z*H6n(R9gdT>yx!H=~6pmB92=WW6$c3sgBNfdi!RpK6Mf5lP4rcVSil?`%C9Z7g>O? zZ5UmZ{fXQBu&#d?Lf2<}K)pzHIsi=RBi0P+_xieVjfi;S6bNj*cdp(OW0xU7HEm;2*WN=o~lkQS3?m~ z<-@L!SA8_+JHI~UHJ=I=Jsv`^3V+d8)Q8;@RG+AAp&Ia(RUDR5AagDIJ--qea>)I* zWMrb43?=LQBB}TxOt4o3ij|CW%n55_<79CJlC!|wJ!q`*uAB;l5WL~c=Y>s|8Z>8o zE0LMzbH7a=+r4=RkoC8*pHIx^*sAk4rB)2pXaNM+n-8qrec7g_ z*(>iqaK`6nqC;!ax5sA28>|Rmai~-W}4cjTiSPT>{->*`bn?PpO=21 zwko;dl+>E{FCIDP##4vRI=(kp5#XMnS=2nT|H6qw-??aV-R0|t&e=H_Ynp4zi`zC0Pwg6Oa3fm@XH}UUs}8afL|gk$J2)2YjWn-HE!`MGs1Ya)m_ddBznH?1Bks4m#io7mjjv}9A8>GpfNcdzLPsH}PI{^^F*Gxcp# z4Sn^@aB0{vRMk4#Q?+DyedosM!LW+k9L2g!BJ)FzT{X0pCD&!d!LS%c!E*Mx65^Gl z;7Cx>97dRu{oUgUeJohs6|L8~7YZyJh+%IotGp*FCL7pN^U!ktZ(ZFPC!#fwT>WAL;J7dJJgB7xdMY_txzf|Z4> zu}E7*zN=*X^SeC3w!D;@6M`hn#uF4)@Lz4d|>%RjKSJJvRP z#^l5qYunn^o-r|b#%$Z!-5Y34rp|@)M3^g(-!R$IvbZ7Wcj0P;pq?Ls|B!$TdX3+Y z%oW;EjXcNPsDnN5*||fZfhc*iR_GGn9`7{1bLt%X2^Q8Eapg>YZ(GpY2VpZKv^LYx#3nXZB+c z=ZNfI)|gQ-r-F3KW(d8fq>sM*@9g{c!FKKvUe9PCL`sK^l`4+6Z1ZgizhBF zyvl_GN#$*bAC%WlygvZlg1wde8f@l=ZZN#*2eIg$pee60RV?%? zx`^V1^l!zlMTVG#rR)|~jj<7#1Q?g2sZ4_*k_kRvKM;)$)W-&D@fq)iAKTr{Zix-o z#fD?Cq5ArfSZuH%Hc0=-KCpBxyo0U6?wLkR_@L?&-htI~PzW;RAT1!$zdHym^+x)H zbfEHQit=PB&3nT-0@g|kI@5liIbS|3I(PXo}Q1b$>CJ{K!5+H4z;7B ztTCWt4V-Dg-VfAIG)H4Kb$-vx(h0-m&eJK~F5l%w=9FvE1&i=_Nq6^YyW*;PV%JbO zpd<4w5BYm+q-M>==15mIrHKSu#o6%c!((HIS1&&_=`#H3jo6jB zwRLDu9i4Ztx1PAq%OcxaTDDd*)3^MCGpjzZeEjfAS5UhRYjSm6i{L@PQ`!q#7U<%w+*$%F!Q+!RF|oyD-?0~QVBWy);Su^A zNs<^#N@0fUPpnH`a<4OSu7H{Q6$J#SKH}O0JoYoyZk!N52CfxzY={TRQv|gud9Wyy z)h>?3X9cRrNVJtyF}vQLd-wgj*Op#N^YY>%eocY6p5$cT-2M>1@ZY)0lrV?=%znjh z0_KZgwJv};_|zFIVI&BfVxE$eECnrznJgucSDcTv{N0Yw5<9b(huF^oOQ{2sfpo;{ zlgDPgQ5nsLY(7lp@g0+k4lsU`o`0O^n@7qTz1UkW)(5yL43-Z6P_9lG8r zqA+w@AI)f^BNppS(K!S0j-H-Qd7w8eQjA_I)lppBY3NyNXJOqxVd#TC-ylxXW6P1! zWDIz{eH2OU^ZEJ=%}n1;z#+eHSpP!fL_@VUsnPv4?<2mDDG6IiBlyL6+^AL!NnO^ci5-dcQCMfz%^6bX*j_C=ueqM7aN zb*!7&=cDh#9zwb|8mEo2Z$a8W1RqW*h6da*=BzU?MUeYD-l1P^>c4Cm?LIynw$B4v; z^(&MW6)=4u#T}E~lI}46C{)%agxd^#rfTdXd?rDIx1-ITLZ_a%6Q;-jL8xdb2$=>^*OrV?&QpA0JKphW@<9 zCf@b(Id&DrWhp+J-uv&|Lq~RZ=J$M`H9`~0sm>T#Ok|Cb%~F_;(jt)^!6=c0kH*Mc z5qyaNB365mHb;lnThdAil zOZt_a%R+E&>!?Ozn^ojUqi~jr@)`c*J6Y9(a9yVol8#|XlyjRD!9!RZSLGO zm{_-LJW@Z@0*=PsE}S2H1S8r;dWB@NJu}Gzvy8QX8Ykv>Hs0A%0#FHQVo!w z$Xf%8YS*tFLn4~tHeLBkx9#}22j?`e^82t+zy%58HDWL^v0mQ{5TeoP$3PccC-`}y z+1##BxT!aSu&bJH#zs5KAEL8?SuN(S31=YvYmfVbII;?dhQK(5kWSBe+@u}lx#cyO znegu6S;KI#J3@8{`b4@f>w;K0lES_JKejSVzlF2#4=uLOBR^qqVCR6B!HhyOCi0(P zTE7CH;ud)>etH!_*e&v0eELOrgz)iQ(6fkc5|Y8J0Uohpyv;lkWn^Xw{1(39u7Rj9 z8eq>AETV6d!E}?K{WW`?n=p**TkqI*n)MXdqRao@_5M4IHQ}~Zcmh$fCah4KH5cU7 zE8uz&6tT${vX5>AG%ir%HH6ml?{setd(Ot_*YAFT4(6(jPZ9FyMkxlc5WNKB^~|Rs z)|hDuz9+7e0rsK1KEiiNe)?m1m56Co!yZS+Fw5To;|`! ztGIv2LimKpZ)(!&i}8-b~a;ol@yf3Ld*#H>HL zkY7-+gE;a);oPS}eD2@9Zy-lP;p)zR`{n^iD_Vi4v-!!^!m1vu#~kbGC|gUon8%*o z50M|H!~1&#=4>b7pZ)@U%I~T9XZmBCuQ=3Sb&I(#)(f&`75gmKo^+lNajJHu)lT{b%$6Ma}x; zLc8I64)nMkcCtIY!XtKDU|-Q~YzJyRs!mzUdO<+vZHdPh*cCW0LY%52Mk*%{u3p^V zfT0lg5Jzhf>BBv5Fb;6r@J|g^ntmENa&Cjw8F`xV1v*NNbt1HBnCYKkBVLDz{V7KJ zQ5{+L;EDt?TK>XI#d=mf@DqN!G{>yoO5G}HZ;7uz6b1rCMS(z}{rAhF3*r`Z_K&=x zqC7lG-))uox=k0^v!~b{>{|;_m||XN-fBbAW+PRczD`sxzO#?pwcT!i5xv;O?%-9T z)|R8GJilJ?3yWpKdZ9`C4h^M$2|MsH-J`oT{9&K4TXUK{mog98C*9=P>FHBpH8J=w zJJ_o@8Lh&eU#a>c_(A9v7U05~(6WZ6>zm*^`Pm+vf`OewF64lm=nc4i=__2G9oVJr zF}|-p2l*z!OTb{hRX1M>SQduGu`30airM^i`U;uwJ<-GT#;0QT8=lC1AK%g%69*>EF;c7aulx zh0=fa*l~?6&G`I zlbdPt(dHzMOS8{;!Uk&7U@u^2jcKH-4ZUIN_}<>_Qw|W%Y@kFz$ zRcTizAb+dG-oI>wlQIV3x;ybA3_C55A+5oZl+w7tGPRB5Lk&~S9ZO@K%VG}|_0-n% zM&qfn%C4B_v9gIsG+FdeY*}Y)X-D&7gv!bu^TfI;OMBzd-u9*qW8>?aS{fcLj@2&i z=~-M`Qtf@dB3d6Sezc*bY5n-vhNfu6^WN$b`IMxplKBOD5}8=wDo|1kbZ*K6jhGcW zAR8=|SDJ^hMv}vTxyrXnBxY ztEq@c_VUf}Z|J^1W6rkE*}Ncg(`0kAf&e>}e4g|r z0Y)*C^j{5j5&?oGUe;U4I=m&~?C!$!Lq@)VTS-X_puH$a*ND?Ez*CSNIG`3}qwdH| zBj-9WQmWu7w#=WWeZ1lz_GY;QC*lgVJoSpayH!6XJ!AK{aH=Xh%UGmAhq{Md2%=ob z_-Sr4pyPhBxlj3n7$teb!y{52zy?86H2;y!=iKigZg-%C`uPkt|2hX>%f!|})Aakq zWe#%{`f+hdCFgg!Xtm22B)Z!JzLuru%MRr~rSBd-Qy4GykTle$(Z;MOA|g z65j87GcWH=Z@}+O|IKs{x?FF%)E89zPv-z8_y$&|Zs#$aXFrJ;{LRX^xR<+uFK1=y zT4Y|{{zgCGHz8yDt0>>V`v9L2@His*cm44!QA;0Nt)5I!UmOu1%31sl;6r?jeORpk z{8o68^2{JQzmwp-YRrMtSwK|JQot{VpNjHGA~djb`Wj^k;QfGaQdBFe)*!}Ovf~CBGD~I^?E^I-S9vzu2SS5J^SE zk#AaCRaKh?+sgDtdU!$oU<9X+yPMlPa3`?Wz{()FI)U}DH~5P};=ij*yb7CM%UjE= z&$_y@()xNHD{U?-Yc7?aHT9)s0xmIK&Q|l=NUwuOg!P)^Hetm^EJz8zY7EOa?zWQr zXkCq8dL4OO)xy;Uf!Zh_-UKkJB5ND2s@qG_>Fj$a>2&rG zAapuAdscP`tBQyO2s;5okVQpQM0C_q#)Wa*hf&0RM`RQg#c>pOL_h>Y5M&f_7fJum zIrsMM4xlsN`+f8NomAa(Pi?18Rh>Fjx0Vd2F(%?{WRmC*~=g{CquQtXNf4yqH~~*AGh%svnXjII=3$ zdwIRGseW`T*6T-huRP{%owySx=ICpz0?quiHSWIOSmVC@x7N51mhXtS)ijoT<+QEY zh}t&T*8?l#Y}vi!JNbFInjyiFUxDEvWTk(qSmb|*LXhTrx~#lCL{w7E#_l1P{~cXk zcMd^}2!{XZ6mh-$6hY0e^j$d<=kfo+e~|VIh-a9h1>#|x012_EkB2HNu>x?c6$jCe zPp`9IXNNjwZC!X z<28w+15yhU-B}Ql`fY=BT5viB;FLxca{L5IGxHz_8ae*Q(yZ;w!c2+^+_yrz@*#Kh zoKk7&BUNtNBq(iA`Hc?Pk=wEjW%_diChHaWD_9!yAB7#AOp?g{z9U6sGcG%E2* z>QwZF%jj+!&2L2Co1cGBK7k(EhVcZISD<&r-5kcd-h1Ls6Mhf$g*NCRx_`ZAv#Cdo zcfEJ8;k}o zFA%q#RxfLV^%7sOL+j;>sF%?GsEuGSR9VU0UcOO1gXMdTevY*dy>~*_jO@fnENH~> zdy&a}p^7V>MzD~VDTT?nN=0e-mK8;-KYf}LI@|=k9TMMyE${c+q2@eeLqxs6TGB-v zBbP%kzJS&g8WQ}_p#A?zD++2&dI*|oLA|{SpGTUbaKLGVWxNGv2yU^e9$_)z(N?XF zI(gQb`XPq55d)f;Uh-&F0a=JXWQg5rMc@bwL+$ovO9aj+rLMeA!+Hno&**_&^;iX~ zZk6M-415*An3sC0&T*8Uh8Y(Ts*7FQ=}q10?4+1=enXxM)qOHX^U$uj_+T$bRG*^S z++(UXIF zG!+~0juj_Tcc}K3puSXVS7VNYcwNBTYFX6i2_Z92`|5E>Ev5o3Q9ZKj(lkt3+m*9k zb5>wFH!M6mrvVKxq67zhkbZMO{Gi?aKh$py{HA_Wszu3BT06BHP^puaWUjek{U*@u z)`3vyH;LZj*k}lF%zIc_y@~J9JvKTS+Km=P+D%4!DkgSWxk*n;#C$J!QKr7e$w`T( zZj(j2&ETXYTzHYBzK%;s(B&kwn>YUtw43VxYwadR9h1GZ@n}){7yw1ysZBwYoQk@s zb2B^TReg6#tHX|k(^{cVZO7L%Y$2rWd$={ri8bg@EQ<|8U%x@lW0l8_*d@7NHK}GQezGj?E~G zs}eHfVxq7RG6E+BIB>{wgflfMA#F|q?w||9rM*hUL}QUctUW3r4ZTjF*RZMg^Km4{1kbMRn;PrqG0IH0Ad7xq7)dX{*tr@AVbR-gO)ujpM zCM55PSRfX2J<=x-j7h%EdnjZI&7qQixmGNIEF<0H%>LyDUUQ-cKnS6Z;|_B>_PtX0 z(4URp%f(!=L|loJIsb-TpRbA!#Xj*baZ*Og4B1`wk%Qz|*(~SECGtvq1OGxb;LiE; zaQ5*AP172+QQAapmbO$|sjb$wXdh^wYF}zUT9hTq;Ncgz$$>FWxOT(`Tzd8IZ>|slF40D|0XmMQNxWsX_<7US_j(v`A94B!^(%^{Y z5!XhniC7!)Sj6UtcOv#i9CHe%)0u*EtE!y!&L-ys=M3j!=PKtK=Q`&G=N9KX&fU(V z&R-+XiCiD~OytXv??vv9JRa3Isy*s~=;-LF(etA(j$RdgQ}o@@4@GZ?iH=E&=^9fJ z(-1Q(au}foD#;%UNGj@IKGqEqnejDqLi-=2(%Zcj|*DG#7 z+~~OT<7UTQ6!(X?zr;Nl_k7&excB2ejr%%&K>X{@ZweLPSDx!nlO1 z6K+npJK@2EXA@pYcspTN!oh@Z60;J!CsrpmB#unHE%Cm@#}hXvzMZ%y@o3_SBzsbP zQg)IrsV1o@>HMTwNeh!MOS&#;P10YJ9!bWgrsUG(zR6pY-%rU%nUFFgr9EYN$_**E zr>sjko_bDdOX>xwm!w{udQN^|AA%3bxY(XPp^`L2sxD_yHyce>WQ zo_1|^{nNG6bvSKw+QhWKrtM0vOs`8Hk$!Ia^z;Sk%hETdzn1=P`tI~E(!b4!$Vke_ z%IKC+nNgcDEaTjaX&LPqmu6g(adXDHj3+Zry2IS@ZXB-X9_GHv{U`TZ?$6!FGK({- zGwU;(GSAJNoq1vA10IVf#xulomS?i3)w9HNrRPS^9i9W8?>zo2N7jg}%d)P^T9dUl z>#?jCv)<0yll4{B$?UM~`0VU#Uv}T@!P(=oCug^1U!J`>``+xwv;Us`diML-`?J5v z8JM#==gyo5ayH~_&Dox_Ket=%g}Iw^zsdbI&z^^CNAn8vO7r^W{VDIxyod8PR^BB?&laIlrVwNmWTr$>5UFB@;?oN?J=6mE2tNx00tz zUM=~s4~x_WzUuEEIV0lEsrYCFUPU9 z%nzuGh$3m-M=_*B!lH@AX~p(Ynh{zWQGpdNfRLSlzI_(bibm zxVUjc;Io5b2GtB&FzCTSpA9Y^+%|akkl91-9rDPKjYD1? z^6rpNh8!O9{ZKKqXlU)wDMMEbePQU4VJX9Ehs_>#>##S6?HQgteAMuZhTlJY^Y9%b z&lXO&9cnG zs|^}6|1d9IsIC^-{~Jd8s|Y89TP3EdThN35mU%}^wPJ6(dPk(d(6bY4c}PsPJPcgr zf59|_#Q6+p{4E2Tv_VW)fWE0X8}%-D40Bq#2Z0|2`UZfmD8Op~|DMSgIhf)8&&*SZ>l?&52L6A~{1g8E zo8~5wtKkdvsYVUM$$dE@;Xg80h~DZ{a~8&v{|z&?bG+K6qKpjE8jYEcvfngBOLgb4 z0s1Uuz<2xa8l38su4W^R-!eYX?-R)|c>0gi6L%Qb!_afE*oJ$ax8YXjZD_OGaAWf} z+{Da={g6Ka;{xs{xs!0X^H}%4LppJ9!{Pp6GSTNb$2xsXN}DUoCkS;wGi?b5v`I%jYV@ELA!SsH^BMV{mdO`VDasC!EHB|FL+c!R`Mie+oC7;Pe5XS`PC9 z%=Iw0!u%EHkDx0G?lMud1 zd>DhamLnq7Wg0+@F1j_fe*aR((Y!xuYM7r#=U?QJhxDWuH6%^D^E+mqedTM#YOp z=pZ===OrV50{#Td({Q^K&jX?t>SL;Q0rLDA$_wAIp~u9lOArQi6FO23_XjEX(t>c7 zD&X=^E`m)Vo@D5~k6{MkiSslC_FOnz#kzm9x+y)h38P1H-SHZZ#~tZ?|JZh9B_p&SHN5hlZH6IHtC~yq1MpZ zA}BvSPn53`7;-Cxp%YIDjxnC0^by^iX+)ht9iz0niCqXshUx~z4*g&I!f(Tw0Hgg= zJ1(%&`P!X+(0D8=1E;QAQ6kr%{(vg|DJO#V;!w|bqZ#xUo^u1Z;Uk7O2<&7zq4YE? zcgY?eA^$|~IE5cyw1kKAN}~$Oa{_EiKS5aRKj8VgSdKBrK+J0Oz^<}oav9DiTP^R9 z_uwRx59A^FCC-{jQ{7dms#NEwHuW6NUVc%%sXoyxnp2C`;NxoTQpLnTni9 zzh9w-p?%}$?;+mm*hi55>*%FF{n;I#`tgM#fTySb0slJx-TpuO*Z6PpulE13gX;9Z z{e70bpX#&NZ^R5JZvO@Q%lBvR+q-W){#NdLSBQNd?b}Jd_C3$`Is4ApSHEw;zW#gP z+IzbYd#{DL0A|VFxqAoi#om+MJ8Va}uDwQ4OB&tcr+Kt&)R=k=Eg!R(tF-HNJl?L= zuGenRZrAS99tq%rPi>vHUVBX2sBO|-&^Bu?W6tw6?M>|~?HIm!##oXp9!oa)#`8Vc zJ22n5-*PY*J^)#Eu&?bFFs;0`X(Ry!EjX0J2 zJvCk3sM2xI^6Tm@HCgph@2fQI)fta9fqAL``fEJqARN$|V?_#9on+yR1utq@FEJ5k zL7y$g<5aE7a9{HZah13iC-z+}?iO!Dj{Fln{UNaq`@8IdZjr#5&(bNQv3%JpvvKlo zcR3O#v<{KOq)T3i+n$!li{-8I3U!UV8SV4$IDPeHxf_yak4VBfteW^yB+Cf&+41N% zlSQsf5gwU<{yR++$qZ2=OGSyyL4RKiy|e;yG8f;H`iOzDml!Jhp-=8BM#_3ILe`2Y z=qH-xSTRlx5|hz4OppV_80Z<(#b0OAZz*q2s&(U1AV)>bvAk&_yDlbH<2!uqMhd$L!^p`$y4*JJQaMivRPV#?KZBxh8+mM&87I&gYyF;!PkK!B5OPnL)VSqTlVKvc>u zqDmHtYG~;d&@#@Kqr`c#N!%z`ia$X^StTzK*T_q8E6j52pSlctHkXN;Zk*NaMV7(!WdGGQ)AR@oG0B^EmBL>LUo~9qApd7)n)1;b+KBmZc(?X z+tn&{CGI%90eU(rssGByEZ31xbBbF?8?ayn5Ly@M2L$81U%crsPT54;(qY^$i1wtz zmJmOh1$YWo13&HXKM(i?crWDG z+7bH#Xa_y;r_19R@HZJ}R7`=-xxni2wIQBG*dsI_@6*K`q}B>(5xe#{L;Be7h|n|P za|z87l4mccJqN< z06HkfIe1bjOb2&Ipg$aX23mr7ISZ5o@C3`&N5O)LE=Y4O2=@28jrpp0h;;^*h`%U340PUV5&@#$*A!KGF2Q#-;^fP z@%_~;GbPoT7WBZ`GDo~Fb0xML9 zDRQc8L9a1gPM0%qBggx47PN*r_>$Zz=gIl<0@)@P$ac9^Hlmbb~jKxeswwU)avwzyaRRjx%ZRE>V>KIk#)Ajf-Q zFaDS4IiHsg%7^5`@)79mk3srAE)GJccoMUBGikrHd`3PiH^}E8?Pts9q5u3H=McXr zH>2-?-2mDF?j5(m;@&slv zM#z)6lf|!uk_rp3qQznr0~xYJ#i}@Qk&0Id zDp4h=WR)T=R;jpYDore9z0R#Nl}BZ%Y?Y&Oq0{9jK0MSozbUC#W-lP<>GTyA%2B6TLq1`r|KoH7H6s6&=dPY-|Y`gw^r3b4_v0| zRfAZr8pUO5fVdRB(jfFpL!iBqW;_C#=tyX@qtQR1mlEfQ@!V&fqsEJK#RPE_a}67D zhs=4qioh?nE$(-r6mS3+xFrLM*p z`wzI~<~nHYf5h6)@37YO2lSCQs@2e;Z-O3v3+A^pwMP9}-3DDanl<_dp(FeiGWuTe z3iRJdXbW+opZW`C0b?**^n`d)-3hJwZp>IkpnshxCP91dFScNH$5UA2@r-zuHUAf& z`CFl5-zRdQ)vXdMaW~C1IMe(Jbq~(({X$%>?iDYqzpAzBZ(@bI5A%X{bw748{HPvK z>!E8s#M;-R>M>|wPlzkAO7#|KUr&oD%yJwP8z8^`2)#X1+$?UwPMiDDE3Czg<1^6O zZ-<_?0c%tLj5Q}~)N{~sUK7vZ7N+MhuKzp6!!N4MxTWc3^$Ko9eiho@R*Vw<0WFX; zz_*|Sz775FU5p{#hZgvu+O9rQJJiQ&r`jd17rU{S=sJvNcBtLz6R}V2!92((;$yK> z>=Ao$qU&C@PwiKqsRQbBbx<8rU#P?C2=vUOxJCXeXq(5NX&zVKs_)eI>Ie0s`bqt) zeo-fEHIv&~E^M(cY@J!sP}874Yb%VW*LeDjXBj=cC3W@sdkw!MT%GZ5gsZ7C-fP&u z&uhd}6L@lbUZ2-zubJCCrEOlTy=LC@d95w89W`w;Tc9$DC_koeR?JG`g}&2 zmIkVWQC;d9tRrU3YcpzKy^)QEKx~GudV}E&-pCO%7NV-REu1^2c_A|cY;UlS2(2gO zM(A=*%Zrv4F=D}-<^?lCT1J@>bD3VbeZD%|D4k)WOon+&%8gu<7*v-VxhM&w&|n)K z5^05z`HG5&(Wex$uf$+fg%MSS5k-Z8uht1EscN*0LG`qa2_Ylkw^GlLuf$+eg^`Jh z2HV)sGAgAq^7_zDxZN}ao)y-yL0&f)2{#%9H3WEXMB300Irem3msHrthVr_y#y(cB z>ejI&gluEE!s`iFR@u&C_C%Z&WJmZ}Q)jlcwJeyqz;@R3w&n|4BF+gq>fx%4;%+Rn zox@&qbfZxkr3UL7t8C*rddK)v+IY3k)@(8a<BBlp(M^SrAe5H=*r=;d9vCIgH@Wag%9{|Q0Z$7pIJO--jq3GMh(&F4@j^&_5q)@YUV8aOd|_34dFb~ ztgBvMX{9ctD|8_}t1}4@zm?9})7x5FTIV#kPMtZ$Hirw*HYbGdrF!N|O6x3hkVBo4 zQsxxumO*Q2K(?0JTJ^Tk+B|>Wg7&s~^Jlc!T0^2M(|KQ_OJ_*wfYjDw^i^^*K+FMA zR;{(50nQ6$l+jX5scjHZZ`6Q#Bg%RsO1*wTS{ub)8mKWwjjC(3&I?MAdLt_hf!GXR z^+vtJ1aY(}J|FjhJ_uv^HWc)9by@S8r?6`QK(T4AR=jMTtRm zxsi)N>nU%vEeL6P6-MSODkB!0Qrf-}gHaVmR22r171fq@lFscR$p+ji^$hq*47OA= zIVBAh2Jsb*wuPZ(!%`Vi+90DL@T{~h4Dz_aAf(YCq#?j_BhrS($c3l#n3`uOk1G+% zkkSiTN-s2Pen~@RwQZ4J6Bh;95WdKi(u;H{y(H+U*X$~zm>bJ&OG2fzQ3|C7%Nna~ zOF4SS(o>p!b*VP({ApI0^JlU>Cwys3+q~kbt@GxFL9f+;_C-3z(LMvsk8$DC<}GY9 zpqUpM{uaz!%qYhKXy&at*fMkajCS2)>r5j;A|9F?_j-6TU-9IYf+tgrCzlSMTq<~S z--9RT4o~hq@vPQqZe2LHjcI0pXePj*nLwRp9U_|9AJNSIbeb8|X=aa1GeJ7dM)&Wn z2$Vr9pG0L35=+bEQFZWpy~X%jwK;GFNqYLBuprX24D+$#_w9 zqjVCyMpV#rnT@(e?Dgtn6bxCIG(2mQ=C`!XoHvyQ8|dw)vUskm8IyKqbn%qt1uYaM zqN;aHn>l?U!a!rwQ_-6Ux?8VpLCchRty39YQc@b;vY;J8y}c!XF>+j69yw*+oO!Kt zkFtO*!{8rQQl24p63b*c*bc4uMMz*calT z;DCRE1O5r-DvT!=H=dQ@+&rhun>#t2TOvV38{BG*xa*3W=PD8*#P~AbB;kPE7 zrAA2Tvd9_p=FM)NJnupjiy5abq)T-%>@C%mWjt%05CD@~=FD3Z^z7q0>%}HHJrSQS zJxYBB@jm7@+*vjAmg+qco?OShzCZ_G$AyRYTBn||Dy~w-OC`TVnQYKsONQGJc8c#HUdA#*mkilF%2OlGsq5g*f~4i1iHmLdo&zxiwru z`QU2^iJ(+ZjM?tf>%7+&(Dw`x*Oz%#1y*w!r=ntLpFt2S#5l zgD?M>{Zd%df|Vjz(ejdd1@HlN2ytnzW8tq0x@jFO0N&qe$4G5UzmmZ3{QcX zv3%r7)i(|uBfN8(+gtH9!-nq}_$G?kw85CU)MmG|wTh;>&26*AxVf|E&cq zxkFfEwkM4NJjw9^F7p=bYrE16%VW%bc^E?U9?K9M~VV%vl(@(i!7Sa;7_1Mb@%e9k~WBh0bil6bcph z6s4RH7277dVogG)pa0|FtN$JUVgEP&V>o5tl%M|u`0$N^ci@>oFm^x+te@qd|3&|S z)5DRI|2zLLx}Q+n|4s1Czu*5uFl-PC`e2Ak7C-+9|0g=cZ}0n$2165_fgjTx$cawx z5B@_~?~fRCKmKk0wf-Xk|G_6xIN<-$@IyK0Sb}r~p+J~WOlZ!?y>sXgS_8BNA~Bvi zb%uQ?7=!L?#D+Yxi^0Xv^pWC6{=MMJhk>;G2T`sY{fE%1LVkX%fAN0_gB393llTbd zFpifwgmX6dM>&G)_u$LEQ*&qd_<>hm=`XP1lPNb`_n$cw`7?w3UjU(LBc~?=QK82E zJic#qo!6PG0y0F@C{uQaV3S{8&Lx^4a+L>Eti^g@KxH@SQd+5F>EW@W*^4%7jIy z5HMs$z!!(|a}58Hb$U%Y9TOT6%JO!YZAftyQH^|W@uOFQ9OP4DbiaPuZ+(<}5@qoJ z4MT_h-{NggAU?E!Z~gxO&;Qo@QARzYxQZ2z$f~djh2ZId_JzW_>mN9wREq z=(P%akSHB9e%&8-Cz(IA@UIOlr5#e!Lt)c}hHU(gZ~Yo`eQ-@MQ|+yB9j zYa^Ed0*9^<=#8@zef-xsRt+UUy*+vpl=um>9Of(8dYwE&KDoSg*Upih{&uD|bhZfUH1`g(N$L z0KA6cg5gg07r=mKl2)fl7lYqM3RK773oA+Sqmm2!xJRNFZfQm>guDXOIe&ow0sL>d z2K~Y3A=4Km9DaqmD{2PXm)?tomXksAK|SAruy}@2tk*7`(;?+!g#9`YDv%Pw8mS^h zs)t6-IKM{tP@e1I$mVp-*VF0LokH=Z=gxp~-{PjRggypfZI^X%t9u3iUGH8)t;XHP zCRBgl!utCWd~FWJ2DSR%8wW(X>)oyH!R{vaIQIniWcLibwY!&MZS_jTX74mcps_?U zk0sK0yx`^$MK>N%c<~)xxNBjS!3=~MZop&kgvRflWV}zqa}EsUW3k9{4|k8jitb77 zX;{X_@;sKE%GP-Hm2U9hKnrh zl}W&!wg{{*aS97_jFjql)cg3CV)VZOm?oM@!C zL-j+>Yy1b{`YT+YQO%GA;kax2s2Biq1J*>~uSLewSv<{BezM1RZYE&)udC=NxTHj`@YU34cldv0v){|Ys zYss*74Dz%b=}O2J3)-XwGhQ*Ur5Sh2U_pRnJ;pA~y8-iG}V{;XJA^)c*SSb1f|x~oq`l-P^EXzVBX9BCfJ z&H}qQia$G6g&o5hw(qeXR$(PC?v!@|zDwQ({QdHN>|lKoD>qZI zO7nfl?+?(*T-dGhp-2PIcVf@NF06C6VIAe?5CdhZ3_W}~)*r@!gVoGI(xhlbI&sj> z9E@ZRMu`mk#h_&KkSkit9fuX-C0Hp~ioXQ#zbE+I2Y+eU<=0=NV@9$LJZ}KUJ4=w{Qmq7`a0qywPjg)CG z57tkD@0HB=-puzt%=f;u0$H5R93PKfGz~i-i0>0HOsbL|GI39s&;}EF z!GyL1AWsE&+8d_UGZ6N0%74;!6Z*o0b_XExm$aY#1zo=ixF)CI{Khz#+@$Z|?-xLV zaW=vwI|HtaLumsCPA1=yd^_wS7;iEtN(My+uFZr{3I?v$gg}dd3%Vi?r{O4#Nt^~a zX}HcYpwf>IPgon2F+2p5cn{UxwJS0I*@l_}4pY)W}K<@J;%6KXY~dK1bwA&&_WZXAd0 zX5xG%RAoYaO(+=OU=tUFh!#D*38w306FL*6zaQdGGwci#3g)fd#4RJUtmNeaE@XCdcuURH=)Z- zXq5>CT`88^IF`Fj*ZWN9Olkani0x+BhfF9))dmyyf(dO&eZ$DdJ0>m|%Wf0*9dh~$ z@zk>d<{E;~Km(A?gq$V>DP#CcfnPU;@Z11%3qh`26QX*c<4_w;g(!rpf@Y0CGpA)w?PD z;LZ@wYn+O!3i#vMj=^}V=M)C>OvSde4}fWc(V+<@G|q$wH<|s-FmY>5h;VDzwbjJ6 zo6u4dT5dusP3T$^S`A2#$2%d5Ee!27 zp@SxL)P#A^P3RUABHT3`c9MykW^P+VzB2xaGhtWvI!7 z#+lFr6Pj#76k-O4Xzhf9QPPl7LoDMzX-sD~nAFsQA3tDZut}&rI8Cu&Z>|>_u z%M2aQJUeq@=2XDjOy~m>+G#?2P3WKr9W^1u;{J(a=?LI5EhZF`dA4Ufxn?>{2)N87 z#-*FMY!fOpp<*+ZG85`)Lj6ss(S(MY&?pm{g|z1rPctusfmDKhu?bynLXcsGD`c1f zL2_l@%^_X@w4Wi!F9UkWgq|=V$ST8?LTup>ZCr~OGPdf_eL&{#5du{`C?7v<&moZO%vBgu86z_e#ZLGgFW6~ z3VR;z-I3xxe--Sf{o`S8_Ai1xmhJoe%K?w^$HTsYeXjR61OA);X7=*~?DIel;$fTC zHX#PK=}bF>Wg8_77;Opm-`GZ30$!)ShJBB2YrOzJpix@Ss?TA+$oLnvEWmGL9jp{v z)I8W@)rGJx!1qBZ+SE$eYt&<~*WvUwDek~daO4Fy6iRWE`Uv(*_!3U`Rj}9N&Jih| zQuo7tTEmxQF+&}IeV;Y~_E@$r&~gB`Y5A}>X&%^j7>i_Ur6= zsUEY2xqb04`&p-GmG3?BN7zr}1Q*H|VP4d2#=pw&7KtxRIGvB|KcfVZI@@d5&sy05 z%pHt*T2i_L&Oc!1gBQ;TDo+_Uq~oz+bAz%$SGSzD45OJ7$m__;N8{8&0;x1-pj@Ym#MyWft50VSldC z*Np~kDeSwnt+2Oi|6t7D*{1Ix4a9TVK=n=b(1tKZ4+TEMbtK#8vJGylfs`)JDnMDQ zwVe8URF`A}VPpftf8;nfvg<8WZ*gZP(Qq;2tJyB0I*(I|C~OI*Ud=I-aNWKbXM{-E zi^Cq*ZUtP+G2E$rK(K{quGJ2)O}XgLF?3`4YdP1oR0rUP=%1@$mA34r?S<{-RBEZ_ z$o`yq2Xk@>`(MJTte_SrmoS}6IEJ+xYAyR;%e2j)mMCYif9l}?v!5AE$qe@M8DVf5 z1zg1uN<}co1g8QUlmmmWoPd!doRmTSmrx913grcOlm%?$1vc^`7O@}dg(Xh+r8t*x zD_Fvqot&39n3C_A|8FpbSxjL!mL#_^-p%kl_W1+bZsI)7q#@4V#s2SN{9T-*TR7Bg zF16XrpV`cx+3Y%-!_Fp3RDt#v#IT&x-KF8XFyMo1f5A4D*K!WK9Ow8-`5?!(eNs6t=Tdl(%W)Q$<8r2LCfhf03^SSL&yW5Ie?p3u@ywHWjwhbujOTcY7|vrqMaa}m=#jpLcdluYBW zH*$*8*nboAa2oS)8uM@((>aZKIE{HejpD?~8nERwlBKYzMchby!znJrE9wuie}Z{6 zn0*drOdO|mh-o;)slUr97OVY%K$JvD(R)ZEUxa&2sx!>bVfx zhip^ncJZ7?c`plp1W(3T9JmfI&ZO z&@ZNIl40-)r_zu{F`X!f4a#9ZLN-x_bJCC^FrXPa49UHA?nB$T*KTL+VlmE^kvPeV zXtS}cv6gkKMjFc;Dg`l{#CIb0{_um-Y)ZRmXDQxb2tatvFUCtJDfx^PJ~GNu>1 zc402LxqQDZ)88$ajpwF>_$%aPR@&q%hbi3cj=|cb>79T%*CzD#d&mbJjc0xxww6G zwHvug3%hnRTv1nXS2UHZi(8_L+m1`?!Wfb^F0K_WZ6M&~%x4$TEKe~1HI7+h%tA@! zyHK^kUZU<`cs%T3Y8k+R?u8*_-RHiM1^ODNBNadQD%5fg`r^DwrOl>Mto65AM zGCiprgN~l85T^R1dp)MB0$?Fu=?L&+|#D3x!#^?_*-^aQ8PPf_hY0fo| zjO0dcO?s`MJt3n6)^qWmy|uJw4`<$`;~dr7a9*kl=VjfC5z`Z>LxI1S#mnLi@g7cx z+K&_26!`oL&hg8|$$bv2lZl=(W$t`=J=?d@9(CD!8crdQE9T6c-YjeB{0TXb?cr>X zVS7B=lh~fd_8hj`*j`M#yK$=BaLya@rwv6O=}9Y&@Km%y6i<1%j+Z@+XBAd$5*{VO z&tl`*&A?AE-k|}pyUTc{7|$f*iBq?+!wU73!zr8(IM{$j;mp1{BD`jNgGXMBt=k@X zecdS7cMqupawpIZ716aur;i(p;Om*MLXVXScOF^SjmN8e#ibR z>X)(p8@Y=D7Yk|47S>lPhEcy%HvCHErBz{k8M!dPPTI~m%mZuc0VQvBcx%H|ybe~KYv1Brv+%qW&RC?_6jpqlO>X$ZycnQ7$RXGgtL(9AZ7SKSo^1AaA!D#U4@57vhfJe%!m z&(TQ{Ne8oUljd>+$J&@MAIkJSVJh-z=iYO<tH{u9)SI0xLENCjEr&As`#|a}f>?ArSfF2RcCNzeLjz4Z}|Qwe+eO zyX;TjeRnKF3NqCBEog6UpE;$+$b~bf%wB-Ag1V1tnZ9sNb6atH`!tRx29ZIg52$Zw z?9;7xju3k$A=uslO(RBUytE7Or~QSfY8o(lZ0*|XH(@r7R`J-pBS!b|UEXvSB;Tfi z@IPrZ#O;zB=X@e$#%8=1Pnp|1zp8X}Pq?qc=10f$<^`lD#~}W*@eG?jXUVj|N$Vzn zX^^H5M9pYvo|<~CrAdg#a^YSv1BkHs)=dac@3}MPwlDtJvf2lCgo$_!{Z(`I%Tc3+ zSl=5W?cm(z#q+f=V$Aw!?lO)|$jQnskIzn)NfqU>WjQhd*aDevODxQh zPvqou49}^wTjbr!V(l1Ym3P=J9nH$3{+yF@@|v6+b&>7l12}nCjkBJ-(uxaV0i*oV zaeKim%u>1_;~~y%xC9orNLNa}up24iB3Qy`T4h$U9QdszDm*GmqVEs0MmrqQ7HL;; zPCS3UKUG#Z)x48yfsL@*)k@qK8h-K;n=L9jB3w=CIICALNNr9{n@D{C5OSGBzN6q*cPx+A-V?LbtK=}!T z3+AVD_&ukE{|IB3JIwgEpXR<-o`ZYw0(}0sllxd;t)2k)Az6(! z_q&H&QA;K^B!%5HJd z9aF=j9dTCqk}b+Eo1JlS&W@+U>@tI)ec^UFn4vo($%P$DE(rSZ2u^P?eq2YiO5_Z;3uBzjQ_% zDI3tg>`SZdtN!DP8r7zAM-~n3p(P|1HZ9-8jX0dx2Y9#Bj{6QQW9!#K|b!!jsi(;h4X+aE0U6vY<7^BP4{*81ujXQ=d^^` zlN|YH`TEQp-nH9^8NEx+%6BB&!>tji^+WRqE}L9gF=_dr>RD=4SgI5Ey{D#k9obr2 zKd-4MJ;kOuB2w)!>9J+!uNXP}ib)k7@QkToo}p)88P$hdv5qI1r{i~ty)5&*{tJj| z6aPN?7veqR&3FnF<4=o^WS$=1PO*<=rJl|N6TcmrF^Y@JjArL!sb`d2Qs}j)woysN z`>2$PRSp%CylC*efjRvyo9d@>`F`w-lCl_iX8$;W}t#UL&H^Nq%s0GG$ z9Ezlt=jhmL(_0|bbb}rh&h?-Nb7zP6p6e5M4LtSXI(|2Hu~Of!wgPRwG z_(y%h9u?wt=WtY?^l&?Iem3>J6yJH;0|uVvlY!@Yc%_&@?%;bk_q`O3`rh3fZ)o^% z?TgdGe9Wyh`Wy%HDOdZd;Fyp5_S;v2b871nI^?0aH*72a0`(zy-On1Mb3+Q<2 zlgU5&y?;Q7;ou#m}=#3*4LQCLYF%}*=CKhJ0gjj*N?o zd@(XM%-(ITgFI4z)KqJvJt|gH@kzs1jT^UWxGO2~AUfEnnApf~!lJD19B_o>^+nu{ z2XMRDja_2Yj&(e>V;#Rsd}#C++w*H*Zpso zo)A3MNFD#Fcob4u@7vo@7Lm}qXbc4=#PaZl`UxtP;G4{{XDxHa%XlYBMSd2AO_x!m zW0YbmSVy+n6c>5&a^j0xMs^VH*a~&df)TgC$@qFoU2qSLh)3C@FEWZgkJ92{)R|Mh zz!?+cR1?WQiE0<`jE(^jk=XMYX%dko564EpA;KB+ekcW@Fi}G_(?1ipD{KNkXzmo} zxQQoCNz%sr*6}-WD+0?!NQ_Yb2WS+m*JZ5Fpj?SclW$a&?$G%Z<&4h^G6gI^%0I}T zIRtu`_)cNKkpkw454sCgPa46J227l>t(S4m=-4>$d||&SX_k10BhnT>pnH#o!nmlY zn93mwSu(YW+y=8}%xxdS}%zl<2dpx@@>E)I%!-)3+czJ)l+ zvRTJ|_?Cf{P{s_u`3hK;h2`RPgcyS`_iI#(^)TDBsm#Xs7mEA2%yjGl!t&WJcW5ND zb?iq+Nn)clh_^J7*o3vKz5bs`V$+#d7{rjo)_w2aN^u+j)~!+N(6OIBo`*bI{k8aF zwTZNsWU8weLD>s}1p-OIqLn8BAUSL;dA;I2`3W*H9%U5nOfPZA$Hqj4JDkZRJ$Y0V z?@(9jt;6B8I=Xw(G9sgR97I%n1}aiDl9oCZd!7VB)S|xZ0Toz+i*elV)DLOtH`QfL zK((;O5A0stSP&l>6MQ&dPuiDImsCt-qABmu61gkOfjrtq!PorE7jl1N|q%F zcGTQ>bC4MEg=jxzV;pohvI5-Q!fk5 zslPxi54}h-)rDkTa`C8x#_R<)%#l?FrToSAXsb*}lZg>A&e*ufsJOV3nlsEEY3Yb( zEm>`{**hlVV3}wY8+j5PKDzxn)p424E~VWjuZTSP2;%~}Gh{R%kz(hlxK(eUM(C5dQ=dzg=S1RRy>8NvMkGMxKU5p#AM|?$wW?WH8 z{bptI8OxV*MzUJ&iGuoA;)|EIj^0+T%23`k`5IU2oT0BrW0)pgJu*jRsz!D4O}(B) z#e`W;J{zX`(Cf(?n|cf>LQ5l^Ya299-0KPU(u|0yce<6mtdqi7p|+Kbzi7~)i^i7( zp8Xq^HjTKbq2ZztO-mc(#1U6CmzOtRF(U9h=c=JYuR4cqaF5gCx^x9|ZHKJjx}@W2 ze5m7h<6Cq0Gu zF#bbbrqb&i{pK2Jp^xu%{4p~=#v{E?G26P$j4$ZE-N+BQzs>H*FYDQQyqsS%9BCao z{v+HoM?Hj|9_bxAelOB8G-3~UT?gI?x~k&rS`KH8^*oEkmW|W00#^!yT-+sq{cY4Lm~{EUhZ=`VWKh?U={P z6Xk})HNG5?%JlCEKF<+tQ+;gCFh@+-$*tBXH9bIJ;Ms}LWPC~rS5JjmBO~x_smbn; z3j>q|o}m;n??EBUl)Jet?~vIcb(titUY2`sef9r(oNo+(VRw>UFGv2ZoEMOq1teGF_ki#kfhMi8klSSEgLvFK6Jq!Me2W zc)}hgv&o@jPnfMZ-1cn$@x9VYW{>Gj!vFlcCN(UZQbk}G%}+!4hj;9=N2s&Ii$~7v z*|cIf=%RTGT5psnvLWY-k(HO}W7SM>q`sVg79p9_^yb8U3 zj?z-mvT`wFG7xCt)@M)M>*IZ}x)N0pdErU{dZ3hm!CFx3$elT+OOLRsPA4qB&vhtKGmgGZgcC6Hm6LT zu$B939lr}}S*gFK5fE~QHlyPYiZ69-iQ}REP{;2R*Be^Y!cP7V;2uTR3E4l%ZQcKV z+^5KR%$tYCe+0KwlJ1=A{~q^vzJ~LWgcaI*zCJw>n4Lf+EGX76i$lUUnFnCR*uVr^ zvRh+aD|6`lQmZ}Q>2wrKsrC-(<{RbfGPtLwYI=RIxt{0*rvs{Yzw91Ex_XCq>pG%O zPR|yH^!Y5#FlU0JsGGYguedrd+L_tCwnzD>9)$(?mX%;n%!lSD(Uip12wwn=6L?6h?bvleUxamcntda8o7&YzJZG%MX<+jS%VGX5 z9%t7Jh%<((9W(4#Gbi=9NJG-&+SyKeRW8ae)b~O0T!`-pW-QxrGbWWsF2*vU{ttxW zS=ZNd!gzx!Dld#@ao^{5+(%5a1!jT$f3&>^fMZp?Kb~`Ma&PXvx#?v}lF1~KWRhvg zlw^9_DLb<}+n4Q)4hu`~QiPSNA}EL`qJo0LBE5+M(kxg|5s)gPQf$xX14K6S|9sEA z$;>PZ&-dQ%uWTmy-h0nI=X<{A+t2q@{7J0>CwvM9(>y_ZiUxD>86h%0i4-=2Ya>{m zzn?S=bp@H-tJ`#2)uMQ6ASpBy>tg+>y6$B;m%nQuIlQT-MsT}aZlP}S_?10})`vBZ zR$Q8n2^EW zRfR)V<-$QvKvivmJ=RuN-yCx~>zaq#ax2?0M$#3Y08K(pkcT(Ty@;XuC>SFxV@`iC zaY3wHD=?yz?2O@)Hx~uJF08TbeazZAy}7fnW4ftndPi}|@vTZg)kL*$!N&2^Hg$Gv zK2`15boyeS%iUkxF_mdvarEe-T`QV9yX~TEuT_VKx(_;c$&zyqf{R%oS%A8F4n0_w7}tPM$i=zRVr^2lj3*7B~0OFM99RS-gR-;wOw%jF=|(zHm6%I0pO$R6ryf z$GD$XuK|DF@;#0p15PW^fd2+~GJLtzKJiQg{#%NTRoWrmXuzK~^&N_NVZ;^qYfPDU zKiL4r{m;Obnd>Ln00#UO%PZUlp#Hodwp8H16D*d;&G)`g8838nf@7YaWvHsiub(vU zxw-4FF=V$^o-g-f?)o!`A(Z>`0{4Ydf99^gf@m1oCPu$V&s%u?CCis8?PKRwrQgq4 z_HY}(!0#NsUd1`&<^JsC-c)1!FI%qSHh^(IUz zg#A(^QES_V#Q`nS7_E({iV@tJ8|O_Gyu4_xCu5vte9AaA`06>laJ|_WKZo;q4&yvw zMw%Y#34hmuyijPDhH-$Uh2%9;>iQDCg0LPTrqX}e+;+t+mfmK9PYo#VdEZur@4Asx zv-ez8MW)A@BJ_WEaPPN-^^}hog`#&Od~=j5!CN-7fGX)gqVa}c$S3;>g%ENl_!sd) z-6y;#x?y@)Z?oTK^(c0i_@WrpRJZs;?G}+|x81D$K!p1iejoeZa}-rk+1)~EDgG+L zId*7Gsy`|RcpY1!v^`PHQ+>T2K;H( z&arC1iTVxrGi)W=7QqGgBIo)+_^PY9U&a`B+J~rCIb#o)1(W7+sG<)JN}Vm&s^rhA;F=LME<0)zZ59D{!_Fo?>MwB8LkVUXaG@S78e>lskOp8F8d*{y60eY|LYK=cm3jomBiN;Sr$=IsNu|R` zm>>@6Q0;raZg(Rx?blShaDbKEm3S*}pX2LqCcfU|eK+7FKMeTu;Cm!r4LIS(fdAU^ zM^14C$OH7V0)LI*&;;_(OWcA6IS$ICc{dHit|GUX__RY&hCV_ni^y|u=A47<@8nL$ z5b6q3*|D7igGVlETGCw?@H_&gw{2Bd=W2YiS+8CiB!W1_9`0?gn>c;xvQx&Qor@Y= zXY1ZlTX9E!-`4KJ)`5N==`(utUGx;bRdAYHEZ@V)DYIrze{RAh;f|fLGF04dLQ}#W zr)KywKel|4U)_n(m7k%#oKHaKS@`URxo1yl#G6%LmR~J+Tf?-t)1uD%Xyh zyLN01p1jL)kmaWGRpCDKs*L@6hAv}_>_HPQ3Adg7ZuxF8QEBV8Q~p6CKyB1*{3?y0 z{LI`4=0BU8d-gbbc8_J&a&o2R?A+Be-p18?m~NS@Ra9p|jI~DjI%O#u=jkbSVf!~8F5PK{4KUHb{zMXTk zepv;2_x1><+99DCRO_-VA#V#$->alZRN|idr(shDL)kVWpJr7k;Bs$jD!=HA)$A3AwR5^LI@lS@Z5m2+H#t>JbE=UY-OElM9XqnW zdG$yddFCZ$cw1*IA8MZI(W0J#b%N%s>rCwyk8Nsl*QA3Emu6LMwe9tL->P5KQCyOC z_}fO~RUOLw^Go?PFc3}^od}yJ)$4S8U8o94RM>jt|JvNSzq^# zzJaY^-;7otPG-<3Is}evcqq78T=r2IJ}n|-a-*4($a1+Ad8Y$GEvKHV>)zPezP_)q z+3iEZiw;)2!R$i!(&r%s(R)*;>w{)_#8 z2LFAu%zps~S2N+ao4$km>~YlF3j9IKU3+XH7a+q=#337iAO zc@LhuwgSJ|@(V!AlIpd}Onro;LQwQWB~`RIm0h$Gv$iNR-BS-qRajY;RI7#85VBp* zC6zNg&>5RJb!O>l<6Kg)Lmb|{)5UEA{o9JgZT-L;`62Vr*BtP8HXu8C!th?0X}iG? zoJ{Gm))ckUOd%bdlTCRVMRHCRhD4tA>k?RW@@llFb{1WGj*!~ zorlA_!Qo%4z#ld<28n0Q%!5D5`q&zNKS_kS`yW95mz~3R&x1?CBRh@O-?rjzi}6;< zX5Gr)emnMTnX3VNT{OzZBkvIBSKw$NP=0;-BN!Je%$z*BXs5wqkcLz`#=$PuIX4>W zFt}0HscEW9jc!-xvyQ_4xr^CX^o3{|nIxQ+?=G(mz@cYwIP{AHV#xz=rOCxvju?&nuXN?qJ@($8v5(8d?SD4(2=WIlZhoSOsVf zCiK1o%bJ4~e#J8M?sb*^kzb16n-Tyu=FxI%`Mu%^`&^~BQ$A}R@jRl$rDaN_U`yR4M{~(7DDyK00mZf{2@i$%+x8MN|hkjD~rzKV3#{eC+oJ3C23Oy zi1#!(6$e)V1P#4ibNVm@c8|k5Z`Ez<3B<}Woop1ct=0&yV7E$YzC`3~bS{oXi93+k zB(h;{F}o{K<5tzEClEz0tL|B{gVHenOY?X%;T6f{g0I(?sNVa1;`YMe@Yq|hz1$v6 zhk8J=rdz1T=<}8ldZTIml)X13alyE0w+4?dVV8?jEQ{d}=m-rhOM7B4@|^C`;}$1o zir7GCSA#({-d3~mAWg}1kHp4zjf@<bl6;I!I|x88k11XQ#f*n1Pa9lIX6 zCR3R$(}b-2=Ig_Quo>$fT9%}+V0Cv}c1f#|iL84`*GoZIWPQCsI?)^QVIRkpz_aP}}9|80? zoGTm9d%q5@hyDE)%bT1l8_=7s1JHnqOEGSOrUXDaFJzVr(Zl_Wlc2LZEmu}XpAu;F z2n~B6-*G+{;efcd@s>S5rMd6}dOzpc2K1iC!Lw)o4Cp(YXB*J_ZXupMGTXjf7m z&g4TQRsp^N#<3FiUPDtbLcGEsiTZh(h3w*K7GfX}^7{foa=SK0e6pXtN<>)da|b-W zn)#U(qT=&v9-q&nd41^V&p}x-=dDIh@A@ju@}Fk=EZ6Z-Zoo(+;bwBVap+a3fD=UT z4Yi(nH<#uXxTh3sYiqzYW-EpCZmwg~y)sU= z+hbV?_!L(2m*(C5K#278wOEB`RPLtrXJ$K+ZvtYDDh%#t?x zR#j-tOtqASPpw#A7#GBmVXHUV6)SA&F=bFyW~{MsEK_=&dbm&`83ec`8ST%J*13Fk zC-oXse+0+>1MFyCd4c5PKcHv)d-5?ry&fK2o}iA^^yQjna^)_Dgk9dG%P?^bt+r`m|Jr>MV|R95_$~8 zd=Ph+Wr}gP%@|dm@O#d&srQusJIXP4v#{O7#qN2dfMvUJ4s)bg(^K%R9z>e8i8mt? zfOJ)!`}+}uS{|yzMqSVcbDjSE%sq{=VOb>um>u?E^H0OP%E$pG&A$macJVCpd9EaS z;|Fqpn?{eHnml3Aq7x>kjvwuBU)Nn&-_fzY(7mpmS>PoZKW*vEsT0si2gpG(FtD{) z+=}@%-tj1Ka4N^xEtp@zjRAkwghNX`uCW3oDP`PEJ|^Nw#@$a;;N&YJ>fvy)$%Ml$ zKI?tbU=V#fj9=Y}@s*zTY=tmm|TX6X%h-P>@XScx!ZE6<<81g zMBq72($Io_fju9__51jB8YQhm1801c<>yd^rVQVMJn1Dw9cun{;JISuquOAl*WX5~bEA3AgiFGG$IZR#stWY(Lo2gF`Qf|=@Ch7Bdt8Z9 z?=9m+Jbj<5gv|#b|6AY_Nb@z+53X94KBgC4B+dmV0Sx4d=gP=^ggoTxXhlLw>@6&* z;<}EGbwwmU%0!baqEuq#3Ec?rM2% zB_1~E(OSddsG=ywhAr|~{vNXx;cLEaF0+@EU`p&|{|y&7K=a7#F1wCDrD;3+;fL~s zTDPXz*(|JV&0az+x!Mn55yq@gpJ>%zQRWdiU%gC`4_9XEu7fLcD1NFkAAp*(^~3Tr z;>Y*7dXwca%g-xUckFYOR@z^TtK#=Dmj)fkte#gYqZmOqVgzEv^K81^DMql+fDNSo z_b~!-(Md~}o;-#;(}-UBA&Dj7lVtuMWv|O8yxm=!5G3vD+S=E*xr;m$MxW-q;UC(k z|4ZEBgMAW@rJj{;-A|_sugU&m$)5jNEcqin^w>JP_e8GK4DIWp|2TrlEek#Lh6mz! z%xQdqbz^D}*bROZi*Y5m!Vs08rAJ4Cw9LFVP6?2)(I5yR6sZ-YR}Cc~Xsg z?|qMS#?jfG3+{cuP26-5zn6TS#=ZB6c-!nVv%44E`^eXEn2)=@L?Jmu)s{aojL3QrFNz(Tx@9QiN%NW(bnbd)xF(8MYRXy zs?k)WBQ7KdQn9|4+We~a!XYaiAz2obM3XNatxE;%N+{9ZkR5K&qcuuMk?gf$U#c$B zP^H>~_3f$V;dqwTw9!+NF*leT;9M?SI9wbt;c4(>z_HqIMz+IOE6)*@={b1VCn~K` zzA1;M2+E(i)bap-<~YJi<(UUhsXW8IDMm}TjgsFzi}{+nm$cdQxHfC_m$X@0St!4(*taz0&(yKpvbMeh`=gz1#fll!95HWZ zvN^w`QFp3pEaMv}q`T|g%)J5m?~ZZZfg%M?_Eh8Q@oZ)!N;xD zs7#U7*42gKl`V?sRgv7K)Mq@&uDXtaSg92+et@!A;8b;EjHrpiI#91x=esSrz$trc zi`?PBiPMDBW;D2NTHhe2?HlL}91pH4hh4WTdMv0kq^dn^ug9UGafka;;7DpXtk|5I z(J_Mhx6_CD1N zCI8Dp`QqUoEJ6(XiS9Xe+ty=M-sm4bkY`9?LnVRRvsd&!N%wJ*UERNN*H23P)G&{ti3K zB30nTL+EbOLjmVwHS|=xecp3(*UNgU(QdgHbJxpys&W0dA85aYMck_v`Q$P zMx|A{Tcd4p_+LcfgCs}f*SoITQPU6&*UL7^&_2s!5Rm1<$15Y7<@|XwUB(@7wz`4` z?r|gRqzG!zF7}DiE{|Dw5>-o$v3HwtKaCKOEFvD2N*gJQi|lUexAi;h0o5zAN9{hv zV|@(OZv|wWmbTm3jee#(B-B&v^irlLds47lHLuaS@FZG?SC9KaCA2D~(JE3djn)m< zama^+NT84Bo(i{UK7n10lvA(rDeswh8x92!J*Yw_e%kkOyI*x$FA9E25>z+*c|xg^ z(o4N;skL;k`F7;@deE^0aOQ{)Nt^fM90@ShER47m zb|AJAIknO?lFy;~Y}oh#SyyrA2g~{T>`C1-eAIZ;RLJ45*#eqp zI8$8NEW3)6js440l#eqo-n6*YBiKQelPizu@wt#A>r`w?P3yY;#Z`5=+WY~%tb;Sn zUr#K}EK0F#nQKzdh@(7jVBK(m15-j3ZFb@O*8lJlvOpjuBRAgUCk*`*}>OW)Dms9w}&ExkuSS%>DaT1&rlbd$vO zE0QL$>lFI#bx5jI+QI!~##;zuzc(><6~~wXCy7FE=xLj2R0foAW88f+&VVA^v_MXv zuNC+M$b}<`i8wGaSYLsC)+S5anCn7^*BrA zyL^CF!_MeJJ|a&7;<_5y|DVlvhZ6(o+_1}qx(!qY$W>PiWDD(?hC-HI(7L3xc_KS9 z)Y_uBUFyFyr%TPns=Jc8!9rK>P)nZT4ITOX^Lf5991YBwPMxILSg5szU7(B^1fdBRJUJu$f4J5+jiZdhg`RPaQcD` z8!woizF^~q3#Ne>Zet19rw}s716~G(!t0BWK23vp)#=qfVOJbJ_YD$ujiIuKL)fTp zko*p}a*S1A%!6tgM!l{+H2DsC3<{oC!-a~KP=XiPQE>3^_9=h|@xYgsoMyk_OY0)t zXa8fdxMHkspiM{BQHK{5yM~vihPQN#3_qgwQf)9~D5$gjhcu3^sL9r=ar~~HPi{<=tb9u{@!BI)*AcF$Ly1}N|1HH-o z%3R0tj27@>@RIGvj?z+6?RJ8`d&U#=Tg+%wslsDE4CW0M<(x0V& zitHSYt#v>T8@0KxkAS9RiM@0^YOhc+4+!k9%h;&Qr-dPUTw-s)Z`#fMTE@^i+CPYw zR1y0{<~C)^>B^qv{6Ku< z(7vJLh1yywH|JCQ!Issg#hm?1c*@vU zZ`OJf&1_q!WTGrnXAFby1EgQqyRN6Us(ot5fNax!x|UrvkynG>p6RBRrNz3QgGO2x zXA4s z3(QX+bb!}DO;{m3dcKcjpk4pBBkPc?kqb9q&1?e_%&X zqNh%*ZHpp|QQXiK&COa;gUN=$ zw+y8Ulv-0r54K$1*dLGgH}*|W@BQbpzQ*B(hT+CO+AZHZ`(K!yC$RUX13l)`fGH`j zC3td+rXGThe<0B((6_ETT%DgfupiZhv2nQP$QeE0>78mC-qMb_X~W!1v=^3U8>fXQ z_~aaY%)u*y0`i4j)~ao1F4AM2k)gx;&8cZ#*4DPHt!cWIs3r=k$s#9$b9v0pz8tb5 zQLb<5udC~CYVM=YbS9I=Kec71t!-&b%hI;CnU?&bu8y&>j;=+R@#yUPmg^zQ%o*_8}%bK*I zCRgJL!AEU(*qxaulETB;q;DxUwyDHeC=$eU#C^(ZAMYovock0E>_BVhotj@eG3XNJ z`gBEA4&{@ugGrX6o^Uo^qgCq~l-VQgr)W;6b~(PQJNztc54Y6D3#tnp(OiH|p^tPJT3x_(u><2brD{1+4(9h` z)#cp9?x?EzDJ&N2r>#!O;da?v+F#U594uJnc3_9q-IC4iD7|gs9V7R!E;H*TO@rKL zM>6=84LTWJt&(kR0*2n91UXV45v5fOvWP3)pbQR z;MR1QGgjRludPlXZ9&^DJ6x*fbnH^RzUr1(Lnp?|Dc15X8$<0A#I_BJmL}!nt4-%4s98Gk-Xtc|-ereP^A& z&cy-=l-P^r!VP{_n?^Q)KVBEAs;dvyBvPTOWL&Mu)&yfHU}|ryt!@m38mq%e6nbus z0RBjIFi=wy2v%Pis|wZBgsNh|a|<|@JP*%A;^-VgPKE+t3VWx3D02nRTfMN82*J*- zx?FoCALV%Oi-u#7+CaP(%Usi$eWazev2`FFZ3`y~(e%>cW;q9tk_pM3Relfp%JvSQBcGuif0_BzhjOqqTDGw<{3sDerN0MG1o-shK@E~d>s(^FygJ5-Bo zK+C{0BS<@YiFw2@Aiy*GBOp=svgKXu^)~uU_hi_sIAtvc2=L6vM_wVhvV7llT|%+94K9O@gS9#}_*_kB!}|Abw~otx-hqP`hw3%1O$~-jjJf z_Z)Uq9D!%kpdv3jA6}n^JlT3BoMl2bT5k4V!r2S2BfNo8u^s$+JFed@JcQ?wm5CFi z@H`%*>!0J-hhS@b!hAk%v=0*pqhY6;&%Z5Pz3_S14|M%A=KJ3kZlLD{25s{^YJZh+ z{ThC~6YYPRUl%Q!(LTbg^!#o@JaK7M!9I|1LsF8MHQ|QX;yS3e9+V08dQe8f>-T#D z9y+nAVfG@NF)>2+-TpJmu-?(cwXxQq-Q`1-N=hUH17Sjaur^`eZ~7Q<`~QeG&Khy|L0a-a-Dsz8XM31NhGqC~ZS$sn6&-+HOXlZ{Tek7V&~{A`#4FPIC%^ zS@t5h%M%QGBxR~9;HmeD`(am#toO=*Xm zr5I0(TxOEvaG!B+@nI{P*W-d_6Qs0G$?NfW9}Teky`hk|G!b}o-mt0N*+rJ~KzY1f zc+q@88Y#>Tig3^z6H}%+-{M1yE)N=3g&?&lQR}QfSeo=!Re9Nc!ACeQdGpj`(yktZ zZz4<4JRe%9B;P~1f~jK=0T!84kmCh&?^=%lFC3ax3 zcUyO?d%8(fZ4^yl(aJd43f(I=2ui#5VZ#<4N zOuY$u&dWZ-lBi-Z!aYOK!csE1j}eY77CO_$#Dzk^0=`m%xEREhARPlCOMYD8r$nSv z623}TD>{U|Raj<6Ng7!_Kok#9C7a~4F9Uy(8A5mtx-QNrxNcY1isD*Lvf2E~j7ZmY z_8FJDMHGdCWL0n)MM@Vk5{gN8!AblR1D z_H6c^SVwWWvK->{#w*x0INVlt2RzY;2$W`I=f}y46}LdO)D`9ZuW2gHo;G_1MxH=U zEBU1=MJ-Txm24OGM_sp-CXuTOe}t9YFM2?I&6g@Ws8=M}Glg-c~cmwIfv^SJ6gZo7tryK15e&Gs!Kb3a? zN1#Zm**@Mk$)%k6Jbo`WejELt>(A_VVaM#1)IP_2g(42nT{w~FujC*63QC^C?+F(m zLuHm>wn_LtSx2bPY&Zr?r%1W16G!-nY+rM1Pv7w?W!d3$OdizPe$bS|`H(Gp$<3QK z-Ec|H>VajDIs2;R%db8=ExT1Zbrp+6_d2x=^AH3m&;SaXOdB=;CTR~i&xGs1! zwgL-S&v*w71Ol8J3NoA>p`Z*B+9_DYWve1An9thCFDf>7I6h+bZa`b_=h*&r04gT)X`7#Ib>g&{sVfDCwxgIp!y(h6y6T%<_x5gnzj`v(H; z%|ct=kM0OC)a=@-( z2s8P$7=jyHFEbUAPt@7YvALO&?VCE*A(!&C5EoQr416E(b$d&IuS2!A z67H`R9IPF5m=e}=U1UAq$!6mVG) zic4UlfUqXqUXO#@tg=$&(p@&IO;_C2%`|?z&vJ#??tV2cV<*BD``v&boI$D;+&yR) ze1o^^scNEehhRBe!=8dxwTbG%0Uw+GO7INe-{kPk$fzLrX4o1ONocqH9&mV>plQ*v zgMj0c2`3Qb8QEy5zLC7wk**6*iI{d*PMUr*ywVmx~$p-SO1|1eQa^ovgE^ zG%>k?f#4nF#WkWajk8)JT8FflFxCJ_du9o#s()6OZ4Ce!> zFX;AMMQNI3r;xM2&WM8<`}bh2f-we67gg(Zdd&uDJ1xAd!cv}5fNUCU2hG;qp|NA(@QrQoS` zSw)B65oj7Z^1QK=?%zGW_R<3ePCKZ-BRMqFm>Gz-Os#Gx;#5@f!NRh898tdIb5W`y zHvDpq$;U=h1|L3<1{r3GF=G}%^To5|hQIWy9Gu10!38Q;MKN zzZI0+suNiVUb%teju}XF$6MF*F5c>6R+nzqME^)}U^w0tY22`G8G53E*(VS?tsj|zC z%8QYduWREm?cJvxatBmPms2>)t?#`K&ZL0T?z35OqdgmYGb%K1n_0Cv?Naxi?u1HC z*5gTt<2(vH!*oa3K`t5u1Pl@i4Zg*<-WtTmmpy1prsY3K#Zt>o-Y zHTmt(W2{PbpU}1S+$A&TZ0qXUcFxR_bGLT2wX8aJbo7|jt*xt%867=#Rm)Kw8@me| zyXtLKio&p2F+GxP8cjMin~2m<$pzKiYH5PU_AQc(e82TdLkA{4N7Q9%Of;wFV91S` zAPd)YY2tetTBbQ>E>J#DYU5{ihayjMH%xrLvaNFI5-YD%szBQ$-v{*GfMk`;R%Fv)Lr;Z%c3T zUkG#1imYMRbKQl%un>S%fbD)+97< zOsZ@+14wFib#@xdmWUn~NEIhlfKvi)6*;%#A<=-JMW?1E)n|7gR-kCEdaqbL&@nPn z*w8JuG*zW_6~$qh;u|~A9J!-W zux@QD1fbH})hfjwV=JRmGs##BWb{LJ&~DyxA2cNcC&-B-57dWY#n`V`R^4x?;AkVE z=PzXQ{iX5^`#OQAQY+~wEJc_?2+sMr3fgEkP|bO%P~bIo2yYeIgE5srB?^Bq0E>n2 z#MuQ9<)d0xX+voO7Sn8K)oF_sowjP`GK1LE#{w*o{T^3Rrd4LpS3PHZ3G3P2+xtr zBVzVGj`F@vg!*`XP7ax~yTm+d$d7_@jdj%pi;PwRIvRjE!@|e^o#RevJwuyeFCi}^U>Sx+ zyWjrrvp)-*!z7?Nv1yPOttTu|F!g5?rj>pJu6d$K6Kg%g~d zZL9cZ?ui=qVAUexFeCm_yWeA!+v3%@P_R=`y0;DBpfi`|2*C4|j$p-QQ7QR$F$V8d z0zRVsc(Ax6@Mv>qO#x;8 zxxEM7x`+y>Y2@t`joRN(R>2mEMd*35G^hXUUl&-dse@Lx+d8NzPPSA=MY=)J?Xvcx zGM(CQh1;=TwL439K&#eS6U-|sjhe*+~RPBlXLtw^3Lc;rx4n(F_7GT&?`jjfp{Cy zyvO*QW0MZXL_CEb=O86uM$P->n`0tSjKJj$20ynxp&*ivEl0jfHc5WZ(tUH^2gj&L z`o_~H+IOUMkq%U@U)D5p#^R|nmNw1QYe8ADI)Z~^e*xcvDU6ScDV+)59G=qd%Hhe8 zGA-kDqfHl1&pTh4=8h`Fq819~4r77;ud@#y5;u1`1Do+mkZ_lb@Q$M4T{%qUBG}I^AmXAETj$H+&1y1Vpln(xPPO;;3 z_l9sNtX|TE#w@!kb7Yc{IY;I%84J|VSd4*SmWcP00OVh3KH+<$ggYvEk6*-8F`e_x zV{WdI2DF&E%6P?*(xqIS9Ydn+$JuqOo%vuqlXH?=*Z|$xVk;1I;&)duIuqa zp{-dDBP{9eI%rYXK~dEaa7dc9dM1~jNC6sZc%Bjn>EE%u>=P1IOM#%S$yTqTdnG4S zI;?&{#PNS{`KoTU;FXd+4Gj~Emo+1)F_BQ9W&5qRx@61pj;7^{7sZl;t)S;d&~q(T zX)R9kfUF@7UCvl1E6r5poMM_tyO`v|$563FLIMbG*fNM9&jkH&!F#>+E0zxE=orin z$JmxbuSEf8HlaD}aCh2E_bc|4(*^yVW#OOIT<+4Z6}yU~2rndp*!!_+f2t5iRcj&W zNGa{2++3hVlXimf(4@_o zCS?{wZh%h^^Po~!rMGqW#3@ro6@6a;$$DG5T}YsX{A2$?x4i#fSfyW{riua6r33jo z{c`W-LSb_+{i1#-`1lEM8K4u0OvhITN{<<&g>z);WmZu^_kS{8BegpcV85yzp;a}*b{?Nf(;{@ZkT9IGmwAs$ou~=<3ELevqe(k>< z9hqiyl93(qu!$lrJ`#0gjQRfiuA9Qoq@B~25IHRQ+lTe@NV4j#qkCAE(9hO?Wbd9q z#Sxe`+{){)Fp>d=HIU>o3k?VT)J>=Qpr>vzi6ecc%VIv> zyfl}cY}ZvWJCUbjfNXFuy4j26>zu2VWkA8Qy+T$0zwrF&|HpiuE%|@Q-vz2=CVmUi zd~ONte{~U0@{_L~v&?sk8DBW1Le?!jtDu9XXlu1_Le&BE?eOT{+atnX{@wEn431^9 z5#O@e*v|jaX$F{A(pRYBKgo8)20v2!;s)slPXvG{;5*<7$8%y^L!YIa=DzomnfHc< z!=Am;YgVh~vb)r$oj8C_41JH|t2Tvm&iKj75h@oA9s9VX%*5We-%a5}& zB<3ginGl!?o|ce^FGn1WwJFeDmC{oxbCh1NQ`#;|phT#8h1cOwSc`(BU#;{jRbkC4 zDgdjcrxjFA0#&qIjuo&wn#aVF=$~Qab3MS+W%zefikSU=qsHe2hw{~w|6aw1a2fx6 z$6s6P_tw;u?)BBy(tTfFII^F!r#LXyCAkj^LA;QS%Un|SwxCjq4zFYB^LEAKu;~)p z=Fn9~>5G!S6{n^<U3_9Iku!yBNDr;WzBB3U4T zF5A+^TiCZHr&Ri?1MWYzLBcD%icj|3R2i)|{_(4!xRvf|X9RBUfJ5Wp*AP(J7J(IyPpH6UQ&S zIL7MAsw=E0ZR=q+@k$Ash&*-F4zKNM(dyO$>J?R2 zT6Il@Hi{E5EvK{Rg&mxpfQ5y+9*w)K=L6#l`nQ_s)u7bdkjezh!utb&9PC(r;gT}V zUI7*XjszC*YnS(91a&zP4x^R-0DGRj1|1W*KFlzukU3%~5g=N_*T%EQ$ZC@Je+LSH z7|IVjuXfS+P?ghfMd^1}AR7%Yni#Bt3mYK-)g5T8<(FMPe%YU`2lU&tM=dmvdY2!yp3U8XawBR)X90B2F`UpRbcsVd(X!F?Okmi zvvdATEYR7Jb5rMFj=7tWF&T7wd2=t!EjEC@>ZLbr*a+v-ROVu(zo`&bBKCm1E%aee zxeIfNE|YbL17a#%JjAud`EflSu1uBv$2};yiPPb+PSjGJ_IlUm@l;`YUj-Q8Py`?nUmxAygJ zEs}I;v9yUp>@H~51AK)O;;3Jge@JfYvL6Wv5foZ0!jLt%r5TknYh1nopsTUSfZo4y0yXacyufo>kK@Qn(Am=l53vIwvI)f&>C86syY*~&bI8j(M1Pj zn;Rd)k$6M-{7}>%c05&=?5Ynv*4UgqVA1HhY%2PcBkVV>(tKNI@5X+@Ul3gZlYFY@ zr%uqEnJyyI#+LYEeq=^_wU&1K3g@Z`f4$!WqgRVHSKF35S5MoVy3=Rt$+D+cuCqDa zs9nOun$~csX~jCc3mHkmC*cKwO>gcBgA>D8Po{1UjFgeYGy2L-lz z{Djr5v5uY^LABuM<-$9q{e7le>n7c1?C4e`1deDeokIouh8~=0l4k$Er=$u<}2V9Oqb#jGwn4w3!4AhB0u@joa5 z6v+aL><9=K;!s}C-cL)C5D~E%K(y4OD%MHd9t+a0a-wD)zZB<|9aatf;^51i-eK>7 ziGQwuS*<${~?!GT$YFR9ABi-Z=!3`p?(q5fa`51p!>FbVJh1vH?4Cmdu9U<(%srR#pGZumf zb$Ls#C;_GPdxfVV;(cjgZCv;^(G<8xS}X!>e3kOxQi#qnGnOk}i%7s4-6LHVITF4U z-Fuj#*ge)uoDW)EIAV*jAKJXgblhg`wz*X|yuzAV`lhf70#&uz#l2s2a;=WekMIh= zGWY@X5Ku9#e+0WhF`!;7!w{VvK|%h71jgz|cMYdT4ruF|DR@Ilc4S=vWZ)h@a!Fe# zF}$G@XN$r&+rB!NUo9>^ZK|+!qOoOhR_H!tWw&N^tJw6UjqN!4=$5|2mbY~GZ13sX z+=IfV_yR1fm^~ajK}qH<@L*A5Af+12Njkz!ximu>oAabybYw8frxsi4dRl4?6B4;B zYHlo3+mn@a_f>AEu#LPxrB_fqwYsS*%wqf}D!_Xk%ap2)c&I66{}GA>+mQG0uJAl! zjudrn;t>Jj%w^t03w*)uF__FBdd<-B=6LO0E!ZbH*wQjk7aM49N@gVVX&#h_ioG$iGFD>>=m5$yoxNV>PEvVawFmY6{|phG{}y^E)>{&6!cN%PF^Z9cLqS54W{v=( zVO6Jt5I)G@fg}pCYM=8#yYf9}AfP{}*dN4cy;kduqIjd#`b{Lc`IukFzCYe-*`-_C6^JO0@Q_#R}w zQ^m22FbMb|93BV!9t-yK0NyU38ijZ)!QBy9jCX-UejD&!VUk@fwgG;doxK0I;LQ1* zCVW!tufX@99zX3ZTO?KGhe%hu=Tp^oZZZ@i*2ZW_Quj=*j58R)Dm36!D9mC?UtmobEnvnC^#V#R=^I;cg5Uu3au-Y=+Y-qwv}v+r z--ae!vSwQ@E2JYM2X_F{&Wkt-??t%{AcETHFqt?)DC-u7l`--kb?QbLZ#MH4cQP61zxzrR1H^I$_Q$R$qp<4*ocgHmNITGn~0_SfyV%-q;G^b6LlJHl~{b^1= zl+RsN+ZGN}P?kQc;_<3bJT5fVwA9wN)EJ-H`cQSOzAA)vQ8h!DVTXcNAmh<0@8CQW z3K2vzWF-fsDYqg@?Qw6ip$;)KGD1A6U=1}?xsna_PTg-~E@37Zs84vY_sy$0!KHB0 zN6_Q(s}4SoLu{jPBKhTg@vcrxA2xh13y1#}4husywY8KnyiCYj{(+h@4Y;SGRAbWT z=AFxZq9;iLX4kutp}K_2i%s(k#Tiu;QgV2Zkrq;r2rLQ0W4gC0;quhFB?sa_4jI-p zrT%&}S!Q7rWil2u>SBz^s!*_7r{Fer zy0A}Q!J^*hCt-b97-!R{a7Z*l`VHS02VP+e4DUo-@UcX9gD5&-U0S8)aHL7HqGFUT zhz&i$bM?h~2UvJRsxb~4(c%S6AF~`Rj9Fe5c(uI!=gu2wJecn#^ui^FxR(oCrx=T4 zZ=~=kB+p~NkCkfcI}n$!>Qq}F?M}0YEiVX%aUR%38>daiAx*cn!P%-NhS|eiiL5hN zWJ4GBi&Bq{LuELZ1sj0q%7U34z@0Qo4 zhm}k1u>a+wFM+uXJsW3dyb9|N3&hB%Fm~O{kL4O{Xi)e{1Bf%2sfqf`VsI!$9KzWd z^^H-RXhR_op(^RGjZ~_>;mC|kl+N>|-QbDYdx@)63QhcvxY}{V)NEj8E8G&8njLw@ z5GH836Q{ZiqV_gui$=O%{&U}l@}IBP7h?{vpN4cJ*pFKy_Cvkh%~s9cO=~waf9<|e zTl=x9ni_0_!#$RFn3uKDoEJ&JloM7HD=2<+J$<}W2@PR}%F+1$&LP)hSFiMXPwM}-6JmH z<6a%%ZKsQ$#*xE*g|FZVN!V(q7R%BB6}tEvvbg>QT%Q!SqUN6EAN2j)ABJcpg&1x4|RT2lbSZ$)N;2=a= ziK5K%qT$)Ie54K~`+~+wH2;e7oaQw5jZ=mwSUv&m(VU67wO!VsM1ikjzIK5TS@?*C0q4rM%H3j51ZU~PxrgSs2G4M^Vw_w=2}P zq&ODQ?B;OF{j|7=PU-=!=dl~(MR>_~I~F^wDl{BfZ;B1pNDyPHRbMPT+m%CBr527N zW6bHwA+dqVc~l{(4ai)d#COa`44M%fYWeED?zW;ctc0H|B|ieTI(H zDg2ckEyfq>I4dKO*+@kAYjsUcHSpCcbXb11A05Zo*haIyP{-+Zrh+l(IBpyEEGu?Y zIYOD;8YO)lPwkxpR)%{MYpUzcW@e5u*g2OaFKl(%;P{G)|^w~ zYwka7LQV^y5}9q>YKLX8?-C>}_%&FPJ3dmwxx1|4=r*}&frcaQfT~Iw4&jV6oNx;q zcmF$0zr_8pWBx~)$oX>(&3!|k zn`fsiUz&Z2@3;kHCVd97s6EK1846!QG67!-Cj36PBH46}Qn6*l;S5)WVmti_B%&*- zO<-<#P?%Ts_@j_3Ww{{8Oy;`7{-lLQ8ZNH3x;Z)^I8Sj@R%+6qk(!8 z)!bW#pUr-2RU}e{5T)O0SWM8ck98@vq3Rkd5@Hp@V&bt&z8k~xSx!ktz{zy|E!>U4 z5HiEY31(T$E=NU?-*Bn7|NGgkbgTIQpEHZ_cgr#Gi%~sv2FTq1B>ub0a+Kv%%Wli( zEZ1A^#J|LAX)4QA8tRv{#E-P`+fHRQkW7~T9qDUTv?%PRt{HA zRCX(uDo?0()vqpA*QndoqtsK?-Rfu5FR9O~Z>jHSqBf?TuAPtiV_(+3rQNUnNc)ZU zruH7HDtaApN4sOtG3_|OaiHUP$GMKn9ba?Y?)aYLM~+`R{@{39U#(xS->N^LKdHZ< z|H;|y+~vH^rMR}bj&z;uI@k4S*VV3XxNdhV?tnY#&bf!&OWX&z4|MNxpYFcc{k~_B z=Wx%7o^w2(^jzh+-gB$xJDxvz-t}6&Zg1Gz=3zui zr1x3x>%L*%Qr~*t;l3+;FZW|gSsy9|2T77)=S=AR+f1>bI)j zu3Zw^=v##clnqzCusJWo#vo+V&e5>ZJnultBR`XKrRPCDDt80H5&WHQM zi^FTe+rvkNPYK@d}0(KROvbC;G|gW6>9*uh%8(a&>)mlXa`=4y-%6?kjcwipjCj*s|D$*deiF zV;_rM82eo8`Pf^rx9eH`ME$AtyX!Bl|6=_O^>^1lUjKCc@9Y0jKO0x$)$vTcD?SpR ziEoG>5kD<{QTz+>8{>DzAB_Jf{#^X^#Bk!`#FdG!CT>kUnD}YpHw_I9yBh9oc%$JT z4YNr#8A!&GP02#?lH});HzaRQKAe0q`AqV6$-gH5mC{mYrtVHXocc-X+0^e+e@VTU zwx!+azVu{zReDSM@bn4kv(gu*uS|b6y(j(M^iR?+rvIFNzfo@VHbxqo8_#aMr18qe zuQ&cGQ=3U=Ix<6K~4UHPhfeLkD-$oJ>R z^ULz<^4s%A#;5C3&!<;WEye;92Yy=L@pi#9HLd2C|rt7G?! zJv%OrhsRfrUo?K__`4ICiH#FqoOoff`u`*DO~9invcB=Ex}EMUojvJJI-O1@fh0f( zNoQxz3WQZQ5fKOw)<7`qh~PfzxQ*ku4JzXfq><(L+Zs9Q}vU zFO5DjCVEWWm~~_B8S~zlLt|}Y`;M&~J96x~V>gd|f9%1rUE}QI62`g4`NkEDD;YOz z+}LrakDE1a;keFmYsXzM?#6Mij{DDPaiQBCYSaq`3^6MqAhPnzG5m({M$Tsb3g;P=8X zKBIx)AAM)MMbD46PQNTTCzv0VYg>q)c361)r-q9n^&8>+UvZZr-;hHY#tkBxocgVp zuCBx!;FsLnVtUjB#CcmJ!_l)FY<=0&HsXHF!QX!^Ut91m)tvII_7zkaPIZ= zkP}A%pN=ou@DxXJD*RD!`6ycEXz*pYpmumWa+F-~TRaQ#`~mKO_!4jHjs`zb14X^% z?m&Lz(oo-V`bn_mJQ~K_LE{=ki7^0l!Kq!~VXA%-e6kB3GX52TKLof7_5FYCcA!jh zCzNaK0d68_8io2e0N)Du*IWSa68@*|W#sh(@|-5}e$~B-c>kO3CXsJ!7rql6Y~Vb+ z&q@6sxr^{#>qIvlGpYX#cb1v2`80lml^-~iH5TXMm;It6S_Z*&n&~5Gvk|`Xf7juR z`E0cmW&Dz>0sS>19uCjoVS0)k*mt-C-=ng_tc!LQ^;gv0&HI{rJjW#W+dChU|OI>U70AZ{KZ_bokxM*;Vf6^t8!^ZW_7M|>Lm z!MH|@C8z35HzW8@3l8Dv*%18L7#9(baMMul|Cerp==)3M8UFz-BrDpL;pEVfi!lA(tVpX{*vFi*=D%GdJti%Yr+-eR^5}F{YWv9qcy~O~WT2!hMMS{u|b;^VH z=Qx>Mc||qeGWW;n(mt^V=ade@4(eK*%-SF4nl=J<0hf*x>M+oOn~)GvY~{T=_EI1wMz}E`P&1 zK|lO{2D@M3(5C|EpMAw&M2(z`vm`1-inIln2h1y99h9*CqNJ%@I%F z)F1JCrKp7S$os+n`ysbiig9rHD6dcs5IxZ*d2m_K2|Ynemi!y;nAi&5rXZZ$2J8ZU z39GjQMWXQr?5}(9MzI5HeCeXt$U^^di9SXp=o^JHDDInLIBo)M!TGCYau4e88}y@D zqRjRT#wW6&u=^0t-tsF^Nn@aV6M5bS+o=j;2-z=ph*&ijxB+l9p6wDb7+XWmiT+Sd z;TVU~Xv`6BgWriF4tbSArw1hcE{AiNE(LNGCvw$kDDO0kKXHf`Z^pYEaqOZN>oZ=X z9dh)uu&KL5Cfd=hu1A^}(GTx)3snQ^&|8c%qD5cZX`mlt0GLk>dxO~p@a%6oEE@BP zqa*MnJjEGp!g%k?xITDNdJ4zBD}Ii_(+4*cZVns{qGdeA38ljhImXdBr-TpSoZ1n1 z?-#)Pyk0U>jFuh1odLHS_`RZ3j=)~3m%#5uXpd6h{{Y+-c)GxQgLd3mWjFxW!eQSr z(@ZpmXrm{USprAaIME(SANJfcjg+6R7i0!9){|sMd?bq5k!;`uL9+ji&w_TWKTZhF z={h@#-oSPHQ{SK^vQal|c~_UCeS{dPpU@)#^!tdvP3Q(gh5yVDYRuR%O?Y(`e@y8z zY`4jNej$HP;kdYoOVmNF8J6eG*%VH6o9MwwA*Z1u(X z5`Au8wlCM$)7Qt>&sXgm>M z&ZFLC2*Zw=B%>zzMuE}StVy)b=}Yl>eSTjLUvFQDugcfoSMQtZo9=7(t;QP2RlaMv zCY!M4k>pSDXZXE-zgd%7uE}!$d8o-9{<~3=`~8oiCeQe{MAYO=)FcWuNk&aFt(we1 zO(gEb!oE}aGU_9O{luGH`v}s1T?6zpnAi1jP@lRS3KGHlg7*eD1#b%87`!fcZSd+Y zlF$G8_etJPeeY+{vzUSCKW8WG38uu;R zH+1itdp8QP_X@bP;nwV3ytid9`p8GS?1xy^wo}%UX1Dk={6-!mrV;vE7|q6|##I`R zUsqry^BQBLakueM2p4`D_ZpjxM~r8UEyfF2?|j+VW^6a!Fupbp+w3-+EoJlD@+dZ* z@50|@+hf~rI~YzMf^54u)(#u$f;G{PLU{eOy#uW2Ow4Qg3Aw3NXQ`oTx!R;!)pRu) ztFc4WZ&W3A)mErFRRy`aO8r({iGA=hQSWT%x1Q)#&G2|G_3HBSAW2XFn<%@%Ve1%Gi5GreCjXjWV4(ir(rdC0oH?8 z%Z>6@d7HckC;V+eFa1F7##a>oQe)KuHBS9rmEm@yv(*J^nYswy@Xp2!(sSg$(R;5` z&G?4)T{RE)YI*T>+$*@dYnB?M-cwofFzyd5P(5+q^FWL<3Og%2A{qMvv#@VsF#1!4 zz}yvQ?_aAxFH>N2qjT>x9^v~zzR{6602%2Y)aN)FjL;QfTIZour zRMCyKf9Ofg@c1ueTs&J;`JY|$=TMTeXtmdW{IwQLvX z$faV9Tp|~X3*;)XQT|R`EZ2&g<<;VD`A2cT+$0{9w~GhlpTy(xUhy=}z5O%J!hKS1 z7B9=EaMIF~Vyk>Y{8PRw{vqEIuj0<-k8oP#PWguTP`-mcu~U2||0zBf-^hdFKk_s2 zt^8D6CodC!m)k{wD3N}#P@FC2idABRq`g+h#p$v|Tq4gAugMq12su#P2HPW3#=}-j z5`U7{izL`y=`sekP_Y;)1NeSwl$Z(I`$~C%_#?)chvXgN5qXz*1G|8B!5(==J|wou zr^VIsB5{|zSzIG87GKLx)FJgH?lSxuHwwI|zEQ73xP>gX$sN z2=}CVNr z-^tH$YPDUY$!J)WL4kuJMS{dxm9P$+7#3}~i@}c79|um+jut;-ME*$(!Y-I0vP?9| zYFL+ru;zQgO6@IbWgl3sMPh~=FV2$V#P8%K;`j1Waj864Tn_8-xAHu31!hc_$@TcY z@p5qkERh>wjou`$64%S$inrun#5?k@;$2vw@5@)k2l6$sL%uHFldp(Fa-aB8?iF9j z{o>!)uijVn!yUSK*Q`dUVQRQ)QKzdZYJ!@iCaZ~Rky@+Z+ny?-C2TV8t+<8mG;4Dt$GGU!(gXj?GTcQ&>ah{kDz37K+ zRtkHn7SKds+VDJ0tU&l|u}CZvtzro*MLJWu6KNMBd^V@5L!9M^-vRtGgcosc%aQvs zSONX;PwULZDOPJa*oSy2IcZ%Fkg`?c&f%bN!?8LWKCE!vi zZ0-RVRmQ?K0XmsaeVK}Ix|3sQI>Rw0GsVkL3!>E@QPRn2Pt2X#FOuWSz-4OLrNAx& z#guD1p43M3z}K-D6-VI=q&DDn>uU=YLKxi0Haf{?w?Di~C9C^Ysc@pA4P%R4V%Mkm zn~agTkPIVDyi9<0PlQHJ;;}OodOuA(1dW!Cakv1wGbrXr7e2Gez)UX_r;2-UvbR@e z#%5qsD zD`gclLmyegBlQ62y@AlKgE0QQiE-#HXuPAcPP{GaWrJ*#O|lttuVHdHwEGCGDz#ul z9ECA)j2tV+VT2ryxlO5@fU#|&oFpg9(=qO!A*aeSG5Vi{5eqW{XpI>dul_Duaz4Jgeorohh0rb+W7O)9OXbs>|6OC`JMb;9+5v_wEI#1B#(+Q@)%~@ zW94z&^b*7d9*GY)4Cup8#5fftPQ%^`2X0G^QL(tW9S5AN1eJ(;gp$yM*Wlaa6mgD9 zRcR_+xs+RF;MSi^7$Jl?IsXtxGqAB}hCs`VKC&co<;fw~aZ@5Sm8Sm>9k z%P^n*ExvNR5*GT^VvG1rd@qh*Jp7%y7WVRWu(_{SHwZ)Bi0=w-hTWUU*8Bai2kwBL zz8&jQzrnoHjy+dH)F0JGk)&=BkBP_Ct+1dsiF())PFTk?VJQ#AK90YNC$QS}6jmz8 z%6|b?K2F3EcVitYMqDZ`5x*6eiz{$P%b&yy@fq%!x?Q}i?ofAPHRd99w|HJSa0}19 z;srS+sp5dRj3#naH#SHnin5r4owlHX%y(Td$9 zb=Zx1t=I^g>}hecxJlfFle?dRMYCNzgWGeT!yNv3%zam(^d?Hp~&WYrkt+9N&` z{}lfayRnA3SAC4LVE3v0>JxQ9{YxEGpQ_K)=jsdCl!w$;uqyuzTk;$At@@ApPJOSA zs2|ji>L+zn9kbWX>TFxt=2)?0L0MB>lYZ7$na@)588Dv}^eioFXw<*!_&3run7_?* zbv5SiI*uPGHS?(pJvqP9Kxx2Hx43n7XU7spUB|qRC2fmh>pB-Knb$gd#qu^sU6Xz? zX{9xB4YL5qd{{cP+Ae&)Qz07rh1bB^&tjS zIhxFhYWmD9%PJgAy5gGd`Ud+jrX+4yxU#y2fk5oA6G{n`+J?_+?bJyFb#cSHrzvX+ zR45$mI>KsUv|Kai7AQ3d420^)gqAfmFdi*a=NMrUJVFy_AF+Hv`<%8o5`fZ*y3+E} zvbdJ+1(ueTbG=YZwS5#fo_$mVW6PPbh*fVJg{o+}tt2KzZw5(+}<&F~tdX~{hM*a%iMmg%5IvsoKW0vgTcZ8WoN)L90! z5S0c3W}B9W#K9DohNh^o^E*0C32Zd0(G<$fjMZo|9M_1BoxcL2-nn9Nd+Q2j2H4)@ z7#k@km1gQnE~^whEpF_x_SR+dBYH-KnRA5}?m(cyK3+3yyv4B6vP!cSWhT{?W-ZD> zB{bP5L}XfJR=%n#Zo&zT94IpxRb^&XWoA)j;%hY_Wi`#Vi4afw#0WA%ajSKO0%azf zs?17MHQ6Ucwoy5?QE34Eg!@fX=vfstDa`98v*2cvpr#P-%}kq`;wPQV>#{1xq)1*@ z*EuF>QIDELO2|Hm1zs0iU1L9;*%Nnqm>n^v&sosc*|u!KGW+TCI$Kw^#hnoj)ah!> z=5DUApTSWyy4h@va+7tq9lIcCE8o(3wp(!RiI&(boje87K`%0P3>f|B-*+3gd@kJR*sG*|=2K+JlzV4-7yS%n3r za$aD`YH6UnS}W-)t)v%rCjsiW+PP?6XItBn_SPkH7RULU~5MW zH6`WDDae*dYk5ewmfM%;zOkfrX~(kVogGW(x7n9OWLKeiU#7J)v~);o>l_2s+zpU( zNR`zZZRmgD8oUB10yB(3DRB5KJ*G33($;3q6W;?bwM z*1ndrk6n90x34WX=FXTK1vg^>`|UAn+d4Z+<}B%091Xix1It%wOziUc=zffgncK0V z(}WhRG~+E>u$ob^%V3!=(O}zxdGnX+h)Wij84^{nllh7#_Y^#tYCO4h@Z?s( zlgA!BxpsK+=!s{org_PV#hpwu14J_cCd~wDnl(f;b3CG%<7t{1)HHKMrkNm3vpM{i z*0W5Mmey+pP^yQqKxwr}bG1oxwMlceNprPHb9HSDxXb0tcQRKsyew`mC^KOvlVtu- z4dXQlrDj&JbeWCXA}%e}a}-Qjm^3`=T}#_K7j(>_$p%LIIjo)=>LzBL8eKBGby*u_ ziL4r9=PsDH0%>5e=~8qT!Ejq@U)DCeW62ywmz9;rv@Kf>rM|o^gfVMeUl~8UqrGFv zV$^eaYv&rTTP@{r+U_>2&}R%ZCi%)X0RzK_1yC? zqL@|HMnS+-8F*Gkw=G_}d<{}+G6R~ZKq)`Lno1sN%1f(wIzW}yaA^d-HUcg;;Xp)u zfj-SUf9+dm=$Z8^Y?%yc=s642VC zJYW(ZU~VIvO|#N+JtE=BGF}=84e$-zc=%oK)D^2?kurZ&^N$3J4f?0Wi^!<;kv{^F zKN=!`R7L)%;~xnYcOt_o_(w`NK1GH%NBk%&i~OM}Nsi=ML~uZ7tScCZBqyM2YlcMf zAf>CGjRUfe`q+9x&aR zoB`*-I=aN#`uXrL#Cnz#FJeVoVm-SWel1pqCDw)!2P?n7hyMqxtV+BsejEPZ@eWL4 z4fAU(M1Cv3#hJcK6=GsFZ8`i^Y7P8z6>_E(H28O@hp{sDD9$i3unP1Ay~TkOIDJdt zy@$Z+m%{!P>qya;P;6Oqye zC$YX5#m|{2D;n-nyw9_PI*B*cw4a`!X>b~vju@1R{%~h5uIUW+9VTGi^0- z-^P9GEJ?i8SrfN6(I3Afu_f+c+@Zwf@fis{;>RV_$8V1NHvX`)K7Ll*kIqpEbK<(< zm*ab4nQ9NT`%2 zsN}SSXpKQ+X^-U}WIrzSc>e@ydF<}Xd} zk>ID;>x&#THuzTX&fpiJ_~9o?IDmcr{KvU4Kf-i{p-`GgOk~X{T!)2XM9_K?mvwHM zI@A9&oP!yvX+fPi#N=WqcQek;;9hX$gHTz)gJ{=hgP)>TMf?S^*CqHR9Co`OHDvxihSID*v~NUfTWA2foJfobccb(a;oVS? zWuvCYLxGUxAA?7+3!MK>EKA2joI`MXtu#Lbe+XWRxQT=br6JCcMDUaP!Z&6}NDqr3 zf4`%1fs}pIy+uN~pZNSGzML4Z`)_VF)QV`wn-Ky2{)o0A8jpo2)+vHN0{3(9cWc=VSl$M`~x|D#(7dXM=1Z`E(BQZ6U2!Z{1**szQM2PP>#UT zSuD&)q&5jvoT{7bnECy}<7AGVjEPK&w%iE!7D~L6s7AfF1u-f?55`8+ zg1)algqYX@fm&V)sxAfedGJ5@wI`Gxdcc2zuYz)`_7uXk`466!j1#|*!w*RNg_ZUg zS^;|Q6lpDawc;K|dg9d)EmgsPaXD80I$p3V6dy4U2LEkhsci1O;osqSr-CA5hr_$+ zSoB}e5|L{--v2d3BkigF$a47q@E81$5}BhA?FSD14aAOe{mEkpqn`#^fM~#If)+o9 zp2K`4U&~}QI&`Zbh^w`nt(A0e!VU1QQoRx&jglO?G0=h!KGU4H0s9a$LfIX3|zCX%~~7 zh;;+!K`j^h;}MB|a8EO(F!&zTt$Y1K2}1Z^3Ju3Y%ulUY&~W%G(xH$H)Ju*vXI>Yzxx z6}da0eAoMEE}`cDZ0)iw9_71;|6S#~k$R152{+FrvUtAW;~7OU&nQaq9$xr{!(9Y7 z25ur8EPdZhxVdoca1=)LuZCOCzkB*7V(<1$-(2k8?!^A>^{DIRzH7jXjlSD`_xc|4 zJ>h%Sx7D}Z_lECX-!9)C-vQqjzQevFzMpecPIOK}PFjv9$DdP>gEQ2yrxbV3!w0M? zVO>c1Q}9*56_6r=_hSz6G1^f|5Bs;!Tux5G979TsGwlDw{@*bZst9gGE*mi)LoR9* z@=)?f@EiR43ebOo$K(;DCuj%o`vFZ64*Wt|?8UXCq%VQ<1V5GUQ#``yP{iIP$%a-$ zjocml2~Zkn%YaDG95MrQ$d%%3u^Kx8my7c-pFD@>ljrh$^3OD%#4cpaC$aAd^GWPR zT7f-3^JzYbok5sSVrLKLlkz2+Ny^Seskf;y_&#MkzJZy9Z(k;3uZ0cu z`w2U2qKVgtbBHkd7r40#+yn+WK7suA$XgKdKH7&c9|3a&zYeM~C?^WPeo})#Q5|;N z6J@R75n=br$AaG>-f^M_p?89(;GY^8JfsHUtDPZ?{RFM`8bZ)o>Nl8BUJ7?P+_iA@ z4Fc;$B^S5(Cfj?paW3bfx-Q1h)6`} zR|q|h&{sj@&)7-gLmh{~T?lty@DS=>%I*ClQd9(A1`poDzBl0=2v-k>KI27y@uI(Y z(OcV->(J-zeEk+M(v(Z zGx42gE5heuA5JVw#4TbVcraYmizSeR^^k;%APG|;2}{&jVisbq>=x6Dn3EB6A!5!! z%zDHegP4O6b1L@vz7PHq=WB@)NR@8WOf?v57emkjb&#NXv@_aP%?%!=@yUZ)cuJ)qJ9Dm|dm11deB(gO-TpwI(qJ)pt^Dm{aTZQr%k;tZfa@)6S+R%`Cki{dEsc;JXjI zu0G}+p7i}hD|RD3hW&)^<4Xy>Ll67&^zJ-)jP}9N-ZyLCTb-)MHyyOkn)bJuyV`h1 z+f01gJphNHv<6knqAW6WekSvoS?I{wnfUiktjWE|-Bdisv;0)v+ zUzgM&&J>I)>DXmD4e$&(1AZ&MhKR-&CkxQK7U7$^SnM5LjT|=0jrd06R(UJn+vIJ) z-y`qA7s8KY=Vm5$X}$;j{XTk`2VbUqfUhifV{fGwdn@-rQ~nFzdS-*;6&T@h^a%K* zurri6NS0I#p2R^1b1{z8F9|K%6SboK+$q>4UWT25<@lEd{`Uu;2jO29ZYCQl zyjYQJ0MDDiaX)iBAA5}FVejw){3H3e5NnI~;9nu{&nx0K=*?}=huff-+n|KopdYtE zDYroxw?Tm0pq#(k!Tw3`y_)$xkoi7{`97F-Ad4x?@u?U^v+%V8@qHTY5yoy);`j=D z)6tnb&uL*B{_Zl-=O$R_5$0 zyrTOmbM`fGww^ir%d$zFbuwr1^(X#CF=wNgvr)|1DCTSwb2f@O8wJibA!alF#j{)| zFqadV%Sp`TWK+h8%PFR$Z^mB8ROWRW^E#b*ZD(HFnb&sawViowXI|Tx*BC|d&&BfY zW_izGd5>nkM>F3uneUP%-NStMGT-Bv?*{l@i|?{Y-hIsdSmu5XbKlS0&t>lCG56!T zNj*t=K1+HJ=6+A+egR9mVo5J#NiSka@5Pet;P&Xv?GeN6ktIJ*?-Ih5DA&5d< z&7n{e8Wjq4jklnsh&Kmr9mS|_GH&__p%ibSg|aft;jx1h-u6vj>KhP#5(&a!ax zEog}ah10IJaOa%@B3g8Q*IJ=BTF|K|{q+>LT50dJpv@Nahy^`mL0c>+T-qzZz0b9L z%Yt@W@%CEKK?^z*f--GFWX2N5bJ4cBXye?p%p|k~`W!-Ba9)N+0cvI_&w`39h(b#^ zw4xiXKjVgW!;L4LAM%_FSwP4X3z}|0b1Z0K2%;33OF6~LZn$-fyPzBH(hx3_p=`)J z)}Tb@V__)ssz_Ysb)nEq=uaIlbCU(#Z9xwZ3ku`H&}*Sk z$|dt1&Sggkm-&$eeFn(Rq5CZexXiB^_nn118iG8+f}DVuK2M4TWmu5Uf_mVLM;))! z!c|$&Knt3P+0sVVzok3GYYr#kK(~&^D-m1p>nSQBYl}A)Io;_U3ixFUddq@d zu^`~G-sgCGE!=Jk0?wPnp$9G8Aq)D}f_}81t`Ow4SrBN^xp-X`&TB!?lbTX*kp-1l zP(=vx&kRA{{vjwA?VI|Vz@N7nZYe{fENHw1O|hWq7Bt6#7Fy7yfS@70*Lk}`D;c-0 zTj*7c+Xnw@vXQ)>!5wAjW((S6L3dlw0~Yj{1wCy+FIW)etn(x2HO~1R3%A39J_3E&^-Ok|zt!C&UKwBAFZ$TFUx}0&>SU8Gz zy_I4kL$`Mez1IqTf}v-v(CvWUAe_&G_aisrJ;@c>?^@6<3)*8rUs%uq3nHAK(;l{P zM=a>)5QI~G%`$w^7L;H?X%C0WkK^TXo&^!J16k=iU^H)EA?dN_+~(lFtiuY zZib$+pe+_ep)YgjE8TE!G4B0txPyet@xr+{^pFL8Ye7F+P*(`z6gEy9+YOh*xV&z- zA`7RX@i|j+nh|3TPG&weRAPlfx0&(!Gc>eYicwbRbcU|VIqDaFJK%*DwA6xDTF^QR zy1;@iwIJk@a~TpDZIH6cc~}fKd-KVf1^48|24+Hg13qiU-lE`MSRVTIM}R8@$o)cJZj8^KaKqr zMg!oL#sK(F8+GvSHj?4rV~l|RyfF#>jmA6hUpLmlf72*~e?I5@f^i|>KjTy`DIQU@ z*LSnTyCj^dLH;Z982o1#|04S@F#b)3Uy(l|bQ=4&pfx4VXb8f;jpN+M@Li059w#4x z28Lf_{401zAaQ;V`Olj%sl-Rs7WmWHzZT~V(@jw1KaF>-sF!Mie~-Em{_{97k~l{3 zUuXYKbpzmwx%3wp^Jn(|jeS89=SAS12-cSY%d-tgjy&6z3%}4-1i#L<7=ELzk$i*T zZ;j3HHyL~3?>6wx4tTt=gAb0u2mfUwNu4Y-TH)6jcQYPTA#^hNR61ZPtC2V_8%c`P zFroqP&Jn6^;n24UFPk}jGsBlK4Uco^Z%C5mC0y=mF1Ln5E2tMp=yUR!k{ZsTf~9=} z&JK}s5$7|Isd_&PYvgp%e5Q8wHr)wf|yjh zML0!P%3Mxc%<%_v9c!4L&zaBbxZHJ|^QF`)6Nsj{gR za8?bq!mV7#TN!^V*X1V8p`H2D&iO3neA<~m?HuP!j(;YnZRgzDnZp|>l`1#xMb7(h zcDR%qj1SD<8iY_O4RJiujb;5=s;-~(ELWc3{5Jop<7lHm>9vKzQv9^jUp zrE_B%7O{UL=d+0E*}#-MVDnH~TLJqn@+lN|xIw~k!VJ~@5T`xFsbUzuNc)Ug&3TSw zK8#`xjAZ&pGPNU_s*xOXB&Qn5d}!uW&D7R1lhbC}sB|YwcqZrI;oLHr51E`>3CHwv zoD$}FKaNwPLpkRXrukaVzm4O!ac*s#PaDS^&OB^o9$v>W=P_;bIJAXBXLIO0=F&W- zzm56a#x-hVKDTk}v{71|s|H`Tk(P$P4*s>oTgcM$grOYpjRtXb6vrIJm?SR!6Q=MJ zrf@r#-jCz-<8qrgP6o%xFu8#*Kg2PPna*@(Fg+Pu?oy85z;$on_zj#-1LxVmb*xY~ zfs#Hn#>pb}2>fDo5Pol?2mBnv&UlP#2>l3tU+nNAnByBsfcxPbVJZDMah~v(kWZrn zZdGi9FKDckJ=8wJD~vMmA7&q;p`2x0!0=Z1`OJr0>}{9wFElF2chp_*Zy~%fmN`~J zBdoM@y9bz(5*m@Eo$D1~NeFPc0jiy9VM=zZxrkHDHO=81a=4}*&L@ZK)q``(;d@s1!Qs1vz`Ub~#+T9IjolaTjy0Q4Tzf{)?zwoYsW4!_fmI8toZ9C)#mSsLbeLGhJ#s0^lgi2Rvgro2P|Mh?`= zFdKyL!5%XB3xvC%u%-F?$=c0I!#{OO%$ypu@Qr$8i2QdlC56lG^$r z>>ZTyT&8)oS_OC<^$zU0q5A%Z`GcpbiNqOzccpn|8uzl0_HI*ysiaF+H7BhJJA4cHQl zxs>CKXSkfSgmNHuAI%px~FKS;fuT|F5*%)#-`(xOj$o^FJ zXR<$+{dV>{*{61lvVS-P9((T!+uf{qm}Y@$ff| ztefhWkBuHX$}hK#8P({QJI0S4?U$cTq&P=$K8RmAIS(7oEA_}DDw;~d&Mfk?*w+~P zN=Ra-7x`G(f}g;C3j5gI4NM&SPWIE_8`%9rC&KY=d;GFvhZyayr4WMoH@@JYuPCrL z9kojU7mH{=8TNiEhVAh45Lc>F)`&?#SRbkfc8Vb;_Rvb4$R`zarGc?g)!=zK_GS-M z&G?1=i#X8_w+5*+1&;hN)|O3Wj;p7b>^ z)hJvmjj7@onio_XK(Ia(&Xioq>Ch@TJK2d*$1e+Tx@G`UI2b2nw%{Dh$=HRc&}UuP z{~vMg9Gx|%_WaW6a!9cRyAX%TGybnqNzj2dMqAafKM47}M(Yp~>n+%KM|6IRbq{J2 zsgjh|NH(1A_cG3|#E1<19PHtXLXW;gX5ci@9@vczU-p)zfXte4-dN2-9`qG8=O%wv zB(cZ=yV0gl82)|7VLhRTl;|n(EdX}Z%^_bklF#v0G6s7GkusCwVDB2ii`cJ4eH3=Q z%|LvG69gYdVf6KN6mpNow;pu*--~LiKKT!EY*w6SL=?ieiUbP7-b#txf%F~rLY&-s zHlIT6mviw=`B(JyICgW=4o1Oy=p^T%P|h>>nuGF4;yhgP3#Ns>WN7AZ&S&6O4TZC# z?}h&$#wdkT3OB=lQr*G$I~ji${jzvP$LbBcP)L;S?J zg>cx5G=iBveX%aVOX4y40=r2Gj%JwZidR!Yt-`@31fqT_y9PmQml)ZF3f^SX{Dv`CQIC3E4ue z4{K~{9#lLqUx+=lq_G$G%}q$kc?(rBN%ePLwdEf|y4OL+m5!azFL1#V zr$hN|sX+Xh6QFU|T)XgmA+sOA@7}W)w=S(IpU@xS*WsJq*mZL|Bv1oHNJpYo`#cfUgS!;#Zb}tz90G+wCb4C18A^tWI z8=PX6cD5~@Z@=PVgl|H82iS^yaA@fksQGk}Lsz=uGWYc4wEWz>%G5l!bX8R*SLDky zU<+k|J-sMjK9--~H738>VUwGbEvjool)S}Z>uOatbyI%+@yqk`)j9U#_u*h*H96|| z#ZkDO7BHI773{*1p<9IqY91P0U)WnvEvu^B0T)Wbt+co}w$L!DbKSD#OIuo8YKkn zxin)D4FgnDT<7u=KA#WaFE{aa_0fj2LRed06I(!7GtEv8?Nm#QP?|!RIQx)i_q6>^ zNV^kDLXkX~C$0pcf`etQhI%6Ek2?^P$a>E!{tTvN-XW-L=EK!Bc_d@EM znCTj9zKXdppzmiULd{B4{!*W7^p<63mwCOVK3}QVmE%u$<>Yw(OFz8QTjIsd(R9Ug ziMQOB?)G}!>Av>=QSY=xf%{}g>fZUV)mrPVR@np9C51hB+?W9!1)W=+>qvG>Ruo(f z=p|Mhxz*`~7MN^-bTPR*_s=plIl1dRXG*+V;*9dg9jS40Y;s0Ma@U*D4jIMJPchN5 zoT2-Z-RWIVf0ZRyB^-Y*IW0NCsXAgEuH@wOm}7}mX(@>_-!77Q6(*!5Gg+nxN=PG{V%;vxO}>~=)&88EVFh}v@IgyNQBuE{`q?9}JvJjZ2Pl#&{EQNA(^sYj{!b5v{eoM&%`VW1aS-%%M&B z!_S*lTh)60$cowOs+b;$@ye0r?$t7PaQ*xd1+H|5&6$)RotT*rn11nuF&E6N@`7Vb z1#^rRZb(1j;}Hu_8baeg#HR?P@A?Lp5Y-m`Jwc!Ans_UpLdE!#@*{nx^V^LJ!bnf* z@~2t&9WohoC~mPop~aGEB$V7v7^N^MaXY03s?bIyDxVrjt{u_Q(zE`8*+;2W4v%i_ zn^`%b|C`ZvwWWI2hUS^K&Cu;|{MwdvEmgBe6-OU`GnQzER7CYM*1(j5m4VM*4y(7A z2Ok$R*g>E%pI{ zNjv2pTpNM6t1nnHgHjtUGH6Qh5rukF#|Hj4>eWyUfY&v6%-|&zrZpbvdcc~ny5ukQ zPIQ3t~aW08^K+*h5BCd@i^{+UB?}|?~#<7oQY={pe1$EB>!ev zqTUzJV>aP2R(Ap`kofBO}z#u$ShNt|pE}qbAT@<)#WcWd+jg z=cf>XGq}h6)gC3|8T!V75srF^D>=y({e7I9Y1o%c3s9s1&%=v{8sd_r>fLNJkP%L! z0vk6j8WkLsZc>66t1upi>$jXihx@N>r& zP4YRCZX&ilmEwwv>AN_#de()bMqfCq%AFCF5}TZ1Q%R{U>&K1V(1ITSUIIo{cSiE3 z&J=rgKKQ}Zao-&VS#jWX&a;Aguf|jF)%Xv^TjuzH+HgP9_}${~rk-!4T3YcB;wmJP zI-L*6x{kktUd32=l17dHSUiA>5A;aB8Z8kID~skxU_vraR%l$HHVOZdxsKcmk~1=s zQ(YPI;}myB3Vw7AmG8l|n#0&!@C&++kKsL!ruZfpE*v<_=Sj4qM!56CE_V>uT0bFDzIP+BImgSQSlL|=*^l^Q?Z8_D*8iW_J9i5 zAE4q_B;@!h1ta#?WC3}r5iARHm`l5GH5ze`@YUSH8oygQF!n=NV3lJTUPZ-VSuB^d zYa!0XrFmZ=Rw%c#2Xq#4DcpZ2+|+uPNC6| ziZ7JAd91h_`WGEBZ7eh~<~1~GVA#4ZMdy6Pp#E5Zr8>w7j&aahlyhdz5FWjxBsBgT zxLCxDjTdM*43O!oAv#RFUA@#EqB@LqsDlCxO0i9vLqQlKfUzYe{uLa@~=$NK>Xq zgFQdQI8u^aKiHzLa>&{c_uR=TE?Jr3bj2>Vd1B!!^>vIM)-*FiZLv?i0IQ$I5I4yJ z4Y{GwBeck1REIDXRp-Z~+ht6?bjGJAWn`qJWMm$)C&$F3*gsDvbJ(q(kB#o?kn);T zyirOy{s-2iIZ<6VIbvfSa-Qe-8;n_$#C8=lG|ERCENAW=TlvNc=&(Dh)}~g{&)e{2jQ=QpfM?md^pgL#urm96)$nET*+uRrO$H4m6qz z=7>X<5^*WPKvSe7xpPxZb|kuzlG9B#x!n1^di>Run4W5j^7QhmJ0qA9$+D}9YI~%` zu#6ADC_}PL)RHWNODxMjm{hOrM)enXg&M9US&_Pyd&JFD8)$|abp!J2ZCb=t<)}@i z`;^Vg6@!wn%TH4?T**a!lI>mNV=`DN#3tM2qfQd^USnQKOGg~YPbt78NnHmU8 zJ*Q>Fx~YN5bxT`X&K^4S?3R|Lb#gixkyW!U96jd3)~c%33&)IKKYaN5@uzJVK70ds z$7Qi3VU>?Y%ZQbQw|VT=_>Z7BLKfxQEMXcSX;Hq-+Fj!#R}S7bo&%2wPct&CC~!I3 zc!jkudRRCeW_&nqg6Qa;o>o}9r{9U!Wj9##|6qkb3N8L9(*GIP$B}Gs{cLtCJXHQ3 z=*`PQ;Su%OhgV}H8=5aP=hOB10596Z_@h>Sj7NE7`MzPL3y1G8>k|$~ec8g+`Eq@& zbYyX8{7$Tx)99e-Ae%$u_u>`Xix$4v!ha&q7k{U!##hn8QO^Xhh|IdAuHs>kE7I{ISPOMT3X1Dl72XTEN8Qj3MXkOlVNI-tA*CR$NH zcdoB*Mpja^(-xf)P3D8Fc<9Ynes3bM}R|(BbO>oAkM`G+LiScpA zCq~E0Rli0d^BxqkPT9nLIW#(HStfm}+j0*!vsi6xE4SK;@utlF(`w6-lftR*y=sLg zo)k_pv5)jA*tVI+2dsyOZ92BTe_=U1AB*OCnHRGr<9yQA$KRWEVSUe*juEOa{p{Kv z9U~JM`ZmTsG%n_${u3%aC9R{!kr6ZFwwcWvW>pgye=MFKK@lCVDxG~s&Z=rTX9QVO zWV2z8i}#3_SxEE!0~Slk<}@u{y@U#l&r7HatEZHgPpN(#t0F1s?&L3W13q6M_jQLu zZK;|*aKOwO=g)dvNp^Q_?lHJWLI3=?uKl#M$z{=cT0D&@-Di~sPYY>S$;VQ(^_Tm6<^ET5zo?p0Q8A_JQ$73qJcjN5(<9$%+21y z!8JpQ^7|zwx_`Z=mAH~p^Sw!#{!FJcGtm=MA2TpL!$ybv8U1^Ys7y$T?lq)$&*Dr+j4vU^p`v2) z>udeJJ+UcW$C5MT9Mz+)S7mc**E`gUNSfAiuS2im-VIu5 zCe-+Y;uG$TIv<*!Yy3WQg?tJ|>d5#9a0P3)T$=sr`1^6CBjd3~9vS}&T=7WORROG< zB*@1g*1A|Yqh+!HxX5Byq6$k4U`U<0dG?hD&W6_3+&+VCtsX5)D;$_DB*Yb-SzA1; zx9mH%qGWVU&cKCDgBH`V5^*uf6@zn1nhT3t`j$-?m^*k;Y;mb0IW{RXuAs;p$StbK zcgFeqHT12W+^;As5SG(y5ifD*65@;*>cE=%CCo`p1Fec{ z8gSVwuOAg)1{)dwplR3Y_|vTTI|TMeaQ>a$;va~_v$e15!g!M|Z4Vz&x1(mKaf@NO zKsOoMC){TI@~hfiq8tSnLJh6KQuPZlZby1ahNTTxs0rWi(LBHoy@!xN&m7iZh#wdNrR}(XgrvD)Z%D(u;RSt4<78}He*A#yn%NhO9C^X)8q(KT2~iEuJki)ZD0l&)9Ft3f zrraP)uS>|Xl#EFk$4C#$zmwvpB@S9IRZG(`ze-O|Aqz5;Qz71BUM9+w9S4|^G%%j^ z0)C^lKg0p5B%`@_vxAZ$JsSTWu3+W(m=#6hUlJFX_{MJdgSbLg$M5WhKY$Ik8oxFI zkCg>XzG%+%6E4SnK4L~qLxp~iNb4?;9M%ssqaL2@8n$}W=rzsm)TGS7q(J%P$~5&n zhU8_V*A5%DcC@^bhvjAH!ex`I=@*sDC2&vvo9@XnQ|m`6nP)m0{}J>f&2)5p>cJZS zA?!vym!|mC8#R76F3Z)cxr7hL-zPS5eRV!GlhE-$z_D_go=&Q*d_eFj-a8vhACP1tVXX(m8;vebb`JMY3qZeDkywGAs>IQ#(i z9)PvY}!ELj6tvIevE(krfPhn!Nlcs@^qyEzfy zzRttFe7x6?%wDDL5wWSUEPJUg$X>_tx^;5imB_EJ>40X7S;8$Qd#j?&VRcDadPIH${gPxRe(moqNi_JPVybh>QMCEO8}7Vk{A-RgYShBaJQnj`x7bxxEY zFRwGYIy5Ei(e^YqcCqb}zjx9i@3JUW=u99!F-_dh_nQ6u4=k@5PyJTocgq0xTaBmw zsqr6REiIf6^+S!{i6gUVmr6nKXIvouBG#)jd3{FLyD$~+wL*LHz+0$ky*`?XMM{;A zmzf{+eIveWNV-$L6YWVzi|;Cp&WuY+kNsz#H=U?tLcFtU=Mb5entc3fXMBn)S#`!H zB)g9fayzl7sc(JP=k&oZ)}(PU`c~c#$YX#by!7cdtOf-!N+r7u+h_6aq!hPfpFPTp zBHvE^TfEwunPiW*cRgR$^*m`wIlv=JqZ2XRN$omd$PDV)TncmWpyr@>k2$FE#6gYU zja_%7`81w5NO-&YU>5ODW6289v3H2)n6IhOJE((&KVYs!>iEQO9e*drqcAn3XEc5< z^atq|9iQ}pF7@M4qzyEddWpvFX(2hqn6BspRJ`xdaw;R2s4d!#^ z8)^9+nFuMaRA0rgUXATThP2oBX{pZfxV8jF_vte_P&T@_&uA5k_j$vpo->rIqHp%F zwPVJvY4Vo#&$chdbh4`AjGF2*$^%nsYtE z=L%ts4ij6T9$)6;G&=?D$|woWL+wIo3eirlV!Ln^kzEG(3)9qH}osb$z4hhcE*_n{Hv za6yFB?$K*+Jx&tslw@~Ya$adxQcQf59+LYEjTGUoA7}$yP3D_VOw~S3Dz7QX#!DdD)?h>Di7AxHw7Jr^R;~4l;8a!gOfNS%{uTku_XRBPvF8)12X1GZrN9GdH{$rK%c> znldrJp*nX|&&tfa{>4f0DcBz#l|E>ky=Y9ufU{4NUnLm*8hiQ%^z-(cSly$dXM8-~ zpC)C`s2;bvzG=zeev=xCq|8VuDx?9~B%}8tg^|l;;6`%cV8b0z zdt#^@L0WH7L(jU(9?>orWP{XUq9eb2NTwF1z^LNlQ7WZr_1Lj%hq>{>Jpo(n+$rh? zo=LnBo!PIat0gd{reI>%|J#JYgBv>s#DL;UO0J z5)+aWItPKG# zN-6RVJ7?6WwWel3>CnLV&`)dNH$xg2cxW{f{}{$mEu%|_S_}W8xH1AC%BK*P4e+Gf zL}Y%@ZDxLtiu16Gf%OdXd)UH1fR%ijxoZ6rZmH1Q+Ees%Oy+)dtZG>Bf43y zVwjtCU2d#5p~-0Q54e3cT3*5y@NTNgu%l6{h8VopH&MylhK{{Hy*BQe8fBqS~^n@>Pu!sx=N;cHr|XAJ9A zI;JF}X8Q0zye&N;DMKXaKb_NvCou*K}24T;!gkK(%s#6Uk5GPgf6gNaA6Iw;m3V~m)x$I%!BycAR0%vn`Rd`E zN0=`22rBUi{F*_$bG*zWsBm;ELvLJJ>S45*X0(R4Z6(f}_%rlu1HXwzm&M*5jl?D8kw|RDECJqSR95UZ%M0r6r2ng{nybCZ24BjZ_01o!M=I=OCk@z! z#o5Yub+}|RR>Gi*YXoeLP{TvMhFG{E>NmxW^T)B>k8zw?Ijk*`=780p8mk(c!`>=b zyxLV;xPlK697G9AKVC|~tY~JJ=3Hg}h}J}67JTsVpe+`@6TxMH5>EkEis|Ps?Cfl3 z;Uv2*Tj2o`xiSy1kLVmaX6ee~7ba-I_Kc{$%z~&lJCW;N-RcagCXdabRA)i=Nw(UG z1i60LVU%~Xpu*!q$gEuqJlAM@3m1suvbZBzGN*U>W@H$!e@q|ZYeu)j z?y@;nr4~PO%>S+#(Fp8%xqpS%ZUD&{~}D09mJ@}rZ{Cljm7?Qbx*yv${gNH6_n{I5JZfl>) zW~W4FINR#At|E4#Ffnsbhuvp22e7vzDV8&1*fU92)#qNKM+z0bHVasg-i~UK)o^chfA^?%oDHp*Fh- zPV-E=`}RY~ZIs00_d(S&r9P|lREdkJ{N}8tL++E<@WRO$?Pz-28fu3#^o@GXt&g`4 zh_!A_zT+07f9HLtL2u(WpzooC zzw@5cpts(Sej}S!;V+yfYtTEtff-(W9ngPqnyf)@yoy@Rq0&*MmZU^LIUQ7li+0~m zf)@7^uCBB`B~t6*W@86t$R<~~g!Y(9n7Qc*>J0_Z`#8nc-hIbY7)jiMRnO_Q2EFSB z8g+fe9(E$E6txnx(qVH2%W|sOHtc^AY@5G>y2>4nxM6N#)5)5-g%t1w`~jbz6tDFW zmt4hO;`4Ic4Li*$&m0R2$?f*oye|HMnm&q}dO2m)YWm~vP#^uR*n<#))XFtzmqFY{ zDQHFon{*EpPXc-*Ql>ZCChZyU=%&QP5?L!dUq{!SBvutTMP z$VZk|`pwnvmY@YbS~Tdbzxo)w>kMA}FU(LIf=FQ9Yy(Cr2s4zmSxIOc)9F0lX*!|m7-eb=g7+0o#DY+@r<0+>QoweTOQx6U*V36j!4!Gg4`H0S}R~?y`|A#_rB5%g(=fjXbo{3=Z$1sIs+tN$tWV zSaLw4`I1Acw5PQpv#_%Gcy%#ad`9}6q4s#!`kunudt@%-nvR8{vvXZ{ z;ZM9ezS8>mCjZ5!!y>*sQf>HKNhbdSug%TEFG+%-B*G#EElP>&Aaohn7s9;m{BGhK zw7YK;PU9>7M6r|VUV_7doJ%$nnBTKd8EMRJMI!0)J88|-O4&vfUBk@0`(dj+U!s{#L?5;vDxal)KdK&IsqkA`RBs_7}{Ro4+$O?WG!Wzf>= zH9i$Q^seK!GETLYdSQ7tso4NKchmLMNLIJ3Vh7CDe_H=jGq2Drfr-D*p(ZNd7uQxC z&cD(tcd-P{jT|*Te)P!5(c|MsjZ8I7x3x_-H`8Ym(=9!26s)NB2_Cd{bvIX#pntV@ z{02BUrZm=`Mr!3U{J9bw((=Jo6(~uu+TFJar&sjIIq{ha{MOPKcDM(99y7PoE zS+1+Jtn{>Y{cqIE8%``=QssW)1+{YS0q;!y z^je*XT4;FSaMW*hPln!Kel9JRug~2B$Z?)<{_{5pXY;wEl~4Mn`Pc6d&Z)@tInF`N zy$(FDA%0PN`q@gYZ-LM6Y-@gBf*ZuU4yJjm-F-y`di%youSjQ?pwI?vB>&c%;k9^U z`5Eb?ovso#?*<1@2W^113$Xxnvb_3~>`92fBp#G+TUTZlx9%(LJj~g6cm-3cb*8Cl zrWI4___{%|25@A3m~5DM6Gsha#+u_zAqqsHdA2g$oS9ZQJl7s?OZa6^LaEHeo{eKp zwJ}r`vY0gMU#_eC6M8Gb=R9<2oJ)p%FJ?U7yJ!D_3hb?TOzcLB9jC$`7WT-e(u9E5 zrkL6LHn(as7c4dw{uC0S^$OLA`tafkjd-t&dGYB=Z~b_Er4OZREBygzskg{3h4-*@ z{!UkK6t)P@RjwYs(^Z;DZ)#Vihd~cZGl0byto6~ypxD?QutB`M=m5unixx zbn@5{UlO|mh2Qe+fs=@dU1JPn6Yh!Dm7RGatohEBSQ5*1n$?5P*|C^kr~jKBi(Ts^ zt>ujMFLqNY(#?~Gc-MVo$NzjE`BQcCM(F0b!8A+ntel6FT%ogVo^D>#K%BzsG*`Bv zj^aa}ge<=2g-Lk?H4PTYhU(_Nq&Eqp3BNPYQVXfCBu|{?sqSg2^0+*1OI2%s4H#1z zkF?64!lkY6aY}#l{@|rbP~w_2=xrgoHwNwqwW&aJ1>%iNyadvsYWLpx57NmE6!)8V z?{pmMW8{KjqI>vv_wR z+S43r-=ih}g&AAeB8%2onkk8JDCRMme2HADabb1}8bwo}N{NN5;vTcn6VGL` zLy0=Vj8<1#Wv(sJfUj|_I)_X9l;CMlWxz4tx4~}f2bJdt&-5I2*#|16aSbj(Qv~JD zoCP}qdeQR~ zX}6jq8Fx=xqBW*6@4oPEv@dnoY!083J=(BhC|x&F7ig=q>XhES`i2i&;I^A0R+GhK z&WvTdC+bWvspLK-SMgcQs@d$Q3o5mlD!(XtRcLGNs^?57NW*@= zNp`7D1UGNXvOn=yB&9c^mww2{#zUr8(RTzrS@M*i*x!OAk|8)o1?pG1``$fhBssvX zPt1YdAJL$edH3el(x~9?lVCH%-zSSV7N5mvr84%LclY)$&`fq<&TgDj@`FX1vlGRC zvJVjY5PoW;ec6eYbNv+|hCSi*9=>_g;Y6@T`v*sSq8>5kX?DR@o3EDl^4X=qNmER4aN%o7Q>;O0PHE8Fc$^XqO7la5-g*Sh4-HDROM~7y zMzl+V(hiUYy^-S*QZTix1f|*W6Fxh%yGd)P-F*|cl+mEX7ZMbFH~4STKm#sr7LTsL z?_nnhY6V_yBk`et^R{ZXLwNg~=Vq^$`BbgkaxG@Bm-$rfdbvKc*UNmWcAca>s&Dyv znNP)4wEd|{8*e$T)D}RAnrkh;YwavfNtdocZ=bHgG?e4@C3 zkDH#7R9V@SqR~Q=3iFG$DJs7zyHKQ`(q^539g`gn*0=d5%s%U!u|=xJdQW{M<~Bni z$}LdqA6n2HZOc>{Cq!}AAxA?ozdCFUr^11Vte>@T(Xapu7v2ULmMo{2m6a9)s*PRf2$&Er7TLc=RS0U{+hqBvZU4xUffh8)U~$6g)n8-3-V#*3l6b$x zZ}aKz7cu+EBJwyRJEw}d5h_Jj7!ELIQ`kK^vuyKer9*O5NUz7Wpazs`NTXC3Txz9j ziesM(=8!<2kKQkCv-_C%ZN1Otv!3Wa0&nx{E|a`2t6TRK&r#+o#i2Vc@D;3`UZ>8a zD~#ZUh*7#gD*PO8hau)YCZpYNwH1y=F#P~3_^imz=KC3FD{AuyF!7u=e_62jCyaRw z{!j%D|5QG&HTbO+_>ICXTtIS zdFgz|h|9+IGWzWmuN}=6d zrW6FF*`z^f-Y~9XybGbHtLJ(t=3wT<%F%>kAQKQ(5eMNy+9EA%ce*i|pA$`unk)j|uf&mr4Pg)h^pMmv3Xahz#7cC0~RAf#J$n)zuL-D@a<{ns}*j=#IajB79qb6dnnfWnm|6?-PzjG*;m<$U5`k;+ zu<@Ynd$IA%qQ?5sOlGvcaZzSrb8V)nDO1}lCL0&k)-Gzyj@8$XW$U}T>NA}P@y#v! zb4U{cZmkVfN7z}7Su6VoCE~N(2`_eCva`VyYg~S4SJ$D-mmk{MdFb-ocq%oX%Z;bg zEe%n2VLoOydd&3J|WaedL4u8iHII33lbIk81-$OHS$1m4wLG)nmzrCYX;W05A@w)UJRDm zg8&w`k2=;Td&hlgzfJZi7MC^L6l&e8FL(072h7wW_>{4iz!yO#1P5KFl_edN)>-g{ zjd2#c)N7~%U+qikf|C5OuR%4rI}-I{jg2EI#qTlMowD%(d*O9H??=6Ell*{BUt{fD z*ST=(SY|S0Gx?6x8=c4ExOH;JhQYA|y90jNX;MR| zE%wFB@zbbEa3I*pN?v0_&<*U14xjyP9XpyMYq=UQvwoBUhhg+b!ccg`N<_zRS1s(@ zCb`rpc*9`v6ZS6f(aH5%nrJlZ`#Fo3RGz#=h}x0$7$PK<_K+}l{+?N0Z8UovHix&P zs&l5HcDyN(uT`;kh16xBfwjC!e>ErVmd2jy!TlBt9Hk2cWUtxkvpL*#OWLv{DMW$^ zT9{;WBhZ`2^qR-)R(iH996tozBno--SyB9j=3QTk^(N&@44AA)vP_lH1ioDces$NX z?ufT>Y||i2cLGi|Gdb9VXdvw)wGB(UV?CP(YX?*D{stJ^)(n~JC-!QoKk(2k(_Rs> zlSi+1O|EG+Ry9Xb1IrtG_v>Gr9%-l@t*={DpBb*BF%2Gou!s3k(hANSI^_F*mpy>M z&(^|E`M5a1>C+Y7(R(BhBH)PQPQ;%<(@mxw-?ZAmQ?wR8W`75dAA`*vn$BjWSU$V+ zGPyL*4#srZ2anGm7pIYYCV_at@+Z-W_Vx8`>PdE0>w`^^j&+?~d*!1||BQ9T;`#VU zN5{zVealh<>C`}K8QWJ;+lSH}Ydi42akyErTk3n$9b~iFQQKR4ZcRsZbw^G6&``lY zmP>ZUVqM7`Hgu?WK0xn00?(T^)R=bzxkVJM~V-#&N6P-ab+{ zxURit%K-W&*5AncMtnr;oI^HGd0{Vr&T;Rtwcl9P6zSW)ztlB#qqTL5>S~ATs9q6_ znm+8qavG1`*_l3SLPDh}R4>)TC3%nsnOSJk`=V)!aPQ+A`JLKhWB; zaA8a90D3%D+#&pwEh+bS8WgcY5&(oJ`K@t$Nk&Ty8Vtms4^nf3<}rz1QM2 zx5$jS?1hgMIL0vAt7BD20D(1-C@g4Sg&vkmp6zc?B2DT@ELU7cd6}=)=e0*qV5kBiWL zX5QeCJaVi#nCv#&EM}(_frNC{NR`iT5}Ctdu<7n#YF>CE9culgG%cvA&W2q0{Du}<^Q5|S0wVh>ecDKAv+Crms4^oA6d_{vrrfk zi&cL&T-4OU+lrG)ZYh4m5(xUQ*}bHxnY?ktC6^W?tOxuJC8Q%@{G4`w%iH3xE2`eC zw^X}A$?C8(;4r&vP$$`}=TOT}^VqADx@fqms;1B6u-hC~yH00TU^yMLDSpdW5t9Ud zWsak59#5b-QP%-hghuTL%+6LJ3JUlS^uBw)NGBW^!pD1qCIrOScZ_<6@vnw{cU!|`mJ1AKO z?TDit5!m8)lv@JES)Ml!tEAVO$~wP$OhQFcF4paDX7W)A_~6%v3;}|h_`B-aPnlb?0s=hqk~64JoZiLjOeGT}$e#k9*~6HDK15v< zVAT$(tHeui(nbMGXafY2_utq$4o%jHxvmHo* zb2PN~HRB}f6-qK#V{r#kp3J+xiexka_tNvNxc)reoo0yp$qu9Ew`$if}#dxNl!QL`SsfMoA~ucJpU417a91?!>Rm(wd>H<;r;EnelNc+2_|+V zU58T$m5=uWH%R`2Xx4`jZt_?qiBuC!CSG)d$LI6Fg~;n=eozd*-|P1H>6x11sffNY zO%mSj?aCm>qmbKTnmL*3a9Cw%L(HtMDw+{@Z$c&_Evz7XN0d<$NL`Rx=ha2p>G%GX zx*03q*f#?n)DU&`1Pc4leg{=*M~yM7P_`x!&np{@MTt1gG6^Pi7p$DJe26t*9oaW=u77_wUb+}Wzzu*@g z2pEbkafl(DP&a%$Ud68Pg~PtW`l`q0wY#S{iE>R^xvo~9%S$bVz(IQ))W<}Wsm~AG zZMm$ss<7T0iFnx+RgZJL^3tisz$1GKx`{fHK3yn5+sPxlLg*CAriGl+%khq~-x^K7 zxkK!3pQB;47G`V70iPwTIwV_cL%w&@g6htRddVOm^ph6?OsSP@!h!H8gN=0qHPN8O zXtu_bOjGB^g$)y3aR>r4WLx&?oaUpJ8R(%O7sFaDlW1?npAcOQrcz@B!JvSd?l8b1 zk)u7m6LQ!BpQCHaaUYoOj=I5y;tU9lYvwENcbk(vin&W2w)as_I)&e7G{JOSqn5(+RP;; ziz-i2q+wglf*Sl!(&vj5w2HpqGeNW0&(2~kh>%)wj7VYdThjq0iifyva0=;yK=_rqo#jKOtk<(|j)o!QD ztXWe2jHgfJXW!rY8Otsrdvd??0C0YtWH7*OC~H z=()uWnB z(sIS>vYXAH++^c+{f^@CqD6|su9Y-XmB1FI6iMulHiy2jToQdYRW-1i^nueAv$Pj@ zN7ZSwd~&tPq*Byg>{W~k8`+Pc`Rx4#BIgz-*`neP*zKT5y~Thqg_`D>%BEq<3q1+Z zVql5a(~wwC!}l@zV5-ka@qit)OINyIG>9XhWq4kAn>`9n3eVHJGK=MXxC!V0g(-TQ zqV~W((xcrkRfB`u^?tfj{06_D=m893`TMl;@xDcTox}U+dHjaxyu11o_Yn^(UPa|| z%vXASPpNk@p%tSJXdiji_~Cz@)k(`=eKq_g|b zWhS%TXfEl+e)YON_q_V77QGL$NZoPgEna-#v9%_b&5N>qD0>LmI8e4W z3QATG98KiGLvvU*blIKi5EVG0x>UU8;Lff?r;OYER)?kj*z*@JzTmhFq$01O@vLk2 z-1GZqH5&-;&SIXOis(nsC}VsiQ1LvenuyL*AL31EPVA`i)DSG-TvU*u@PPOUyUxv$ zURp98Vcu+wUU9h;L$cAs{MnHvpVMZ|b+}l26KiQ>&U`M&nEg@a$W1n{O)W^c6wHCd zg4CW30}WA&-fK16B}Xck*|T@XCMwf?Ycjc%!{j$pGPfu=@e6Qee2+}aWOd938ji!6({8I$}?Z6jBamk36bt zDpCthgVV@?jh12Y3|~&bL+l;49(-U*T=j3VD_terT-*V8n0xv*A8tzP59S*D|Ue!hp2ne1NZ9zNSkUF-pXTe*o6&^$j z4C31NIeq}&&f$9@njOLSBC!B^z*gZ+z@c9RzePF5JdWoA)z4}tYAebUuHxm*D-9(?oi;p$63S2E?T+G;@O8fd z_JyzsFVV_Y-sz>NM!fISsW|a^*oKz0F{|x%CnkMKr($v%6B)x|>#`*(^l~bca&Yr( zY%xyQrVVyvmnhOls~wA?p}mG(KQ_%I$!#NbR<@zA=fo0rxm&fnM3-H~oFOi5opc%Y zH(Dhhdq_1HY<|0^zW=~ehmX4Zw2?i| zTiJW;#svehj=^a2Kz04_(nL!K@B%%o_)XxYT39x_PnAbV3`|mvhZ0s42}r{lR$9gg zVSIpNfMRT-8|i)++G$YcTy#i(Vo^T7xF@x&r)Two6!nbCyJ(Y(>0J&apmX=Nc6KN8!Q_e+i>qMZYaZOMD?2mj?zAs!%}iwL##T34m5{6R zpjGV_y_j3Kxj!>DnnYlLHblQLSv@9wINsS185#`NPLKC`jCP9|rWB+-dr^7Dn z1H#7!G?V5Qs*d)d&ZJdD4w{kzAFU&*%KnLPi-#+*&O}wCQ5Maz8$l&3p}1#Y#evQF zW7gk~4HAb_JQCSZZp!4t4y)0R9B(?EoQk~}W%cVOmhD~ZuOn}Q!B$?%G37C~WSaVY6gHK6tEJ-_zJYL9HhR5!pOXr|=5Pz6?RfP0cY8TWwYLmYDB`r%=DzTodDc6|p8>#M?3K`wt zy#0Nta9cFm)a+cnSK-4Ik%mBh!v=ptsE+vt$L0FSg0@I+d(>OwGI?#PTW40QJ+d<% z3Z}v~_tF7pQ#Phcct>lShU%P5tc@B~)z9?4urnDBCjE-eJ9)Z`d6S;%8pSU;lYu}3 z*F%0RzJ)d1g+1A&H2$aqXgU)AqKIjZigFB>h?uYJ6QJRkixo`p4lD^6U_Z_tKV?Hc zzu}be^11JznQV6EpuXONW}33g4_el}zO!RpZ*PlDXO#UGN6K42Sd$#A^Q4_-=qdF! zf0w>vpD!<7a>_m(9s8WJWHHjpbv8~PJTP!Dc{6M0ecIP9=-ewGks}tfWH4JGf!6jX ztR{obYzZ3`h*UZufy9ijgOf^epX+Ij51wU&iEHv9ZDP4Gq}b-+j7y45%G(F(Nu<96 zgM${7U4m`=+@x6K$wYm`2VR=#4YDZdoMy92bv2l5d1t`u?6Ml0`nL4;Zt3gW+}pdk zuVrad)6$lfsivkW_63{i=dz_H?yWW$K${dr(g7zrQ4EDtw|T~7JHuhOE2d@gTiEaE z-_p~wrGH>cPjhyvt!-I0yR5BkDogP~5^RO|Dt4l6@bv88ncFAg?x1L#E{t!?JwD+$ z*TOHLA8970+h#Kl4U%oZZ>&zmeyGhk;P|!8DWB2!RkP0w^GFyJ#7~qEV%sL3+#ouz zi{li(giUxw!A5-HRi~9@3eQ`eY@N~a#v6)(F;yu%OXV@sZ}mo_{?)?U{8xjG&&m+{ z78^ki@qH*rDLdo3vJdrxx}kJ>sIFlsl^&{3HZ>({nwr?R;Lu$=Tw6O-n;EI?$=9H9 z$wZ#>4BGk&Sz>R_3yl`n3Ez->Tz24GPucM28zcacQO|KC4@vuzXaj07hWo$8?!@u_eQyZ=^9TlAOymE=1_KJ|~yf zmf(^cd!yG0D_b8vR&&{g(P}GRO_FS3nhGn>>Pi+KD76Xt`g_G|u)_#_03-&eXV9r7 zg#fLbx%T$b5T+3<8=y8KimQzT+$wR3$N6%P8%|%#3`WIZua@pAmH{ z8X6s#q!wG$>W;BZRf7wo;Y_gBd#w>;w;4Nz>ydd-ll^zX3XR=aW}P%?v%jvW;Akn~ z=+A_rKUcnq-{)u=Ae6B{8Cw^WztD=1gTAX8yoki z7LAsrXH3tWwsiQ|DcNrLiHNW~@%CP`#UvYtde~y!H<*;Du8xb+cVLO36Tr#q&c&KZ zn$&(_5-XOrVpi(7vo{?ik=>>)p~+2SooTo$jW6A~JsW!$?AzVF?}FZqJyW2asot8J z-V_{AKN;i+U*=^vA-e;gcr}?$voZepyTQ#pFb^2q(%rpfuwgiz9&X^D%vMPVGdK51 zC0)#pAGsY4x5utU#AqxYCGIC>5mA02My@u?a*)WVmw|E*#3?XXKs;Y*D$HPMrJuCn+^*4Opl%^YKmLAd|&Q$MB<7Ln49oHtslRi(HQyM|oN@st7MCK6& zz}4=y{ur~BJ|ig;KY3%y93IU=KD@AQj%k%AYNXf4cx0_56nx8RMG7soG1jsyi&!4% zI@Jy&I$QVPUfl!L><94$)Qp}`VW`Td1>B-|Tmci}4H(gT}5K=D`cmY^kM4_PPW(Q z?U*lk6thw4D{0OiuvsEr8vp}}6+1iTPwur^CEw@jCIj~G4`5HRcS{r|k2o3F-XR$e zDhR%_-GQM0T;2nJo#&AW!1rlUz)4ggxe~mdY?$SLcFzlJP;G-UMlCAoW=o*hQ3hqV zTkf}Eb?l=HeKxDB@I#YJQ5VwBA=^bwlZ}m&O-+;8>||4@*5|7Y1?&9&I`*!uPxg9@ zeTrgsDa`6JqXZ)!>9rM(QF@FXFTN=VX2WbQ1lk&q0T{6Pvdfy9iMwcST9zF}%I0vk zswx{s`esPiDa7iuOZ8$oD*FN0zi@-nkj3OAmGS)$#=Mee6l|IhQ zPPZaxO?562VmeOAQ}nrr1v=Bik=AfErIM&dLh8rCG`XPgbBZxH^eRyQL7K-t8RV+1 z;45fV!DboSL^OpbcQ7Q-sdIk0W{F6J6Xa{5W!&~M3)w6VAqSuq;`h5i)!paYl!{*Qd!wG4Ge%zn`m>VQvG0&dmsGoj4WBEY9mZXC@h{>= zNWRoV!@?nzJ}M2PQXiFm$(W!T$D%a^OF|+mB_MLqH{P94Ka$-YMZQPp5=}*KY81CZ za5c5;I5#%in;YI;%fGLn&!5)rF`e*d6rf#$4z!G?9H8ya4r{I zU<+UvJ32NEx9k~#caX!V>U~4C4gE1deaWX`$&4hOcF)oH;)Xww>yKU^a$&%N-|076 zElhFRn9(S~m`_oB29wR$<&tYU;;FIGiS+Q6u1Lt@x7Y&)eN`kgTwgOjIuuFuLy03n z-%1v(+kvettu{oXODw31av&%KQo%r`0Zk~CaY|BdJ4}QO=eh39!W!D$Hfsxg3?kj zF{GwP^U?c*7MJN)vh1~*9EN8MhKmChx9I^{_F3JQ2My3{Q|3>KIUhrxLl*#!hI+2l z2;xuhmHJInsq){P)iQ8pz&U098p+u`qsJ~;a_l0X+wN~$(b~4E-Gj(n7_IEr^tl79 zI6G%4#nxY1*h*xBos(bL+PX4Nzo;Jg6DgmhhcXhC_eKzaOdDLtD-)BMrh`BYcb_M( zmT)i-Vvog#>7C>8LT7+|@F}_T*eK0;y`HAOUe7Q#J~NXZ_cs^BqftZ3e}nEX)v+`i zu(`=)(a*@elc7q9Nvh-@J}nCmyN@yqBK3mzlg~`YqXVrG#Dh4#>LFu5Q?PrS@u3I( z=6pb*x1w!a1i)KKd=e8!wcpEq^1FK@2yh+wQXnSX^dT;dzPGn9a~r zw=yQ&%1$P3tJDL$Zbt-DsbDafWQ?DJ!4&Rw@;&_6?yUei6~B(%WN=y>NKUcB;5gl- zzzSfk#b$@M@h9ImD?krcz?Gs0JKcn%^83nTe|A}f+HM6`VSJlDpL7NSNHFAQX>yF9 z<7qg09@*`a(&sm~?opgg_kToMYXZGHDFWD@j*&u8LMAWY=!4-^i@Ry z?!xmHyAyjru-_LpBC2s4@aqFtTGsS%E}+B&@b9Hnq=bt%qdtEu=J!R*=ks_IF65v8 zrc1EWE!^&l$9?#675q5B%9UI8F#Cu-zB>#<+d~pS=^Ywwj-nz9XY+#jcXo<~Wz~BR zng&v=`gkj5*hh$WM*Y^XGnVI<%YVUEh4VBtbwA}rg@-ME{p-kPZnxsZPTl3TM=J_U z?{0KKyfB&zD}hahd9@ba6^qDrMw?&xq%X;J4cNx>>G_&Vtlqxxa<{U5rNXyOW}nSy zH?dU~o7r0UzR|umpjZ_7q4p;Xs>SYSM*Twux7Fs7AJED4hKR5n_F?_>25{qPt?&vP zX^{14?JdYS;V>0$fJ)fPW{h?j$wSPAD|KwW&Fo!n(*I0$QK9!EK9*uu^*?WWOja!_ zw&F!d=6i_M*tI6BAFUOYMgV7X8c|wjOZ|1AY7envb%o>2*qJoB%(|c?-xBddja5{A+6eu9MWU)8}|GP=_X1NBHQGs!| zbDJwoK9G-+3uq~~7+sda*;X5ne5}cBw5wHxGbDBz?D(fZKB4MTeJA1d281)f>viC@ zgBDav=ddc^MUfv9+KKnY1NeAC9p-a-)b7&U0411@g;jh}`yzeb?eYhlZlC`dPAa&U zQ+e>(Js0|2tOg{)S@@mHe+I^u*W-lxjb{G!>@D^lILr}rH+TwCYiinP&WQ7kSIvS9 zBpPqu?H!dH;QBl~HsDblOmbRPPcU0uHN2?bqo_=B*c5M|K0NQbKO5uMBO<@t#x{C_ z9;Zujc-@GykMC}W>hyc78iRN93#vUB^wkFOJ*h&6&_o`EtwMp_!Oq1#_ovoB$1UK9 z^I7T1b+^FL0I6nBJ#RMQ>B!@^%3t4vN_(BELvaMWfm%PlJ5+j2ww-1|E3CjbvTJBv z!%Tn{g#67)^aDkUoz8%>fM~-|kNy#Pq0jR)(Pa~2JktT*H4|oPFfXp*{Nr552ng|oU)7+82A&{yKt!#INUN_kBiN=V^Y!FG4K zici-q$+j%1%Z;UO^|gkon!?ptzqdAFyE8gc8*cO53ZKooCC%B1hL+Ljoz_IH*WXkf zZfdGqwQy)U#P0HzM-?q9#HO&XGTK4yKC#~Rt_y(RhNn1uLSH>oelM!Sc)0) z{(cLQ6@O2o3V7rlJ=b`G_n0P_5;wNqi;P%a_^+hu4b%3OBmPX#~guP4EZ$s-QH`V^n;nHKG+Z}B==rR?;8X-Z zqs;lsS!AE=6WPr05i6RiTRQ{FDYOrArOj*956|=*Fg$d?g4DtWz1d^CiZpCzxC5On z)lDn5iVru=96Hp#Y0t?pNT%HnS7EDR#Wl+{jeh=+Lpluz;P2wuOCytOHnlH2Yz9KR zNYN3*?_tl_PT!KSD8*h$9JXGhK*kn?0gEbZpW`y37Dxu8cgK_Pvoy1y-evZhk*UkT zergb3fsJUJ$z_1kk54?tVqu;gS3v*$0Xa=eu!-mtFMzB=xddTHPGgzxUrMu+4s(Wn zaWH&sRtC3{)tgL0J>`uBd5MuUs#P{qh>}RV;Jz4mPcQF^#4*GNaHg!-+})0m6MHO)YFJ! z>ufH=#gPqW6H>=+He)5xpJ97Q4|dWcvft=E6xd;cwO}RjT5@{EESAEPVh&bR@HLWl zTw%APmnefoHF6|!zeDgL7{#=&kC$P@KpzI`#j{7yCja~a#!@4PFU$-fG2~jCY{&jf|+*x8kiuu~4vz%!;C>+(RW zgY~wpuDZRJZ`hJ3Das8C>paaV+3dd5>lBxA7Ve*jwCbqd*c7#NPBK{ zQCn`Lu-|UM-8=cY2r{)vH#DzJ%Y}{qSMB=WZ!vBTBpEpix*Mi8se0tjX1BFP5KlI)$~!$KqZU>dy!5J|E7;hm3ATr@KNFuh8kP z6h$4ZX%(wtWmb=;@IEr$*$Qu3EPLy8f7D4YNcgWKt$nk&giT9d6jS04VSz&NA2$h? z0A4SSvnA3d*lYgq;{@P0fy#Z4!-oMsj>9S6*d3sM6z!>1>_&Vk^85Lr%MCB7A7lJp z4S1(G$!?a00KXO5RVx2C!iExjm9)G9zX>rg&H=nn>|@`PItgxlmzRGtVx_=qOYC7M zNizhmc^mK_Lpy&8XvmJiX&oqz))x$%bZ^<>GKDDKEMA-AB8mc`^D8> zkIm=yc=3U6z%K&Wa7x%JP6-zwLRLvWoh@-AQdi;0fH>v$C?147#|J1zRBVMe-NR%n zkNJttp&^M%fPX+Wu10FWJuHZd0}GW2lOndtx`i^Fd0kd{53`x%Y$l9AR{ITjFRcXh zxyE8pToUjcYh~52zyOBOJX1)W)R{7eo_ZCgMz`;|P-nF-4rL3PkIiU|?WoRps*}yiT7JOgb#5s2>-x1H>~a z)}t$)Yll@a9KZKSB;H9R#6_`al>C26be49sLsw`~ibPRsZf6N(A1_Lm=L{0h= zX)yS*i7AMU2X`M>d_lJ+yUa$t_)EnRtg$<)R7?%C1nr(tx>xNnQJ1|A&e1wv<+?@44%lNEGfD94Al4n z8o}IfgZ3gZRe0H##M%NTC*)ksN0Qsm!$jy5n>|<;_5`-6cn{58 z-`u(Tw?yK!XmvFh6x<{Ho7J&l>hnA;FuW;5oY#^PobMw2yRt)s9xAWahyl*MManBJ za)I#Q<(1R`euKD^^15gM-$1|1tDFJ(PPE_JI)P|p0@Y_wJd|C_cO;3jQGLqVIG5Mw zWFFFmiHAaO;&wY+9x9<8J`30334^$f$Gk7E4v5vo-%$2xSR}5aoYD6|2*y)odT#*N zq}4;C)CR=%qu*r;YXG03UDxQK0rAbLp0&k8ggbT_nQM)Hr*TDVuX&e|`6pPw0O2&R zAR^5lpxOchq#M4DL6~XsHS0{~KsFSFH#8Uro86b9{4JE1Q)e*B7@oek8^cr9I}~ex z`d9}fyAkDSZs=#%@WEjJ2(0Hge@C&a< zt4RN)^?jL%(zus zqhDxqowz?dWZxtIym?&CE(GDORTjGuY>r_JhE_JkWg0W+!Ol1=tHe9{W!OC+*yJ+s zlICRwlL7j7c>zjq6o1XyrF}UMRT(413et{jWqtGczQ4-iKp7|;xkO8a+e|j^D#BZ4$rF_L3UG`xCC`kQ?nqc*)Oli|qzQv8yIqCfw~eLQ@HI-NoYf z+nO9ua{K+wRv@OiP2+RkWCrQS|A&0eo1e<(WW|rz@zRMjb5KjIYF<$EIed<`{Cc4j zbBVh4bXBg|VY5p}=IUgUCm9HJwK%O#;&MI;`(t&H2sj&O*ypbCgv0g{kJBc;&5nkB z=sX^0RZY#uH8tYf)zN4*@KuWy>36%~aWv;P40TjaljDIppfgWAPD)j6;BkoiG?>8S zs5Y<3=rWmh;&Dvo-S9ZY_r-VF9@4S1@<55vEiW`6*Ja+f3_4@}B)e8`F>DC>6${Nd zy&}H5)0~rb`mK$NcOXq3{h2(@eX~4{F+GpRk&cEd3AK|#Ef9|rYQTbqSegF^yvPyY zMSe}xhiK-m)aTqe#D^$gaVvNcx*w9v|A-ekXSYHmJnf6a$N}>__r0x1Mp);0M}y_l zSX%&Y^`{Wq#sUmOurSL8i(-!kt7;DPB>gHPP3y=x!3Kw3mp4u;NqOZkfZEaT65j$J zg8TT&(p=n|O(RzLXK@0bE{R(nyyKn+8B@$=w0gif<)=AW39j{l}uhRez zP^#y=>*!PR1HDnI;nayX7b*sEw2%QmaakDycRnm7&)wg zbP(c=9gdPZ8PVy3_Bo|O>r1gBN}+$s@;Hf&g!q)p=naKEI;+`b3^j)#IoTBPNYxtj z`20GnRkdm|6PIAPPlj+aA4z-@q^tm(?~}m!whPI~}P z2;#fKq0o$-2VF9L&G|F%-#OtR;TYjm+ZXVYtxnFmh7|}T0(ImvRi#c`c zPqLn8{hswk>mAkytmSqV7ttAz3ndBqqg5FNoB2azVe~EL|v_JQV&y4P|s8^ zQZH9O0?l{XBlbpnmwm)OZQt8|fc;qe-=Ml*?bzfv%yEL_Ovgoz%N;j5UUdA~@xD{Q zs^E9lIGdddoMX-v&P~oEou@f3a$fEHfpfd_5$E%+X4eAOm}|A`4A(DRFS`EhdfzR$ z``m}SzvTX^`y%({?rrWn-4D5+a=+j)d%T{wC+q3ry(8W!?^^GE-mTt~yk~hY_7;3*U(lEI-Qv5)ulEo8r~G^Q_xB&;Kh1x> z|AoL{U^1{KusLve;Do@}0#65C3j8_nUQi6$f}vo2usb*&Tov39JUDo4@busX!OMcz z1%DWPAoz4uPt|DE*Q$ONiia9QU7?ZCRA{fzexa*Ew}kEvJskQ~=%vsHVIgb@d&5Y( z8qSAD!_(pQ;RC}*g-;HDJ^bD9pTqBk3z2xFG13`1EOJ8RN0G-Pzp3_AN2}|r^VJKh z;lWUSe)V6gKZ>?S`=aB~J)-+W4~iZWJuUi+=Hn4jGYuaJ9bIzn%M2J zpT(Ywy&iiz_V2hU9*8I7E%Dy?czjL#fcUZTug1R_zdC+P{O3n)&dNRE_y)k`A`ndGz=?l`ArEg0AB>hDC zrS#v@A7#XhEfdNdnmImmM&{zoL$#J#Uu~kcskW!~*xH9{e^vWZ?VoGkuM_Glb-ucI zU1QxobqCdbvF_Bm^Xe|GyQS{lx~J-XS0AV!sXxB{rut_ZybT98e7E5rjlGSV8}H3p zvK`rV*~_wTHB~ijX}YrMvF1SY9?k2T_iH|^`Pk+!H=o^nQS)WZ*EHYKd}s6h&5t!d z)BIxd8_jPuf7Bwhm|C1IRV_6wjV&E5{ViiH(=B_pY-~BD<${(gT5fH*zvbzcS6klM z`CqH0wWD=Y>(#A4YJH{6*0xXEkJ_GW`%C*w`*rP)w*RgD{q`L>PY%Y#xyjrfxxI6n zb6?8cnEP!=}v5$<5d}-vmk-v?GN0*MCH~Py(wnc|5 zx^mG2i{2aSAN$tWQ{$QOHRETGKQR9B#OTD?6CW%-ZShYQKf3t2#jh^@`{I8s(Jyf< z2`^c+Wc`wpmt3*r#Yy92$K*bfXHVWS`M1f!()`jRmR`2>(WS2~{r9rOvifC%%eF4N zeA#2mUR?J6vQMUTQ`RZ>RA{PZs&{H+YI17z)IL-DPaQUO%+$G4_fEYz&8Fki!_&u4 z-!T3BbaAGBX5GwTGuO|&zFb|NSw67*faMo2zij#Ump`@q)#Yz4e`oo>R@AJRT(Nn@ ziG1b8t|I&D{zo7G(0X;j2ZAL1nO1_|Hm-k?&M)4#?yJR3#e_blI}kq&;92+|Y_l@r zX@d8^a-1Rf#4}Jf9e9WIHNilKc#g1IJRS1IryWlUtM$uq&y#`~2c74A*WDzn*4+$P z_kZCSER~tV`3}_a(~e&XtEId!ij{hG`Ct;hquz98Qt|INiZ}{5Zo%g9Y^sW zILC2*2ge_TexUXAacb@YKF{Jh6gC~F7G z+$h98^LQBd{Wp)xgqU=%5Zw6?4?$ZN2(Hg{oB>_Z&c_z)QT#WK&82cTYG>&@p_vZ6 zYXULrvY&cTT^8avS9@-5-E^aX_5ODcM4JzbN8ufxcJ!hCeS(Ms=i*y*7G8p%*h|?Ps-;``D?!UP?l)!MPS_`AyYGK|g>#h0bkbM7M; z_0|8&V^3(MKIMJV`=|?z7468f3m}=#9x;43;TXWt%PuHB%U%#_KJOuJMZ}&j2#)`B zz|wQ#zj}m(xcI@Y2YW*Zect!7R}mX@rvpYqVd)hdA7U)8=i@od_CW@Py#+VC|6}4B zSVbR$^|lG?haI#nA=tzQ=(DSZJS^tAac;#fa~}Dlk%|eiP$zN!BE&o0hzO@E5z{n_ zD5iO6vlrvty*U1Vx=-|?4bP$eFAI&59(8^J763b-VKxKyH=&(DI}2YxLD&Mtjt&QY z(;+;e@iW5Xh++8>_5z=W)$5;xc5wv1pF`YA9IyhdXMcw#{Be}C25np<_{B@nFG0bM z!@=ze1jLv-6L`V_7AY^SoIP{cwYYYe(1^pwt^)qA0{+eyrf@`rSD~fYC?we_h+=gL zG&g_24(Ur6mroJ?SyB(iqWA1d9E@;ZdAn&&$8pl+Ib4sSJ9q zv2;e95Yl9k$CGdgjtLxeosRNz<#(0gt?UxK=V9FQcl51G7(u_cBIac)`!nD#VhpW; zcOv=^nwn7iyWeQg(>1CuU8{wD>wTOFKDU1D z+BIID)-Jpg;~-A&#<Q0x!|Z+7)dYl#}j@m?CP#hsYeUNHS6%X^P|{y^-aS zwUGlNhewW&oE|wZ^7Y6ik?%(S6}3d|QBO1&jYQ+oR212_qTSKq=wZ=gqhF0)5xqKk zZS?x+P0`z;KaSoV{YlIf^Ty(_R4f}?7Z>A(xEgoHJ@H^X60eVs#rKJCPCoXr2yLW* zmXI%bw{Ur>C6}{X*hB1(>@VzH{PftCa+E!l*YAk;I~1%W+)_6Wr$;fcbl^iBIK z^l0Z|yzu)XMSh+0SG>M>UGehbw~F5^UR1oWcwT|#=70SAL;N+oO%Prym5guB*A9Oz z_gehbS6;mV|IT>zX+e1PH?M*|;d~GOT=VLhSBGES`s%?;Sq zGVuz0Q(k=D@Halz9;TuhNw@fuqEZ|qW>~`9fDZFa>0AwtUtg2XmCl#GBV8liT!zh_ zrR$^{q#sCkOWUQNO7}?*NDoO5OOH!$N&nCpbatIr7uCh-Zk&IG&*!1-d`eR zc5zVbz_>aW*3{=fPrDD=9mcwo#H<j07Pm(hJA;9mwk_2 z%YI0a3E6Y(1@=4khPX^TK%5f4F1Co>;^E?#5rN<|@j!UF|4iJ(-p1^`NE}7Rz+Z{` z!M-^ph7rsCd*TMM7n#_q*gwQQVJVn|k2kF|A}lLt-SuPFBnG~@8KbO6SclwvtA#xg zt?Lx%tWOiZB3y}R^j{S&M+WalgkNB#|C4YXyN$`hhwzM0n4Vb?1J=Wuk!>Kw8W9b4 zj4fu9i0*$h^x4Nkt9=O~jh@9W#{B#VyN^AbbI(2Z+;i`{i>!m5EP>5B0)6pmVuCzfjF%(D6s%vEB%5$f z?JUtDXNZ}&Lt#E_syVPPrpxJKp*&kGmghjrE|v2{H!Or@=-GQ@r&uW$i?wo@SPyIB zLY#8HTx^nS#4WIYFO%!U&FK5@M8EO}dAqnr-XXTi--!FM+Tam+k9b&a6%We0#WOey z_i5}rcuGDX-j{EPzsXm`i}DGaxcs_!2fM=Fh261Vz9T-B?~A{SFXch;wft0kB|i~2 z$RCT>2QaWPAd6PL^N;wAZ{7%PX1U&|Xro=k&1;u61+ zHwqVQFSkqEp>42>?v=k2f0a**-STDesN5!=laGj> z%1g!XcYhx-EO$iJXo!_C&OSSBY2Tv*I=Rym&+YS-gcC5Z;zAiCyw9;!U|z9L6Z)A9A1gr`(TI ziVvXwtyB?JfnIp5YFA@$)7~tF87(zK%~aFXQng%-RBO~a+**Hux)5hsuT?)%>(vHz ziMmnUjQjMiP?xD&)wSxU0zw+S?B0Z*kWx#r@?@2 zC~cboC&YzG6h7Qcs^i*$H`7c<_=gO<3tT^Jz?tB5pN7HlR7Gta`Em{PiDrB;Lz@%5 z(3OkCV(7~dY_c$H%tk=df$75U6tNoq3&c{fQgq<-j!xhvBTf(6OBe7IYc#^FLiirw zSK@sjr?v{IuY`3_iGNxjFGjfeu=y4s4 zhjV>>xcMo7`S&rVVLRO-utAfrQ zgb~G2SuJbu4Y!WR?hV|Fp9Y;c9GZ6o`lgrBN4)}V_?;XrUX@LYlw6CGEU5RqsjBBQ$- z>#{@6mkaRixC_0n&V%)^4EMow%N~4HK411?XW%Ni8n=f04E@_5VOd;= z{_S4W{6Ep-*UI(iqc6gk;9_|RG+CY8BsW8+UMjwqm&wb~hhHIojJp7?f=;|z{zP5_ zi~CynGwA6CXte9mGrcW;jvn?#=(S(SU*fFAo8_-yyWGmw%k3Cr+#!D}x1cv_M1OTB z?3ufu&rierz(3G~{z?7;=P}iF`mlC?CS!%SYs+(E3Yp zU&{_y5KqV_s`uCmkMd;dg^ay{Ef0a_c zEMGxiF$TTSYw~sU)xGjfwyJi?chGOFlz+oq*ZHup-jnakz48OOPkso!JQicZZ{ZY%I|Rx&Lnvp_rXMQ$Bb0CmOv$7q(50DiYZui;lSBT$;zoxR4VobrmGCGR=LpL zFT~oVEU{i?V^_NydrQ13S8Pyu%BTEd9ou(7?7<4DLREx4#wD=rN@3j%6u(h}#6=hf z4N>LdKG>z7L95O{4^x3YCJfs#qN;H=K#i&uAFDcX3|4IeEZ^90|L3 z6fE8*)eL)ZBX)ka;&!7p++Nf!E=KP(7X8zBSg~XwPl81{1y=1e^i$}q#93lC_g!bH z+2U+5M;umXi^tR)Sk~vNxoV#35a+1*;uBb{ov>9Gszu@{@wB*A+$MenyEF$URbU

i7Q^N}4?W;A^nu-~N3DS6+Y8Hg759Z}&=X#Wev$gZi_}K+jF+Hi+@vzVc`4I@r=Tz#hL*{X*DqPV!CaX4u3T zY}x+-w!&|rtM9;B;s1j%ryc9DPFKHDw_wKVR#^7;tJ`2*-;R?7Q_#UJ#F}$8g{84)r+3#ZTfksHbs{ z)U)b2j2~XWolZMp^S#8D;LETDUxgj`I>r)j!Ww*A?NaaHj-tQe=I?jKHJFKM7C*tb zX197zy)Qn*jN!YO75SUkBle1Y>I1b8r^W79AE^WCV|7q{qCQoBSBGG49#;Q^#rZkx z%rDhf>TBGL`!97w{aYPX->UD__x92Ad%G^^a;)xNQq?-TRsS~C8NXrUH)8zO&~Lb^ zxkbN^=6A$vHr|bRqZ^F((HuS!Hqsew{&ITZNI2pcy}V;VZ%?;lbkCxm?yjZI(Y;H$ z7j-OHy{gMGx>f%&Y2k*H<^@Z77pz{sa9P({XLD!Is*VKK30<0BiPn+$F=8>MPCsBbp7(X6=<3D-DV z`;k-C+GH@G$z(vCqtz&=rq4*Ss>ac(3$E#IYPPpCB`NK(!s-%6BF^>`a*2c!#?0^N z)lnm(Q^xd6jN;Hv8$FW z>+DJ)0SMQO4p)b(QpWYqFkDs5~_6b~h_6c!}t!Bm|R8ztP6h+f*MKLIu9unqE2dJTv9cfu~jv-I+Yr(i>lTdgMdbsP!Q5&_&3;EyBFCf#j&ELN_({!)!JeZ z&|*|?i;-lDPBNl}C>)6xby{tTgCQ=>t%;Ks_w*VP*kTl;)lAI@)nYKbHJmnS@oI>A z@9O2tI#x3?!1h+hqb*x+**D`91lxwtbMA<8C2IArKmDWOhb2c z#feO~F=aul_|64ZLF}ED093VB8H9!ng2GxMRW;YyySSY8uK3VKs3xP>%?A0+2AR!9 zYR$UXk#McE%gWc_kWtbmv!wNouJ~%NGLo#Sb#(Pp#ae?zY8&l~n4*+LvC>h};i^cr zbI}R8MXC}O$5e7t%HsZUppt9tORV|~S83$~{u@+A+LD)4EbCdYZ2HvkntoG*HFF4r zY*I_kb1X56u*6W#ODtIpN2=?!lCINA`n>)mK>gOIE?v~y)z!VMqq}p-0{b$qMEkNh zzE|tQS5-GBEJF!3CDqI+$d*BCwW(XH?cKUU+z8e+o);oUcJ1RX=Z?ECcvPXKuxoTh-MB)G;=sjGlQCD z4#+eUq-i#~|8Nt_L^#}}6+l>bW07#ZL36!9bG<=xy+L!mL34d$GPuk6EKX&vYItSJ zLQrPFsZ5gbqMD~_62eAOuymP?+9D2z^%w<17A6h9O`a89y-RvJX|RFbzLV8+^XTdR zQ==;ubgb;6B#~5$bK#Ojs}Tnlo6bcy5p=g<`^v5bJ>8v*uBxg|?pnDDN_|zAi7`st zRGYS-XIW48a+GsbNAHCRoju))5?WXH_UHsP=U}h5BrIFf+hHJ$GBvS2g=d|8Mb}Ei zH@po=Y2ucL9>plCHVPt!%D``JQrGeot1d)LO=d(B6$$e%SX0YAO?9}AhXYh-4Tt0K zjd5_b0Y~D(6KsYj*bGlFi!gq1wTp!`0eY#&1(96u|thWj$+Rfg>!l zVeF#OnMAbqsE!!KN0{62XVWZPt$QT=vW$l#W(VKQm52AHR9&zJ7AfPUo?p@}Ht1K2 z7x6)x;$I^1FU|2Ub@4Bw`6b=rPP|_Yzhw2}Q@np$+)Gte{EMa}GoEL0-VvR!E+Fn| zrv_xa=+YW5@qCE1#s#a^nK9cVTIRzM)7~>wOhlWIz-#tLw5dv+LnDVs=8# z=n#F$!6R@wm%z6Vf!QyGwJlgNf(0l~t7icpP@iB93-djg7gLBuDF~$S7*@r-t@dMX z59Puv)JOOgm>DZYnN&;Lgy|x@tYcL-zG>L;RRiBfF`G6HGncldUA^66^74+}rDEpt zrOTIMu5CJECg72nUrglRdB}^-Qn>=(`s|=i;wv_-wI^r}9u1v~5EP64aQ!yFVQ~o3 zLMq@nm5i`5B`I~UNOZmDnGJZJ^DUuVcIQFBfs_E;O({4X(D|M7J0a8WO4$ndG3R4& zUvb_oWadVuts&*Bl&?~2GPb0SOxc%Fn)Y7C(XAp=@%8Lm!?*vU7hMm^`r(;uSpxj z&o9$%gHJ_j5uSkr36=F2<&6FDYq4@E$fdL#tfRI5qOV2&9{n=VD_4|u{4?cWr z;GOu*AvpRMJco6_=#$X{C&wed=r_^t`X$CM*w1MFMfXRK#Nx&vD}V;5X7LyO9&2!| zzv!FM!?D;zr}@V;o9~+5BhgQyZ&M07O!SrLmgpffeC!uF9Ekqg2ty@je#GdCL1vtI zOnk}6zklpFT2JC~EJu^Z=qIrh3|~zP%FI3n7vu9s+;^h;z?HYnyrKtDua6-ur%uC2 z3F}~@|GR4tz zRDR2;^a5rOt|6m>&1es5B& zV?>VvcMP|+ejBAPrEz~K^`Yo<@MSr|?;~gp`{M!_AX5C4)1-WkQ2N*bP5RmN$7vY+ zmxMBZ8GQw(SQ;q;n}Kl5N5uVJBxni&LuZ(wI9Ak1+&kT={2XRsi|8d^jdSmag zaHoQ}c2Dx?r(@B6K}%e!Ck6k%BMdEmQUtOb{y+R_I*nc^5|_YmoURz*e{o2FHq-$1 z_Mii@@I6`%^OamJlkw#=Gz-_a_U)h8$#17>LnnvQm=jYxDK6PCC;Xw+9y2Lq`eUuF zUq~Is45__XrnnYSpJ?k=-~9v*NzuN3u>U=% z`DMn%Z#>0Xx-_Tb>c=SW=VmN3C&V>!MULOhR{eh(2c!kZ1t$k* z2ImCl2NwssgR6q;u)g~8;MKwFkeZ|47=gwTULH&MdAtzh5k)zVD8l#-FM<>C^y0Y^ z&vZPn^n>&8EX1=65BU-O8}VF<=UOoU>$&Fz7h+v^FQARVOHtNqgExW~cLeVWZo^*p z$AV7n!2O!Sz$M?Wqi`gX`yB) zEj0mJUP;($c>4wrdM8Oe3PEoIzaP+GV4 z)mZtnm`0UYAB0gQ*7smkDW9hCq})T}N%<~~C*@umPvTSyj3?!O8c$*kjg6>KO?W1N z4to0tRPUE>ffw|4So(w$R35>O>LbzTka(P*s6C(1g& zBf{>J+oRtgTp!T`-!0%N_@{Z#M~ZtvuMf9borXO;l;7#dZ*=r) z`1Zl~UC`Kv^&~-*u^rE5JiiC;zQ*b@s_&!G{ouiq;K6P>Q#3FfPZORoc+h$RXf*+} zngCi&0Ieo~Rue$0382*k&}sr`H35_%a4nu2@nH3T;0`=@;n{}g0X&c4c}j?UXw$$8 zc>ap#4LrN??8S2c&mlapaRSh7fnx{@&61ylCmp3327aA})uF?q?}Ph?vHSBYlyDzP z_p+LYn^QaBzYuc>INKY~t=Q{yx)`IHL^mX1BP8KcNWyGLLbp0c%ty!z`h^T2vHvuJ#)4A# zPzpav;YTU_>3C3jzYk9cPbr>qJP7M=z%vq03!ZU!CgYijXAYkEcoyU7#Dcpk#D1J5&fb|UUK zcqmP1d7m9mDxO?eOTq^#eN-Y)=>wHMQ0W7eK2Yfcl|E4E1C>5d=>wHMQ0ZHT2UPe# zg%4EtK!p!f_&|jZRQN!J?_Ml3zFDZeR6JokWAJ1Nn;)7h53@XZK|CcQ!S4cv2?!60 zeW2I}ihVorprpRH@a(~}56?k7hw*%c=O~^&BmKchhtjXZGaTuA@QazKyou0j4Ul-+ ze{mPqUp*;4;B}s~!=eLg61QVjA@;T!>-4ZnPp{FF-_vS1S^;ORejBZtu!`+U5si5^q^lpE~K8r-`wD=3$m+>zVJ1<^^`wspkVujV;;J%CXSBY4C z^#NuF_u*d#c9(pNJP%@@fkPa|KL^%^eSwv1|H7(Rh4rwR&?Z?DvW5L7G*eh;%oH|? zY;XqhkNqd35#~(vDsHSbodbBToC~)D`z4aFCuIp**HY~8bz%kSTBLA`yal^BZj-kG z{BK_tyAHjSdN$Zet8U=BDHhNJ%_CpZg=g>-F ztT`o)ug0#QHmo5fzOP}vuVubp$b4VNd|%Ie-#~kUuyX>tg2Wxn>n+Ud-+|X}W3S2{ zaP}$Y>`tE7eUUl)5;)uRUu2Uwo64L`X3i!uXA_yTiOktV=4>K!Hjz1-2+p=b25J9e z8q0M$b2)>#>|!ow8Zu5?&N3wZZmfvRW?ttouieaRJM-GkytXs1?aXUC^V-h5MlXtg z9+r16%X==%dlK_KiTR$#e3vZgKIVIX`JTdjw}J1C*bz+f9%Sx2nfnFI{Sb4%khx#P z+)wEz^(5)VEa@f8{Q=DVQkHbZl3vD=K9D7S5KFp)>jQV~VxAp4f+6XC`F9U=pQp$( zPzX0PrvX8C8W1#T4Cnre!$G$jIOt0Qg4Q%3=uQKI_B0^q(iqN21$x!MLAx3dbgThE z(;5);t%itFBQA8W;oE0I9_Vi)oYR7=vUxlfE?_|wfQAYDd1@?Z6reW3g>J-i3qiRk zr2(PO(GdB1CYZhkgq}$IdZt>?dFUnXLf8u^Ed6;d@K`v{nWit{Jm;R^OSn!G=UG9J zPNP5Mc@8HN3a=g54Hk5T1#Pk*;5^rGxSK894Hg8P=Oy;N-NM~zL0c_oy9GUBK~Go^ zXfaZG&BE=np!Y0jzXg42L7!RBHx~4r35B+Srw`zHEc6uI7l;$yR13Q4Jo`F&^+%#xXVnOx7UK! zTF^!dy3~TMw4iG(=tc{oJno=8ymwhRlw9ZQMfnZrF$T)5^c zko%AYt+JqX7Ie7+tAVti#mZ&}bD3)*Kv2TdrJTWEKlGqjgWdDwzt=^Zs=eX0V&4rG zw8?_5u%K&9h+^d3z%g#_hr69|clN_=HF0?wdJXe7s}TRayco1S9+&rs>6`b21wCg$ zFImtof_&3`y}q@+jez%C(5DvknFW1gLEo8>PgoG8Vy9GmsTMBFf`WjCvu~~i5w3)B zgDqUxg6b@2At0vD*JMFsENG$yP4~?&RNW|~m&UdZ@EO!FKQNAm?b zM!{AS;<a?KqENF!VU0^{Z*>~aX=0HCvu!(V3#QO$rVBAZf;H7*!;Lk9_CK$Ti zg6_1StroQ1f*!G;CoJeW3qs09dVm72amu?a+3POxCIfeiE#vtu^=6ea1#lKXC9}q zET3@GPYAab@4f6#&_;r&mE}LhxJxbQNG;cg)uLwB$*YE;K%+&0Vi0rq{& z^v!<(&|lfthp!~`wdA_|H!Ntk1?{zb4;M@`QT1j%p!z8}oS!pt~8m7tlit9R#$Gp&b_Vj0KVJPWFAJ zAMP#2L7N$IDcoVg6%^qKu&3))~on@lJ)*n%jP zD>#)V3pd7u3T`NPEp&){3$C#s;0kVL-0c?bP7B&_ zDtO6)_9O34nWvwjq^S&jV?oei8W)0x0og4G+ANeyF+$K)q1g-tEvUqTpwkRr=rseP z7{fV6lZ6{&K@%-#x&^HT#9Ase&w>_O&@v06aIC{9b^62E-8aI$g<J(> z`*?IE;GVwE;cn)TPerEz-Vwc+!+a0-LJ)-%*nOVeSJ_>_?&It(XZI#{pJI0scSS(U;Y?u#8lSKVbzdR^}?o_i-D(zhJlYyKw)iehqhl`Yqgb>S=cW1ouvyWh})5 zxEn%>dvI1D%7T*wrTCTl3htZi`zm&yOK}sYiJjUU=3BTQ+UWj)$89}u7uZ(7U1wXs z?i9H9+vdaFW+U$1XX}FdysZfCEw_f1fM3Aw9jHwyI&l6AO2=WgDXOn~82>y@L_+Bp zep3<0Ud8!H5+^xOE`MYAdB%Upm<2fNPKw*GH(uiOJaX?>H^JSeB5?1+Nr%L-d2nyx zkZ-a3CgZQ<6m~M^X?7oH_b9%VOWabP5BGI~UI5W`o){f!N^Aotq#GX|wbT<{PsYDBh>9LkVw4crkp`Hb-3E!?xoB}xE8 zZsDRXWD7}@Y9o4Zk{5i{jZD?cgqQ6czMbKV%DiNyuP4m1P*0WfF4r(F*OCNU>EKE5z#YWlhjHmDnQ~gCAaN@wark1U=VH$NC)7@Xp%Pup>2Pi0 z+Xt2X7B0nNYQJ(ZhhNP2#Z1X!@)a(uWS|yJHM^aqeKX}EHd8KeQAW5Z5nO67+Y@pb zMvjnwlCjN{nwU%F5*w&o2uW>WGtq++8z{^nPJc7Eoy|mp_y?EhK`!Y5=JSJ0;c(VG zcQGcwwUf_f3^3&Z&g(*og%d#F$^g|DPF17&x|PG+%J^HkjNQzuZcc}F8t%LzUUjqY z1P+N_pD~n9H}iZW^LZo1RU>T=0>6=I`_zW-!+@WMyE!3^VXEtmT-O^D(g|i6xQFTA z$hpks8ehXTzS{N^4o~vDk?ZRouJ_qo@3-n$Ov611Lpikuc1z&?NIgX`QHwnSBo7<8 z)PLtv{GCgY!Z4lXN#Thxi9V6GN$Y7q_@;4l&9d^LxOXkSi0!s$nt{_8oHPUd+h$L-{l zJDI{x=4~g(9mSkHhdKERj?x=!YHC)ZIY#ZucjZYODV z*~#IrXPzIWUIu605uYz+K2KsegLB-+wC!W6c5sdjoK6Ghbq0qCa+n~82{LC2i9+!m zhxamtL8d3j`Sx9ddz7H5SkK6|jxeYr7q3_L5y83#fEBZ)h^yE2u=G zhf;%!x)7@uRPGRy1hK;?lX;()&GW8?laRM66}ZI?jC^OXE6q0Qe7R zCzDvmxvZ0JZUgC6K;(8p%wT6?K;5(XqtC(*kT-U{vGGvcrpoGJhu*6mo{j!9^ zmvCIz^uUbADbi9#a4#mg6ehwoUe9$M!P(is*s!ia)+uV6#Y885z>P!P!5^k=vKkwvI4l+kl2`>`S#d-u5GcZ7bO&iy}gFsv_dKs?w!a zphs0P{SncJFdDBhKCC3e>P7agE~mJ>`ZMSRUQewb7HC&&FK$NeYqA7Mx`U*V9iaJ~PPL;l(} zgzy~x6%O-jth|$Q5?hR~usm$%a)r2E(COQxVMw23a_b8*pIe!>axP;zV-mOx4&XLe z&b1lhSRrnYAu5FoalJI!HUQp0_N@#NCCbk2F`ZMxum$+_I2TOH8(5Yu!7e&r)D3W_ z;dF4o6b9#l60fkc0GLLmXMm9->9-Ka4N(}C7afU^TA$}Jg&}U&A*PeZFVf9*bd33Y zjCuGO=W>j5`Hagr2y1(!9E^2w5=vw?+>NZ6uVESr7%pJg&#<3U*o3u7u*w*775h$N zco@Uo40khJ&Tu)$!l|Q(^<#My?mouv<1j&n3pvaVE*f(s-;%p_0LJ$9hrvMV;_(h4EW%;GriN zf3ETCGk$Z8Us&7hkJG<-m9mfHDcOKiWhrGPr{c`OWg>a>?ADOnfUV;pc}??FxVMiV zJv$_~pD}4dNIo}lLQ6>QnmXx>kokr;&j3N`3N(N}`;w?uuM3AsUmx z7@Ro@yp!E@c3Gd(3U1)pqQkl)f^*>7u(pRzed87Nc(Y^u7p;&E^mfp34!B{~bS~WOBUK63fx#CmWF=1BgOv>%0J=am;P*@e9ge*F(wt5Ckza9C{sZjQu2x<0MSdNtoo5FwT=ODJNl4Pr{@D!zGtfQHrfxPTKKHC5n|w zW9m4B<^`1w5KaSDI5+YFj)z*o*~zI`fflB-(C9uGF~=fN&JpHHC<$%WWA z{!fV=;#jLm>k|d9kdvH-LOzdT#|EY0;xs(+5YzHcxNQ_m&Q<>lsl%m`crwnt+!AY#=lhl(L zF6~N};3e^xT!FQs1Sc^}WyL2bq1NE=5&}^^{*gA^%z(ei<829~iiGGot!~CsWoIti zkdFxfJo&(_T-C8^$%4u$tCuWTx)Ns<4V~7td`VN!vd)TCs}^!XE~Eyf-rmyMHlloZ zu@HObA>O|B$&;qdzkd)mm3}Y^x0>v2pEjdu%QZXxCZsn64Hup?tunGXcNSFN4w@2} zHx2ss!t2lbSjc=Dzz$xpykkW}^)z&{J2qg~u5(exN;0QiNdHXyCM{Za;lk_7_cnrS zP^Wk8U)h4(H)%Ny+&xCmHTJUTKovQTC>a-O?{CNVFGbOL+ z?OL(ee)VPWzXIVMPy_=JE`;tq1)T%Ti0m3%NavAEumAa7R*iG5=e<*kl{z7Cb3ZYnN5er0j7T5mu8dmR0%W+ooLED<-; z0!9V8qQ`I?=+nXnB@cbhSHzap)MwY#c_SWgHr*PFOJmDyw)#S^9RHy`H^bwRwmyg7 zmEp60pOBxGk!Szt*F&X0Qw=+Qz?GGk?MPM&GqG0Mb^NzZr^lb2q4N4x&zlFO&3W0@ zP!~Yugei|LITrp(3;!OrB3xzS6KV~7nPR*VZzucXC{vpmuM9@c-#Fd=@f%Kx{|*ct z9Uo;+Sb38FJ{(oQ&dPt(N&b7qLh*#@Z`)w_J1{x@W07j%FShV|a9rsf27Zlgr<3G+AwpzE>#+&yX)LV977V(mx@M%RcQ|bIT5NGWy&<-e#1@x{^EJx)>T|mNSVD%C zZ)kHtxFJ^#IqV~8&datP*TO7uTTA-D25(wMvcv7j@C3tsOA8>^FILx-R2O?3p%gXL zMk4OXNpY$LTEwqSayYXRQU?1xIk>;tot+VKO9(sZG0nF#MFvc}IxMdj5|=OIGH%?= z?BUCp%ZwYEiA?W*X?e~0k%EFqzINUI{A_n1kpCZBk}m#ZC_UdhU$re30yg%MUB zh4t<-l$i8LF+dYb7n6GnpE#D4-FI7t$Ca;Ah{ zc=Ek{FCGcW3C`n3b8_98ndq3#OPkNGEgGGlRh*KU?9LlLGBoCr&c@n~O=pD9 zRyU+nWM?EKXJid%J-6}nj^;v_D>*U4JtQSPCpA3h@|kCBo?Dv>jxiO?F`ASitu&BZ zuEvv=)A)DApIFNci7q3mE&Q9}V>ZMYZ>3X)sTbf+N{_UhPHzv6cqR$c`J8Rxck%S0 z;u^aHHI_*|q2zi(FNIEt>nS@zbyT4sT$yEZ?bsFLOIt3T{|(j3q4qNed8@|PKAvn> zJL)?wZtwW@xw;;XZ)scET77o=pycCUI!Rg}6^Z4Tj3|Xv2yAKjLj9F#e%FSLlJJLV zu0R85BY(Sk?)5qyOWncxUt`~0PhqD-8{)z`psy}xzWhKM(G~qdAHZicrYL%T9Nw-D ztu*LNxCXq?bYOb~rY%hTPt_iie&99z+iiN_NK)@Yyb{vnC;nId-15&o$$u}7b-xk*SHwz-tLZJsYU2A?h>d~vkWYa8qZ}g^R`jntMTvPi18TbsqfWz(8_(U z<~;Sigtx2xOJliG->b17oXhn=bSL-?Y^eg4xQe>nX6Bp_Ji(Xp-v<3Q1^qTj->A4= z+tjoFNgAQ@PVB*NLhoHY%$LyjD7Ek(kkWw1U+(i{WF#fI9WHmU5?11xw)POrXkkoh7lW+6^mt)GYfp}!8c`sWE z&~@A+utDRY9>G!TB?VYhS50|pV@KXN;nD>S4GS)vP&H>jTB<+Ckzo&vDjBn?xp~!? z;)(tw*JdK(9(TUWIdr+Re!->*6E-cV_qZM26n8;_Ej^?4g0W-Pv}UIJpGXgQJb`@A zUYExaC_?>`ZoZtQmOh(1V2MA#ZCB%|?P~nH*p5chqVd$qG=2{@sA(O4I<6FW!NMPu znEbNBlcekLyQDV`PqL`-A7ElK)#$}(!Zi)H6^)O;giIc~P@h0Gf*LS-2wCVT{GXfx zS>X2O2j#oDf&5%|u%K@uuDIw!Kb`M6emAj&lvh05w`(43IM|suTwmuih%P|P)?7C# zeL%mQPR!R`I6o^dFH4=5otv9|{7_Co$n6b)1TRRyOWzrCXJ5TMl$VtS|E%0M;3{TnB#_w`pwv8u5AB!tcS<@Ab&MlO94Q@#5IAC#>;(#AzEQc5@dPp>BWJ(q;)qwq(P)=?j=*=mZe%gY3 zXQ3-C*O?s&6%Q*eaOVa_ERjPfA9*fwetBIuBh4MiNb&g#8;j)qgzP(G#KI9qYm`OY zr%Xc`>}vNg=(XslC{7tmC1D|z(Z@(C=~ye7&Hb_t{qD<_6YZ0t zE3nLm>P=%05Ehp%3-K81YK?v81Jcwqf3e3#nwqcwG(fOxm@K}eacpFpTXcb(pM89Eyb=L5Q=+F+uHefe zrt%;@{t#!V4}n1nS_J72G$*q!q*hZV-(oZCUJtU%DR9pRv!RI%^|UjSBns^@ArUE* z!h(X7;CKei7ZILPX1)*}t)$BA{q49i#0ZVgUEk`^uv0;Yfw!xtwIu?oV_1heD1H-5 zJC>&!fls27nU)p(!XKbTfQ%oGt4p3VXe`5oGiZcFZ&l5P|M_w8VBbP`=pYt>Lj!hT zZmz`8weOMPD@IB&Dd0T%z<9=|LKru$OYBo?Ik zQ`ac9)0yvyvD2)L1L|?g8NG-%8?s<%FYZ5RTwZ3!{8xRfN4YH3=jaCtwO8ob%;l6kD-It7u^wN^$xPiz2PB=Bojk~P1 zmPY=lg;mU-1F{OgT1u`0CwRjtr3Ll@72$otu-F?{RwVeO+E(jyo^tPRX;& zpJn*e4u8(^J3aYH4&_ceemvXn&sJx91Ifw9kKhYyj^C6e>J3UU=cZ-peXG8&Vp(E5 zMuYhHZOGVFtXnnyU0l_p?ezv+5k%6Y@lWI9p-%twe)xmpUvcS^uGHao$wVXkihkh_ z5FT3X@6po;PZtieR;#NYsf-q@xe%*0=2E&X*X?tAy#5q}9oYeQj?Z9|&o^*L$@771 zpVw{=49ZoP#W5wGWsinWFU_`VX|~2|4xOUO@evQ%$HZ0=0dg?jry|OwcW>)=j$w*Vn&vSzTcd#Tw+WQuz=DXc6G1J^h z@|KJsM7?6t&fJ`wfXo`;A0Lp%`U+XhVAiCh9Pdk#hgpH&)q475G%dvz8m_8@7Agio zH1lCuok8WOp$Q1f+^kctM$Xh~-^B$| zrd9bW>vtV~53EgXRZ{vd8sTZo20Zo4@~=FvIs}$bEFQ*u61o@u{o{XsQv7#twW&o9 zCd)OQWmv4T6W38W)WcYuK+?hVCb%tsGylDig6mCxwlFl^ABwFk9lD$}>eKkQ(UwSR zbb2-`J;o!y53ms6Maw_I^512Yhx}h+f0UQ4Y@IHb*Nn$DhsM7nu!z*clhvW|`^0BF zKGgJ;TlkN#)bKegJk8@%_%g+K=C@g|#BVcR%>UGM#QaZ9Pt3nRT{H&opZ{KQeVm-! zXwm&Q9Ep)-(M?vAF6Vw&$`-%bj?#GIqa|-V=GFNy-Y6%HdUZX~sJFjAm17(1jv*NL zQX324Naj&SONTY)HHNjA*wcL*TK0cwfb!O%&B4kBU!Y{D$D8g*O-RD@9}G0<`Hu~g z?6g0LnkcoZA2#V4^iHPwHxe>Aw@) zbl%3;BVE{7^|)rZ`9?xmjAGm{Hg7|Zu!%^j;?DnKFP%Uq))-NE!>Y{ zE<9$}vHkr6Q{ly!FxS((7&+N`$!s~kzvHsz0TX)0svui2yOW*A2sq}nPf1JpZSD9f z@1V2VI?0GR_tzcm7kAVXnE6S15cZF2UsAft%qVL;ySlZv88neS7zRylQ3M@te088% z_}CCmPoF~D^YrP`+Szq=vumGDB10zV{=_%L?;qlO)`=;P#`z;gEI7?|(Ck$D?kQ?0 zEUYWc==%}_NuAeF%%-JLe^MXI00Lvj?>N_V?7b>g`c5?psh(fYC+U7#5D#2XI7_=%9N!@)>mqk`vO?|zW zXY^^Rq^{2PUmBPCFe|@oL~~%ogqo_N>|B^of&6FBet5`$#@dDf1%uNw^Ai4}@D9NA zXK64qH<+867RnB#jBz#jJ^7v-XR2FnHpSQcjTd7YS@)^SP(u~;eJE~5iFxEG3o0-E z^CyAQGMW_Pd2aA8yBgL{4cFRFa!WAL*IMA8@(~b6kN^ z?_24)DW7|X_#4X(rRcFl?K#5-j%`TKP8n2RTrkMza1L~(Iua9|1DfjcO1vpKpL_Cg zFKY4d((q_sACBA8lyZB+@Q&M!L)(K?0=5S={#|jiVGo{8B}04B_=8wH#ugEWr#@ff zKcx1=>96P){s1-z#KM!#*5UW#8>_~j-!J?jd}AfcstmS90isw5l(Ii>KoST3U zmQ~n{k8oZdd=-JQrmf=*+Stw-IA%q4N=m@(N-v$(FtB;xpflOB z^k?M_$uBDRr=$ie+iHf-j!-F>TIMQxBI2sp&0J+Xe}yA_na0C^mpCjQu;P)0ME;mF zUrPOzjy2T4mJ*iZ{7JmQzRQR+hOYxN>X$MnH4QX>q-ofLg)C%Mm0=_pAO4_Wwdn9< zx$E$|@I6SEWktX62jcNOE27K7c!Mr&3m?J4`(321@J*LyI=RZM`J@WPqE8)Xg>sal z^R#L8lC3|5aW&HCF-&WC%o^NNGNH*MQ}aCdleJ+l^a zNw&gn$Yq#Hi6T)M3%!LwC`W3_EQ6SuLFw5l&|f3vk4|4Z@3e$cuWCv@{!!B4Q4@z$ z&1f1>I(k-BV`q8B09U3ht8vrx_KW7$)z01EteC!LtT!0!EFafUP~6fna&*V2P~l)) z?VLWuRbSb#VAJ^Vn-*Y#S<$Ex*U#?2!XA$hJpDie4_m*)Z^%gE}Oiuy?x_kxt_bF%W5YL9XbiOKV$M9^h}8& z2X4oI=5p=Eyf%-P8Bf|yt4gj`gIYqXY6MgCF0mp60w`myq4nLseOs69+Kht&mm|T^Sg}pNweZNRj(#7+x zZkV5Uu+;A}`0!JfR*k1ttMPl}0B*G!Ppwnq-*Z6m2HLpz>jJ0?(c%C|s zXJvS>Wh<+%8xX;oKepPj+z$okuTb^bV1_Eq@u$4e_+)0^q})vTY)VmPuB&f&N+>nk zm-<%y(=PP{3X{?I$)sqmbRWMXEi>2WR_CO>Qu1~%FDlX)oO5daZ2BkWZBK*y3;%bsqz;|efSug z@5VH($K^WyxgCqh0tML+!c05$u*j9uXMTNQ@ax~qFO8@EL*vbrhZ;|u(fD14)jbIP zCFr#92MAB2R#+uv%7^h`M3=y*@>(yCnYbOpe15Gfrbs6kx`#O?ZU9$Hq?0p-K8`v7#R4bqRRv)7h$XQ|r`%($a zII$AM;+EOEVsTee+d@5m&PI)2SS_@DCHUR``#s)E}MSXJPNO zun(Q(WVx+W)sWi^F`i~yc$uChmaHA=tLLeu*yx3=@1SwL@Sr%keO1r6)tw~Vn6^l9 z4Xnd7Tv`Gan9wxb)OaE8d%vc7K%~gy7_#mZ0*}T}o9s?ZEz5Iz>~<^}&2oj@W;t!i z7FQlHEIch~j~VzfY{UIIw~O846LE$a4_iLWI(cv!j{OirqepB@!5_a^Km%v+fnkd!ydj1gM=G#PYyX4y!aH?Hi&PC2P<`KVFL zT1s1L3KRS^p+)M@WeZiE?hUsO9M~SNY#TJFP5Ii^PoBJ?J+G`hJ8hObKj2Z9^C;qz ziQ!qT_v7-DZYHu*>WZm)ez}Qs0dR~hJnTgevC@brxSLWoT zXS-5UhGf>BF=WX2NU)SX)O)-bgHBv|dea5U=N;tF%y*_c>`73$P`t^G? zHA|x*7J6_a@2%o_#4`J-SS^|Bdrr~9@j>+e&`%8>TNgw>MU`asQwIXx)W}Xl7nC_b!&cUSIA4GIajwUUNv1Nc6-+$^6R4K3*cRYb1xK-XIo!2fn@77{59WZ&%ND8hv6y7(CRumdTIRa?X7_ zZOGCs0X+868a$dCVqFDohR?Reg=POiY01;3t!r-Yg?ecnC}l9GZ+)&@o2>?q)S75w z<$!P~H6tU7G?6o2Z_LQbJC$lNXaz0QF0h7#+r?kZb^$!B493H{m}y!jer_!k8m)y_ zbkf@BSXgQmzhmF{IMgO|ocmUo7D5?W2#hUN_bt&D!g$L<@RNnWzGNXh$~I0cuAeLf zGwvd`5Jc3n5d1b+2&Qkhwh#&}3&Bqog6X?VTL|MU3&Bqo0{hz4BMqj7;8(B^4D27K z$2do=T61nIXw>*ls$C1uv>USMSFjw6RPMPbmI~|!BbA56k1e~wuV6PAp&y-T+6{gM zyTQOdbcRKr?Ux3&Q~|4N(z5ydHrV(a$Mn5NTqJNj0j!^TqE<{WWsyvb{{3S=xs2;3 zb!U32asM5`>QJDf&Obb-+MU7-l4_C zRYkc;mGBo|L=)_tV2AR{R!*)w=3ls!)gq=!5)uEKsW`1D|VAD8tAmlf^K z`ffyhBAE(a?C79<3H$}3l)5CtqJSYKQ>YLtM}^O787NhLj=VIt?~3e%K_d%;4TF6Z z6T`(-R=Y#5xOU>Kjm@R4=c2=&k(mo}vxc-r8mA0(xt+;AmnYLd5W1g6q(8%Dx4BrO zVp0R6g!@8sgjz$p)wFd&N}^hxmOap)>v9%ldIMM};R&3*G{@)5>CR|3U(wXmGrelu;KAdn*j2v3 zkcvz>*^TAGeeE-rM({Z&HRSR*lWeZEaieBC+(Sydx__FZ`=iQfuplaReG=`YjqpW@ z#yTstHX+p8>m1;1gRN8^E7BG^g{F3_LvIzktUjWK z+nUbvq4Tgsoccr=g>J{XFP0&zu8dC5dgs=W=@@B~lUgSfP;F&b4#sS~X-{Q23af@I z?`fUG`~J-n3(3X9ii(C6>o=vp6MpXWRFrzb0lLVO@&%QsJ(Tsi=nHZXv`Sm_%s9MV zJu`zs|ARu?ZnZ-1i9Ri!;?SLOp=ng}2=`-D!n>IVm;eBOwtX6EIv14^F;rKL1BGs^zpjVL9qfgCO#=Te>p-avx<2*%h) zR62c+@oJ- zP%#H3ci=jgn{WuFNr&tIA3Ee+x^jMq4zyO3TWsfYpRiJ4yIxc0@yD$|Ai7aO0o$)Z z_gt>A-8B*P5`bD)18-^?#j-vyAqRA zBcalvaTymDj2x7aljE>E-07(iBrH0sH-lkD8vR|3cg?>ibOWw6Mfo z6+Pdeeihr;6xxB~VP4~vyJOInAF;xt6?@1|p9o30*9uLo?`w{=Ri<<6dq28b7RJST z_74yuT<<2N7qPNL$XvH;O{+LyB!MC5Ax-JZ!%EE85}o#(oQ%|0?arT@T*qSFG+*jm zWecXJ<=E%i6jmXaf`tsIN3l&&UfnBWkcHNP=4&pFMd~N^EGNF})=oXm#Br0;T#mHb zaf4eY1W89{R}Ly0QA8~(yJAq;aI5plc7&+&89Or3<*O||ZNUhm4@$x=2K=YpJNWwB zuSe1ncv=cNB%1SXi*^##sI4W;$sKY&WS&cdy+pqMWB=$Ts%NxMa3;DpI-KwVbJ8cS^~F5aNPIFsWu>Rvk}+m>=lW8qLbDv;N^w7i`3F}FcrK9T z^Eg;%zC-NMj5(QiTO*@OGZUBx(c!if*4IQmt^FnaK`Yr&pIpDI<4S>%?UWejNcU4L zyL#{}va1cg=~Vzc2}jccCkw76sAf^^a+a6Z{<=<4_f*Qt$=2#$IqQe&pNhGa769uR z6$lMxPz%ts;---($CK9oj%-^ye*D_@c5>S~%g0w#Oeim(P*E|yTskJNZ^QJ~l=bcH z>!;LDsjQq*Upuv;Vk$U8a|9!C?QAJ7ywkSDd+3Qw{OblDHtOOu3rl^J4o$MAXAUTp zy%zpHvB|JHfwxUJ@E8}bhqZy3^K-sad+O4PT6m$I~|2FZKG4nzBoAF})JMdjOMh97h{o~&w zE|1k9+lKx9AEuqQR{qq6;{DN-=>Y)_#0mHZj4Mz(y`yOL030ar4^9qAw zYu-w)d_-?};-=PJ<>MW z2MSA&UMx$ph5EW5C9y_hL#2ozRP85;`P=5IUiE0tp>LF`ZBX zgc|Vf|956~CE1ufzW2TN{bkSY&YhV%=eE=CDG0)|djz433c*C&sUsX= z*l69daBK|4d&kBDrNz-2UoDsIa`if4mGE+@M)81pdE{yVfw$>vG_4e;sBA$JZ)L*3KUTj(h~o8AKCuj-ACSye%=Oms?%kT z#@R{9}ZOP9Rt1s|1>ESmWHfP)~>3AUaHOqq8IJEDy$-qLI$pNbCGu>&97@lvyuYB1vWp1$|+s zPVY{%lxO?W)~Mf{v>3R-3*j~33%hhWccQtXwm(@;eWSoi82c)@Z*Bw)rk<3`MWQcM zo&jD)Ir{E;;X2{ZrR#D}({b-YxXrc(KtEixVmV_ypWQmj+M|GaVR8CgD7 zYPzX|v|wS3+4#N0&u*0YSp{C=XX$Q&N3oX*W7H(wpK&b5zqNE-Dw!mn*wE0TWzQ&Vq?KIB<9bPEEbr2 zvioZ1^&~2Xs>6+CW|gIFVb|o$cfF8e=3QrOKsr9n3ml zKYkWG?g9Z=(-YRRVNxzcT6eSTX<1x>XcNH?3S?L4W@%-@)*YIN?vLns`Z&@ZG{KV1 zZZf$V!uFcVV2JAiE~h`>boq}1R?0ctm{Lw}SmSNXvFGGCB*iyklx~)1$AfyPo@6*K z6U&OF*yn=9bsLpqUI4zayZhC7?^h~1P1hG;ptuWZnV&>f_D0^2*IcbZjytH?vloAG0 zN|n7UrIh${rBvBrQc4MbDW%FjkW%{KSgEoPMDiZmRxBlc6?;`<&=H@g)S>M6D0PtL zkV?t+M%nM7@(r9Qxl(>7J4}!zTe-Z{M&d?MF1Jm_bgxlY_F!Bx6m;#TRZz_*XJ@po{HER73cbfx9W-j#Cim1CvK-j!S`9#kwX z?M_`M%#nAe2EdKdylJ0ci7F!vnk}@fu)=7wV(e^eH;Qyr%DU6AgX;z(bsfQFaNL?O zw=l;dRZ>kX=+wdU%%#Ls>8g*lR7Z7-8T;{+qp8@qFlvruLQ+7sa-qOzrS_7QOL6S} zSj3!4i%WfnKAS=uzS82m$68E#Q}Wyu?!!un!jnYHorvV6cy7b~Q`rR4iW(wniqYaM zYd~|U^KwhX6;O%yn8VhP?rsjBKBvu=|COF?iZP!{W7g#7ggJ*f6+5VJY=L~Uvp(bn#SvGpsSVGo_<|0+VWAmwx z@vJe&94^hez_AAI40+TBRelbFnDH=T?&n`9zFlvSA^_}sq@0hyY!haa`$TE|Qd?h@ zXV3G>`emuYyJO`qqKB07o8bTX?N~Wc9Hsm^MCti{to(_w@*9Mg5Enxfq(YwkmD?d3 z_+-!f-P)VNsY=lbYqVMzr4IURY_2it^xE^+86%QCYP!6Bi;Z_odb}>JHtw*uBy0N;He17#%I4t;!jEmO>6U1SJ1{F} zE!eN#Y?6W2YTePkrhad%%3C|Xg-kqf{fpktXlt5TN?4O1BaHH@((CA5nZuJICQL}> z8eYPLbT{EVrL=_esFX%RsiU;_O`4$9+%Z;u7ke0yJ7~_+{e*>-T1vQ(N(uKUwT$CJ zDkn^#++D&HR7$H!DOL7#?x!h+;Infj< z2d~yk^I9pT-cjzp9ucGDa&S_Bb*%g@Xbfad3L4;Q>;*~_U zTn_6Jc$L>8B~_-hGbG2{Bex~A2hB;u$GMZ@$!QJ`ERVci2YY}%18T1WCC{p<5O}~gcgmoV5IjIepo$6gQCcBzWvD}=^?d_ZA%-Ph@v1v|raw;`Bo9#;` z`wWY(ShMzuMT@RjyXK0;D?2vNp0l~5WAmKZn>)}k6lNVe7k7atP~<^iGkM-O7wI2Z zt$%pJ=EWwvE8@REFX=tO2zDR5j!A|Hg59musFa{gR96c=@_@H`;#wlV)6#YiXoONb zZQ+4DP>%s?vZv*R^=r4^@t!MQo>`=1soQ?B~a;YSklvUeD^g5f&w}%1d_j%d>z3hAFA(x%ou>w6EJ zo}L!4n*ztEbv7x{Q&rs>(rN=vlf(G1i&*lo%jnWS8c^xlmURy8($Lx z8>W}1CzZwO%Ito>mM28uz?2Nqm&T}7U)db&UE9%nh}P@X2hEm%&Fo0`*Hm`KCCL-B zd~f&ADC%Df8=U6$HN8{!r?uLMdCvJmidTIx&YRdQV+zaTESD2l-MqiG**ncp69 zRP?W&V$i!HURQFktJ2`JW_!xZr*|egS5HbIdQxjT(-Dui8>*)7RhL=0?}|}Llmd=` z#+2(Ftx|a^L&=^w*{-!+ixXX$OjkPHok?~lX+{@*0L}Oltqw&q=-g@IKtZUvPl~92 zFy{XvPm1O4h&un3#;*!UMiZ}g%hH%+GVZv+Y6mjLiq5kyG1@WM?qPniaVb{Djm=te zXC9x5nUW8lj2N9tuvHo^;7E43m}-f1uI}zx-I;C+Ykk>p%hLAtB`wk1XW_a?xGvO{ zt?k;sX$UqUunZYwdpoR+y~&ov&CQEjs{3lJZeyk+1IO~lh0PW1=~E)PU@#ZSc68*M zCgoxc;c!DN2UC&K=sXJFn|eSj4+Dl|d>uD87r77s{v$&l-!7w5yQI(Lv)1%fOj_EY zj7_*LLu1p%AFZ0USAAyXeyc{^-0gGvHI8Yk+f`CksC~`kt~K3@k{FuqWU42f?o1I} zq;D8y>l6%+(J4_Ec@a;fCbw1hm!$`4vi+${e|0RGh~qHS&1qT1R@5*Jny0W=0 zjCcz4?nR{ADvo#~@ZlIjBt{ww@%J{jrcz_kI-Jr%o5AUEGq?RCM@Xd(m?d+AmU*4| zFYS~tgSV zSP+`8U$JuRb_Z#1TbZvih6N4Gq(KH1M869s^qY!~XfQV^or<}VW!++fy*ty;nd(R) z#+KU~4jHQkGBu;^G06o*6Kibhs&DPCOt;4;MZ8ih>h?s@8&!oJNGY)=`6J3WtO+i1 z3K^8cWD*6|Xilg56#;Ez)gqcNH>?`n+ zYe2IqNc&n^d(GON)}S_{Z0$^3ik=Bf-XBw4NpI37dNZ-sM7$+l zRx_cWG5Q`imR^bVpt*o~X5ok~HO6rso{CF}@-DH#MjZ?&lz!KskG-yF$mDj~J!;khI=b>Z=N?K4- z2RWufxB?~lVo6J(iB$=|Mo9rBJ-8={dKL&xGFq^|h`6CLZz|8i-JymQ8OTzz7hM6rxNj1_4wIUjW472=H0wqYQhZ4YCtmk|-3flv@ zpBY)@rP#H?i_o7aFkg?7I(8YWkV^>W%TUt7Eog~X|0{g98L$2@@ zlI19GFe-28r5aR}C=_p~D|{u26KVhio0aQN6e0LZbkcRs@P;Mw8%~zrph7wvs-Yfl z_@{gaH2k1Rs9}*(!>!c$pivBwOpc`fX(oXX2%9i72~_KSfq>8G^ZR`)=yUjj@&`Q= zEgT8&`n^fMBl)q&*_T8O!bLJheKsI99o!W^AvOn2|)2KcJRWxO{ zt=Z|Fn|9wci~@R@C(z4j4$6p8YA8neZeWMvoD0wfz>byjt5BaJ9%!0yB`FSfY4;%^ zSZ;;=zKH1Naca52kH6=m>=b`8>CYb=eSW8QcOx<6P0$+3^*Za2KZRO~v=p?*p$N^& znD10CRcWBgS$8adNFbREu#;oY$$cr;P7o(B5nh63qJnnxoR~FZ=J%j!5Ou(Bm9V{5 zQ&&&#LQ{wgPMl)bg`~>9GKBSswYbeGr%Ppz@7+9UU01kesFK5AAJ(tgu~xFlQkaE1 zsxz%IKRA1frNWV}>R2_oW_n9p4Lu(=i9Xn=_3=U-V3ZUXQ-EO1pAgW=Ta5%1m=qXB zG^NR-n5KSam#$$WA*rIj3_DA)R+lC10Fa5LEj{~mN1CS+kYuAL11T?+zeyn$?yOF= z14gsKY^|_osybHpW~UL13{YYt{euZz$zXf+h434;-|kl6N}7H01ph*w)$;tN6_^C4 z+{`{KK0|C)v9~pv8&ch=bQk`o(_Js7x>99b$z)dCDmS& zNu{mhT%2uj&}GhRaoOxvCviEU@t90%T^$$cGU8!JF%?h7x}mt+#R!O)S4U;G_UvPD2+>d?RBQO`kQaVV7$q zvn8a?FVnhC@H&hJamZppau?$dx0+$>N;MqD9b%k(|DlsY=!Fu0EeB&@@0wlOJgiki zRtt0%+tkq$t!8nC+NgGVtmYqXHW*#xVN)ouUSSiv2RhHrrx7u?Fp2dRZoxS$aSL`~ z1n+4?011CQZ(;Y;=xHiw=k znlj$YYh-hYdaB^dfFuRzQ0^D&*xaAEpPw(^kGR8hzx+O>e!OprOylrAdLHM4%DDpn zI6Q@={OH0>R6il0ad?(EJRS%8mOi(7?DPdOhQX3*3V5n}NIp_iI4)XVMWsk7tP^zy zt1%mf_i1ikKht1#xgA||D$3^cIJ~!Ob0`01!NP4P*J;C$MyfWSG-b-k2bP=NR+7D( zsJmaOTZV-!>S3!@6qVq?gNij;O(#`2yP|A1Ds-5PcumVbGYz+gEs|NO@St)ND%8~C zRST}btEk^C;9n^p5p>DwxD0qRRZWN&Z6*wzkx?B1PYr<;w6+l(xt0tqNYxI#vqm*- z1P+ar5w*?Zv1uZeZtkj_Qt$KF%-O7sH`mr8cS1HBX3}HK+B7}4G~E<}6mEC(U{iW& zrn@F?RQv2kr^=S7O)hJnYE@aMwl7Q6ChSJPg;buH#m(%OuwR7|R2dEGf(%4-3G-P7 zxJW$e$x z4uDjbSk^IQ6|F-ZOHG&XYR zMR}Nz;R!XoOiy2}HC|8O{4S5_Kk*`< z>6WWQHk{Xbu_NvcrG{xgChVi=}FKN3;`u0#2ORnxIlBE->s^$u*a^MYzPFz z_vGK&UPI6?@^i}l(sQBYv+|cr?NOg_gIwQEwV{Tn$F7}h2nNQU!rL8%ci0EexW>sE zyoe|3+)&{ci6(Uov37B&L22_DFqme?%vlNJs2O_JlZ_7 zD{S#-44KLeb5@?duwwQ(_is4hlHug!z4Fh58p;Er)s~GWnlrA-<`AsZ+)X_J_)=z9 zE@+z6)N!iW?f2!sjE8f>%j$;q?`qq)ZFS4K`SrGdO=qyc9k!x--SK?~-f=?z+%x8N z9=g0^MyRII)7%>>n=}-wszJYyebzSgON25!ly;|@;p_|LwA>HHUa^SiG|;ai38M~^ zr#B`j=BBa_M-d8@QYKB=(36nQba}dsSUyd9x=zIy&Gr8eN9&Rjs+%JyM@-Ae)#}Q8{IPjl~-Gw(Y-X zqd_gUu2|ku*4LZLjn>tsprf>U4eD^PAsy*y_oaphJ0#eF8KZ=GGll!*jFZ^WQg&w$ z5LUq%03GcE%ot(lF3}j8hYDh*&6E z%I1Rdmha=U@3^h1EnqU}LROPXRB7U|4--{;PMNv5(q-FmrsO0kCtlbL?ZRW|=e+UOI=ZdqCe_K^n-t6=Tk zeGLt+>XDA}d4pj2kayRql}wg&Yo$FEaaPaF!8iX^#oZzsWh; zW6Ra1R9VkJdHqnei-}cfoy!sC8h^+U^LV0ei^?_YFb{J?T*$)_RGA`PM+Q2`MBzLB zDRdT2?8L(JWIQ_%@h%pgk-MS><`K#|Y%Yxt1wF-D9P0^r0E@x}0yv*C^|-xTTJ}C} z>X7n1)Uj@~x_We7N9S66kF1;7vUg|e-W_dOo5rXMnyhhWrYi|q#}T)gf;yAd>TA?C zFFStf^kXUX`?1rf9>1)4Qq{;lon7lj==rYBeMYM0)Gz67UD6uRrYuHLqthkYQ)L|y zolc`NS`#`m7(^8kkE_^g*b__3{?@R7oT!Tc#{}i7Jg&vYkRnqKVO&&dQt~ojPi0Oj z8?8SIwsDWaV$_XBDg#btaWaN&Hm1_TJk?=!mFvu{*zB`4>eN-8t2;VYcXqAn>|E8A zn^A*+HQAXpH8a^h7UKh2Qzqt$Xw_Q3)ofL%ZE778Re_+@p`U5AoMbm!&AK7&r|qk| zx>t2{tm^Js-QI{vR7dorx=`0%x~l#lo#CC z{(8NPQ`4DD{dkt6ZrYZ3$|BjE>um0s(%k{Lnv2o+d1t9V>HmmJu+ z;Tt>-hux0-`243XD@){Gv%1(4gY}I!TpEgRmw%n=W2%=mdV_j<{!95RZj)DKh@A%h z{2@rc#G(ASug7%DcN8rN$ey4kSzVn>R9CapDyNiJO{u7uQi=clO-Nx|pNf(nV~PGk zZdfeyL6eaw=PEHM%MFVDqG;}Om6EkX3N9wmM6pB%wwQwZuaoa5xORGics=S;nZc2+ zD?KlNL&?GJl_ewD=;wz!WC2V5O*p7Vcm@3Ovw%Y3$AEE)1JRsO#;hpAUPqb;93eL9 z^!b3GA+WpnR@snfJzaT+Y&yIku&A{XI|(oIE3vZ(T>xkZ)G`>=B8;od1l%gh7*F03m0YZG_B@TsppsHLt-WI?UY~B=TO3sF#Y|FYBMidxhri_)I1Z+W6TeV|~<1ee@RbBX5oU z;#ZJ>o0qesF_Q>VYMd15 zO2K;6iy784=g7XkBj*esGN@)IQ+_e@LebHhuDw%Qqt2|ZS+_`!toO63YDObmZx}vl z*6b69`wtt{TeS~huRa=WXg3-RT3v5D8&+M&MHY)hVqDw`3k((6YpX5+$47Q6-NFpe zEQMx{VWV9==^%&fHeE?+wz8dOnCg=my^8g zl+`2vd+RX2-s$%{*^vDCV(+T9wpG2otJ>RF^_F$UCa7KI+NzfvR0l=EuiH!n%VJCmLK(V5}OkIX^hJU{1L$41HN{*@MG4FjTvcP zaJnnul-r8oWwVM+Ktpu8Dmkv!jHcN+kH_S&8m$$Ps)34%p_)XQ#f)x2=q=f&yA4ho z)lJll=6HTiGN^>xqG((Zy~SYF=H|Ax&Z{+>jm{87-MZMWy%oZ<{5eBiU zSn#abEV_TzFlk`9yeCG97jaB8GXBj6R)>@a4FpMzO-&GeWQh+Pwi1;9oS)3^X-HTh zxf0x-Rx{24$=vskxPgrhN3X%}HTBwUJ~rJ9le9jA&uf}&cRC!CjPRN3v0Huln~i?E zqaUY+KF6806rr-VPX6>IT&{%AmvTB&oZ|slM@@1#jD80YuGwMlMS-E$mfzFS0|m9A z$3Aw+G1=hr;Wfe9S#|YPRejy8+C9DHK7YB_TkiLjdqER3&^boLG~)UF0W?tyALBa_ z8&)NIi9ZKS6NR+cMKOpbw$f8tibb-AuxMcXCK!W}pi)H4MdGzH(!M9%Us*L!9&1Qx zWe0OIo0TFh25(Z5l3s7Z;YjQ!+w9fT=h-A%-a;SeRu9+L4_AkpQUSQ0Bhc$0El=38 zG?&fxS9*LEet(6}Q|W&n${y18xuH>l&QDg*kf-<8d=>N)L{;ERg%%M;DGIn4(g|nI zaTEf44=lgy>Lhvq9b1D;*IQ&7{OI`auD-*TG}Z5U_)x_x1ZK1u;|@sYpq2)D7Ys>u5dY((FQbr&7lg5%HkIm%XP1ENb*gheTGI29d0l^RaE{q)7 z0`13x&s>O7!ks6|xp1I6mB;g2_@Joa{Z-(UlTL^A9e_O;Sscj}OjB7J8iswT|D30e zEq#jG_|)x35({$-JL5m4t-GAGxun(2Fa0?g>@epfc#Q4^xtGSL4-t7wBUGG1#Sto= zX|XsJenn{pjZuwA{0e>npJFhfcqCgpigr)S8TVl<6>K{;oQgtrs;u@OAPsc>&oU)z zr_Y-!W(_A?Rc5_YC%GaKsb2E;P3e*BiIBnOHba3bc^ev3Gt3?&=W{l%oKig+lzb5w zBx_~KwQ4V{Vs>s52oWOuh*(qYXS}_Vl?O&WCV*^BdRQ5NPyu@ z>P|_UXo#h!4)n!)m$!O7<`A;V>NIX&vMmwn?e7g`dSC!0LeolCtaqWeYKYPhel9Yg zc9PPX(PYeBi-9Ud=gt535kXMrCHPp=$gNQmy2$^#2lo3$8 zjMnYDKb3Wd-9!jT#ze;doPS3p!?RiLu}Vl7&=c&w*W{@O-ObK@iQVa+1@C+ho0J?N zn5V|__ju<}8}@Ls*7Q^=>rdtH3w-*MjXB8$i#emw27$HP(DUXp3c373P+u~I4NqxQZzXrTW_KGuSz08c&7k&!L14wSWPM(rxx#Nfb6Kr4k z+(1i6xd;F$Ry`L1r7Q=;WO@ayObJp&dMln4r%3`^-*Uf?XZpdc*;xc&+ia-_CGl}@ zB$_Am9P%UFQ`NTnuMiebE*jb|0u>E{vXEj ztoMHwyK~?oL8P1nvJ|xY)kQeTMY?)Sy8kj)!5|G^UyLtIbkh!-ooMB32L@m3xEMzSk1O+ zFz5Z6{9YQ@7LUVV6la*Qm1Z{n@JF%$faEL8=5anO_fZL>mS+DrX$1A#LuObmq|d*2 zB2i?84YLsBaUbO9z~f;rI}3?L}fdFH zDMOIps1vrZKXIL`XHeQbA#z8voD%{=QOpss)E=_1loi6eHUFyh^ZXm;AdGxj%o4WQ zw}c!{T*ObOHUD?3g(a2y{%(=6n20ycMI5Le^sfh8X-U%~a{@(1K>l5%BgMY>pezu~ zWP*XR()S63M%h_%Lw*RBl?DA}W%*m=dor0{@7%JxxR<@SyB;Pj8RUWDJ4$_nit|Sm z`-xt1SgE#qg78Fx{YMaUyG|J5UOq#v6;om7Uf#K0`3zfbGG9ZB_a|PIzt0rWd~Eh$ zJ5=+Ll^y=0*KW4ibS8GIPYYeMDL(|7z+fRWpZ$wZqrBeAY;8_?QnIE2uk)y7>`%XTE z8%>hk*pX(V zG_5pe&cfT(9*f1R!rKGZh!J3^B(n|9N{E~z%!jAuMtL>?hMi{q-Wy4j*!57Z~*>tT(2xtT3;huwCY(;e}L>cXCo$7yxCqv2pK_$W>sHm8(I_$&OL zuqW`=^Pr%PsORZfE(4paVj)1Q=Bek~GTTEuAcG;Z% z+&9?2H=CXAL^xdTfhcFe@360~Nq)=8d|!4o_Q+@*V3(5iFpIbY9RoE{#F?KeYw6#1 zdBZ1%JijsrJyOu?y4MrJ^9_j4ARz}G){!#0r9F7)HSs*)g(wcXz6KJa`VwOtxr+3c zY77y#(WALkx?JmK3&U=`)4;SKBKG{ZdIuc2k;POm=D#%PX$~a{|KiiR4H#VZF7gn3 z6LGlAFVc>AX;)9)uuRMPekgB7905Cf(CNZgzQg%k#}PJ!d4(GSJ$VkuV8AqjbV@LeDB?<4tj}Pxy9DG5RWow6Gb@`1 z(^qpY13H!ijM^{UA;Tb6coENg}rrOM#5T}lC{kMet4Y9qP12k?h$MmJ8bUpjwk zAWG^uI0su)vo0NNU((dHq`hg)x@I*plX^&z9t%#LzpTD@!)yrcuvWmwZ=hAPkq2)I zS~2XCU_YD`#U3}J7l5F`-Z`%2TUAW05AS%zW3m~UQ{zGG&F@4lyGSekg{gR_-lg+m zFOjbUI^ez?w?Om#qzG34>}0}s{d!p*bCbU=&1HG-QXF4&STl4c7yr={S|8m)dpo?< z<@LHYxWntCk4C(r{fB>gBVshOf{*)X@*M;KD`KIa}CyR&Jbb3?%Mg8+8(+eU?(pfw!ed+wlosE0pJ!JmW z95H`l^k7#M@2Np=9VJ`g2r=mV#R*=)MBga715J0C{Y4L~Hc8qGBMXgsmpiz^WHvZ8 z$C)12xXeyJ=kI7d%xY%H^`s8k{N}8Nn@#zDaqhLimrP|)@3_il!}wGdzU7sCyMhP6 zePH$~mNFj973jgR{N$Vww6(wU)M6C_2lkcsFQ{&u*5OZysg7x7DX-t(wtPlYFxs=U zvu3cOVz8zfe%wR4sfW*KT|QJ%HQeB+Up}i@s|mu^mNZD4kLqk#HM@Qhaz)h6%GQk5 z*Uzd&Ka3QX!}3Bb*&>O>t(~HQi)=)WUKrCa)20r(i_{y(d;N@Np)*?+&Ly2RSLu{% zGToKdvbav`yVURH?IaI;mgz~m%j$!OpM_ux%jvUPXUV2aKBx6+FQ%&+c=8)jDDE{^ zf`5Y66d1XLlV}_W361SILsx}Hh7+DZB4yDjeD#jgbs zu+b*fvYpyAoAUDfU$pCgzRm2o-E_nb-%c(&|7RUB{bzL#_S>OzEaPViF6?b!hZndB zl#%#x?4BvobjA&$1KS-?AP=kX+8@{JA9eTwHu~0kY%0}7qIeNL8N>CE*=#W}kH?e$ z&}cQA^Y55U`>9l~iQ)??)eFF1(8<^E0k%dw9&*mju*IN=G6xDLqP&U^vX$a`*k|6H zr)*z0KmsD)s46~$@-yXfAIfh9M|d8}>v$VGUp$1$J>)rc3p7+0puCaKV)u#jP<|8e zCDs2fB19?WtHiZqS0bor}6U%(&s2lxWF4=MwE!0ydwg;L0m zUqn&s_#A<>8%3FPyvDU5e=tao&XIgJifwN5yKujoHwjPhXGn{Hd5poKDXDM`0OM-4 zS&b}%VVl{l8!WLy1D+X9p z@l1G~F=JNivD!j9U6aXmzR_mPYMeHEM5}8t;x!h7$uR|b0Aw@NHp6ffhKhqmlU+h` zL!1KgC^rZXBl5RmbyDPJ6*d4YhZbESw`BMJSp3Z0do9UhFbiIJZ^&FIAXQihKu^Y~ z#EmKAo5?&hVzh!wR%u(E(Xid6Z_=nNMw`Vnk6$wx4x@UFN!RByTKaTGt=?=j3{h{_ z@|o~Ae1-TKis!D3mIpDk$H*J_gx|)TjYfyf>KSzRnVeRWO`~gc#`sKMx7lJeJ3Cmr z!C*FMEc$L)x~<_kwg#4Ia-FncE>D07+d$Zmdm`CcsnZU zVo9je2m2f7Xk}GpHHB8>82iA(x0xdnv8lQLl``VPN%m?7PF++ z!Fo~0|8BPVGj_WVnaTAA*gETxeclasD5F-*5YK}6LMNZWHu3kBwF0cM>zc)M+qH;| z?7WWegroRk7&Tgp%X)@N)vouLTplx)wb5+f<&^rO9=LE{{+>KXpzT(KS6~ZTDDo7? zrrjP;-r#oH99ly~Cx6eEjksuzxZUy`iPm6_FyS#ih3yZE6vaYI zMt)e`vTY~rL1LFp?#9Pti+R7%Du>utEaubI{K{xYxz!1ad7aB@1kfs|QPbFHV!~9m z4Y@Yf$XKwE*oNrr#GPCP5q0rLD%A#ausVonGzbHS%!lr9E>Jx=lxu0wYjk?MO5=?< zB-pGYt0BdEL%NLmxEvhhl?_gw}KxlYR)$XOYuo zZQ)~@T_gwY76U!`lNjh~Urog2w_7EkRx9#$gmXT*_YPzSA$JE>FH#zguik%FRGgZ| zpFHZq53Coy>lpQBA~A|JH0In&;%O0%CglEMY;Oo-RHE3(nJIgSl&CGTf5_)0YN`Rx zK^#hnHfq2-(0Pf@X(%rm&g-zMh*BW-m_YEf@kQ)zNMuFo4S0%zl5B^_!UPasM0=bd z{&Fsk6nqtahbJ_AiOds~=mR2i$#00^I>~rb_!9E+xEBI2-dn=>8gNbu915=1ATK4I zmvF2Gyoz#N!Gsz}SOjM^X68dbnwhKh0k?6qgJ!1FG;wAM-(zNiE2Nc0X5KqZ&@sxT zxd4+e+v3lvjpkrY$YU{Ub&|#^1*<7zOMMKXLE$fn=?Nu}B*DB@a#}(zT8n%KTOzJQ zuhCN2Y4sBA9bd~V9SDGr3It!lH~r%~N_>>eOcg)mnZZ95wuyU|Mq<23H!WdW zy%KuFg1E1biyOxIDCLPFxxWfwDV9gWHj`U7U#oHQ*F~}3j_fo>-4IZ*!ber{SA`G6 zeM$EPx`xMh=68l|POWCX-faXoW?i8+@mK9;JUgT_dR%6ou3iNNBRJ51vctqJvaOj+ zvv#hd9_wJ|k`b@5M(eV=LON}W$$P063>6X=$K*4l?J(P>m=FM5W}u9c8T^yk;nLf{ zP8reW3h>jD_-o|v2iV5(!H*f8xfV%hv}v?`MvvWU_n0Q@)K<_o%RGK=i_U1#X$`vR zm@%eldRS;e@7|2sbNn?nfF31X-q;)=&X4wD$0(e~bKN|POK-BMb(4)=yT$G`_GwYM z-9JxEbm(pJTaXPiZ4!_gw^}%M7 zpsYl_XNt$s$|-R=GE=6YM<94~1QB2m;{{GX<4xH}bB>}hK=0~8$Ui9)j&?RU6h`Oc zSSVT>2_xE{I~H)J1Cg+W*c_tlpR%LH_MN$$C6$$5gUk686h4D{QoKcYdpBH;;?PEO zRn~AQ%mBb^l(`(6$pkLPX+_f45|^WOnG8r^G?B|O7;c~=N5toa&9GK_h!|MuH!XfRy$Bk?NH}V=`AHtbRGf6Ha>_gnhnZ!SU3;+Wu z+{nJ*MqXAhAn8lSpV$Qh?#hjvBl95R+=%=OFn1KXg@+2S%dw8g0!GG2SS9r|&=}{0 zrr7xpK|kz~Ojw{*y$t)V&wt!@^ z^FvH7Yaru=fa5?SW#qfDJ3WpGwPN{GY>1+QtR#~YTSp@iaTz3kz$KbZP9q{Zh3i9B zD?h{n>urx)v{)@RMQW1u0kSWK@R&T46p^GjHFoyJ_!n^&G*>6HeTA!JTfvei#;*6X ztz;fhd;m`f{0m`!XvfZg)&yr0{xtG=qp*)~uyDL^j&Ox=v+#iMtnh~Lk?=h;vH+`O z?Q8%s(H66{>_B!5JC(h~JNXcw!+*_h;NOUC;($0yTq5o#9wweB-X=aOJ}bU1{zLpq z#Z)GhS5>B3qFSdqNOi30EY+o|8&r3zo>IN0`b71uT2x!r0qh~xtGm=g>ILf6>ig7B zs=w0=YUXH`YSwEG(VU<;Q}dwKpoKxLwnjTed$RT%?d95QwSU6a?33D8weM*^*Z!c> z>72T-Zk}$L?hM_(u|H{A&jfP2v8HR<1 zm4=OmBMqk*9yPpVc*pRWvDUcSxXF05@l@ma#;c6C8XqyfXnf!JFJr-EHAPI7re@P* z)39kT)BdKTO=p-cH(h7C)%1YrS<~C*9`jk|OU>7qZ#O??e%X?+EVP_%`O-RMooiiY z-Pd}kb&K^Z>!sE&ZGz1PN5~3Wldab_)3%3gmF*zghxVL(u6>z(z5NjT3HCGXm)Nhe zzhr;M{+ay;2V%52qK+y@i=)pm;#lg~;5f!{remw)ddF>!2OZBjE1k{G$k!uot}|R0yRLHm!F7-8G1qgFO0r2osZ45+j+9Q3&Xul^{vh2a zJtMs#{X_c7t#Uivad)k|$34wG&%L*MKlfqoE$%bj-+9y?$rJTln5AGejEO>43mf(HCr-Cm9KMQ^z z(uC|t1egicg(iijhxQ1q3~dM<9y&dAVd(A9r=jn{p>SEaKD}%6M~pa(s4td3;m+`1slJ%i`C>{}_KD{j~=E&sfHM}@A! zRS~JEs90RFwqkR|F%`e8xS-U0x^>KA;^|IwaJN_j-N(p7l4@KU)7z!}Nx24G%QD*YIh>H;vXtPh+-mO53+BS^Pb5)uTJWi zbj+l`_WF8H>V0K$-{i;p>iSOWyQ1&GzJK=X`aAm%?BCk|&VXy6f8fM{N2aKztekSe zl-s6!IM_6J=HNp^;i2K7V~1`T`gCgR)FY?9KkcAtzngaFv`3~rJME2WA5Hsax;WiB zy>fv*T z?-_n(_=DlkhQA#VMsy>#5${OtNbAU?k?A9IMiz~%7+F7Z+{n!%uZ?^)>K<(w-GB75 z(I-d$H7hb})~pq?E}iw#Y;ksQcEjw&vrn0Q*6a&sKQR0G*{{uhclIZ9ymKbaSvY5- zOu4b6$WHn4cYAH!%i8gkAc`LlC3v}dFG!t&@b`O{oKjd(NT_3~bvV_4YnK1`pist- z6Kwx02jn2W1$EPbcZkObYC8Do!YDri-1Sc#PYR>z*|_IPLBv7dJN;JOD2%FphqCtn zqhspNy1%PfXFa}e1B^d)JS~ihH9|ATcj;h8{2mrY>6;k~AK=)51C+LKF^(^kLm0+& z9EDHt9mMxPaJDcAH;?pQw2$<&K_&0uEgYW-z3?Yuj&pps5 zBKD8@KHwt%)A1nQ`M-H=6Qbg+g0$^~HMED&OLH zLNy(D*8ta|?57TbWdO%D%5xK8Qwuv%g$6h{%vs_8N}Q#Pfv#I*>0@Irb^MH1@kxM&F}_*Tuai z+(%_JR{t-LrPyWtiT8Y0gnr2jwN2UsQ1CRB2E^foReU2IuorCo@kIMkVUxYZUgdE}{;J@y~cpeD)l@{V6yp-8@LQwRA zH#iW_(!Ku>CJUbnQ`nu*o9cy1ojt~-fzMW_w;wutQ$9O$V zUWtD>4>^n3yG6UC%4iU`}zJW%yLh$lVJUfJW@&V2l z;FwxGtl)c0LJ%;t}$_z0dcfwuh? z@Xy0DGlUT_AtY2M0QPxgK_(xtGbbM)SiB14wI(0PL?Be-Xv0C}G~yjNYH>tyP#OKE zI_Ni|rOMw~d_$j4UZK#z2*Okxbe)c9@f+KZWB1C}kQJ~^cn)RX3T|c+I@ngMj|0#T zU!(jbtR1Wuj9-k#f%Tvqeax=Iw-IZnf;^1J-eJLSY74;+OFXg)+X+U33liJVWQj?U)21okz#(;r@;u>hD?sWIqI;H#W)9En7w#3z-*e?1s>@k6DzJ0tI{OB@ zThyCas5DTZmmEml- zIouhZ9o{p%HoP%>cz8?rjPN<(^TU^gzl$29wx|^KN5j#0G#$-F>!WSZ8*GRk8a*ZY z>*#NyS4FRh-Vpsm^p@xy(Ys^L7-ERW(y^M@k~oiR;?}q$F2()vF!Tn4@ul&VsmH&k zOuqtJLLTI8!q#F-wzA)`2ibe%)hHj|rb1K3i_d)~FUOaYVh*U^H56328t! zL^nr|K})WPUWt}m7rhBBxjlN`vMJXpd0H3Qy&? z(=X*OkBu7oE5!3}6liCA!e8NG*X z$6FiUYJ4lc{mt#y;@^qe|0)REU)=r*-L?Hr`E%j+h1>hKZ`$7T=2LH8A_#B(630dy zhrYS*n^WJ!+WPxTnvdnV_AphA><;pu7!~7~F@53y*2RE$vUs{ujhGm;_O;$CYo)n^wM<_wdbpD{@&b;WPM5-px-! zWW;`WZ%pD%m{+IsQ~7DUoi9bZL$Fg$VO0&l!gHn^ac;G+R@f{Y1dZ=i!ZpIRh=1^B zL|c4AcoRJ8d%|aYl=t#k{8zA|e?<6-*_ng+SPaqm+t?)Nf)}tw$i}pm?ZXa1RKLsE z@&UxtOZlBIqv{o=7IsX)po&)ed8FSG2}4I+Jz;;Tw#H* z2cmQxE1WHyfXL(9gwy#c!dBsF;E^Xl>E9QwW`AH>;cFq!EDXVDn3YK^$Kou_s-evu zWYgFTICCF_Y*L4^BM@8tc>W9aE3D7E*}d$3_IF^O*MZv&z*66Uej}F`=yHO2z!P|d zD02yAEFe^{5O|1s;QjoplS-81x0m@;!{$w9DCiVHtR; zd2C47o6Qpzfv1{-XtJxoYpj8WeTlFSTPp0wmJ92_x$VzZAe!wO;Sjc7IGk+|4rTj6 z^M4#WSh$3p1uwP3gp1f2!j+I+t^@tK0X*^b?AOB0>}ugQc7t#$yI!~jG3@VWe-iG4 z&%wRykHTy0Z^Fy$FT!JpiTii%^{ z!e80Li0M+pqQW|1BU^=Cf}`1HK`-opx7AePM0U9FIQz3O#5#n_iN9fH@FX_j*YHBI zf%lS_UT8xmf*w{UEM(J#rEIov8u;&v*_pzPh++SG_FLgecAxMPdrG*S-6%W=I(Y^= zLHI3rrZd?V;ZycD|CoP-DDR*0$N5vp-TMSE{$KcK{Au9JXZUmculzCWLfpu2;jx{2bvmo|O8iiig01h!F%m7uM#rnaOO-3w~nZgFP zQaAvd?Y?Yp;Xq)BBiKgaNKo`+z+oK*PWK9S0U}vnC|m}Lb}2hs_#L>no7k1Y9qf0) zoycJQM|O?yCs6pi*iG=6yhC^xoY({GR^fH_EMiBXTOcYEYiz;vy8c=dA>@sx151`23LF&m0-Kf z2X|f}w6aP_LuJBVY?iQ?jRF(NZ}XY_)lfW2)|^<3a7B6gkORCJfHni zxBy)Jh3s_UJa($^7xsYg40}lU8+$}}jy)zk&mI?EU{46ovPTh3bi42odsFxX(#Aj8 zJD|T=UdL-dg@^bQKFFu?`FsJN&FAuY*b&-??~5Gc2lK=Df&3tTC_kDX!jI*L^COT$ z^gMnMzXTDjxA05(nfwd^leDnq_o~wfshNcHh#OGB4hX9Y2w47&Sl~EQepBAJj^b|- zb82eww`#&YYVgh{6w5UzzpVJ%g5J5m_}hkF|F-zs4xJ;dZ@d>hZsHUPh<>Z&^2yjS z!VLwFkj0z)oW`C@gl_;mD zCgGmVxPO0?Z^C(*T-#>Uz6snx7XK7{yaxB}4L)u;?%WrpN__{Qo`b>VZNT|zVLjfq zAATJyU(N33{R6nNFYa0o*lUV4mxIHM;u*sFYw&9p>fD6m0MLMyGRy;b=YBG@Gv(*0 zZSmdIzPqw0Xv#|YoxP}KJ?f+SSEDuSQP%+wzFvKsw83<^1YH*oFRy7_>rG$%Ms zUvNN|$o#x=GT{^Zu7Z#RzU540@Bv~Prp1bXgy|V_J%TEknFW-<3hZr@>8ukp!X?}Y z>?eU{r-5e+!V2bQ9;}26^9dhojONlxQ06#G2#+EkPYTp6!^(gc9!HL- zO5q9QYN}>6EDOq43mjYy3FBK<&l-@)s0mo4lC{WGy&X8R1DLoIwCX9)tiJ#|{+mq_ zo`&V!WY)*}*#P90DQpmwU`WWasi2j-(v=R0uG>&9S)j(BxHo6u_Jvf*7tFU4*d>R|MzSQI}tSf zWOfSs1#lbUaU-JQ>EOW6WM=_yw*bSP14{KgB(pW3QZIn=Ux4f@7qN@M%UmjRGg~2Z zY-3lj-+(%`0@qv#{^n}n_jbsWAAzFY$*yD9vm4lr;OBo2tbCL39(awL5w~Y8`y;!B z{fXVmZezED3hcw~1YdMFyNCUm-3yw2pRf+R#RK3n9s-qr7;9!La4z&s;8Fz8`$ySh zz`avIA)a7QBBjDph&BH-WTFl5>-;Nd_W|r#nag^Cy$Cw8iM`BT0abn#+{$b0b@m2( zlWhl{9)b+}9ebO-!`_9o@jf_^57>v0IzGbeeo@%WK4zZ?2eD6ugV{gXXY6y(yD!+k z*q7`p_BHzkl<`~k9s4(Qi{B&X&@kJ9oG=9#t})KJ2%Y2G!U$JGL!f~dh7M7f49Fd7 z;%08)R^bqC!`eTT+qpwPJOJdjk~orTaj$R$_aPH}KsZe1_rg4a2)HpG=Lwz!ua^e5 zSB4xy<-(DWiz;~)@+{95J_VMY3yM<%nv(-BS;y;zdEjsxg?FL1`T<;S3pn96?Ae?G zui*~xk6qw{d%ywr^2y*2kK%p2UpR&j2*>g%!qK2sL!ejFz||7RJPZzX6kP6X&@E6e zVWF^xOv@JXJ%l}lMZ(9>3*EsNfzw`s$V^N5GGQ^_TX-K_>E@q!V*d$1s<=D$LpgMC@qxo zT9&Y-?1Y_C%32B}ARH~N{$0>`TRcrICpgJneEJ(GiT1sJ$LR; zU|U}X`}}h4XL$Z~g?6QO6>Q`}w)Ag??eJ^p?(5*kaUs^A7VOqc(SDBIz#{m`@4$Cx zehF*)TBRFyM?Ng+MX;);Do?;y^iJh2~rl#?Ize@+u16+O}ibI*$(AL z@L|0iR@q%jf$|ad+3tld|99B)LFK2)W$-%rE#`^qu@iYWtof^9zul``rCh1}mvV)6 zAFQ7zmHY6`rw6d!e-P{BhqXtvNAXpt$MN;3-@^+016B`zgmp-k;d8JJUw~csBGwi! z!8&{eUm)78?a^M<_G+&wmng4eALQS#&e@~AjwfDk!XNZC?2x>w>{a$D`?WW;{WvxD zfc97Ip!PR>9qDcD9qnE1J=mY`YahS@{RnpFKeSJ@Pqojqe`<&Dy{0d;FSW0+`dF(zo@^lr^V98 z&A0SrXO^kEB{zCxt9m5$J(3$ef*Z+1v#oCwIgNeYk^$X116r+pvY~=LS!83gwNEr$ z(B0i*8DL861DVE(7ABLnfg@^3Cd`A2ht`X%$u9fg=sb;m$!2ZZbjviOhtYHWTv{?A z2~6s3WI`MJdKizM>9S6f1WywLTBdDWxq9iaodh7!+?8lbG}@<+t}xNq#O*>gZI+qb zd6t=3jBR4ZB2~9}CYmDXHnK>Hnj|I7dLMOJW{IL^Wfj#d7~9y~B1&oIzG&=gmIQRL zgo2Q68Q*T|Tf5wHY!)ke8%0#F?ABgMK(FlHURh+XC^9L8D3MIcK5f#)L5fRHU(T^B zM%GIS?3K;v(@T@7dL_gA5{_e6Y=Wq--?V1+&?aUE*xqM7Hd{_wWbPKOD}j+_KX$|F zp$#jtMnixRM|5(s zQ8KDk7S$??XqEU5K}chJzj-di(=s=U3_Wd|Xi&0IvZ+-zqP5R*LUtcDQ6D9e7$-b# z`t;w{oD(vW<#l7L^@MC*w{=-h5Tc%Q0x2QO2`uoU;qn9T)uv2^RWHoOr*%yF1x$G*>W-`5$Jx|8%>gR{q2_d zT)b`m5o5d~X&EvYg8t|63kB0A3hI_Z+|7&vp2q{^Ue?98zkitNiXt)rq7jmak4@*`?XHkwytRC2d{#ppax$t{+ZMt>$6h4KOaC6&qk zyp;{BN0zLfJ7fDm zpSFDc@bKEzLu;3=Tw+N=&x#FhI$=D(f%}V*rk1GfIvWLu*SSdZh&P z$|8GZkwU(pt!1}2=@KI)s;56^B%?ukWmEd}(qyV$DR+I1j*+n%1hPAlElMpicMI1A zZQX4j8KbRb$<0FUlgVDodcpto2E(AOWh)vb)h)6W`siutw`|B7dabhYt!?%VNAz~G zQ8KDk7S$?=Z0#^_B<;L0t6Dv-O*A0cDB04%xW7TVYE*ff^MG(59;+=g6IOK)N=y~&XL#=f==%jrTAPtUL+?{q^; zpDwiY8JS2S+3m8M`&%q$WNT^J3pk#IXP^EK%b8rf?aU*FeMggN*{RENaGko6!_|3b z4zC|+Sh{v(O)l(O0c<>7U~C&#VE8dEZ`sJE^%7dSS*F{tatotu8(^8Q72xp7zHwN`$QF0EIF^j89$C8v?c6xD{tWZdk+sXseVf*ghyn%YV6XR@ zSFc<@B$2XB-K#d$mu z2_o7M*DcHLX&73~^a3Zus!5vepiQRj%44mOmAl!oVr1mBp~WMc(Je-qLP<9XHJoS? z#xj1p^PvD153e3MJ(D=eGMm6#HKLNF&>l@mNqmyIjd(WA5=~-8!Y|8sBB@XCJ=}S? z@6HzuYiE&?H*I`VV6Z{l8obC(+MRuq%)aT#zG=EfFLgFD%A&3scbici_`{aH7S zjoCMXlHzQhWkn}N!J>i5Y;uyKwK67~56Ql)0-8i+%=V;^`9xB;_oRwR3KJ4|!TzMM zHNa+aUcr-Qcne?`O2t##0z%;5<$&o8M)5_t^?)|wnXQT^&pQ!5j2&$i``LJx6#LM4 zR*k)3q`}VbebsAD)H*F)r)3q}YZq-mS z?Vut2wRRhJ#_oVGlL@;(@6nSTTnbL=65cl5pvz1iBydQ1RC^roLG5kqVPU@q`(he$ zQ3(o4xLyL%XmG^;i5V*cpqVO|B~o z8(zOwnZ0Ic{b|a)HK(mP4SQ{KkrN(iDB%?BuF>xz)Rl|tB0T%EfI1aVwaNFMpi^)O zXdzNiYWjyyaPu9UMNk$>0ne#rgjI1I<{l-dXqS5~;6=8V6>P}b-U95m;|rbk^X&Lq zv+XO}SBhG2oqapt`)&6leBO4Gq84vq+S=`(*gwf{DEwu9yM2FQ$g!(%y8SKt`-K}F zWd)Uv*#+H>o9v%BKF;rUEVh4}S#KQDh#zB%9Kh&h_`SLJVXJc5&5s`Jmu zuXbFJkJmf#cafu;ub()sKumSMA6JARMJu_JYIYSAuZLF_&Q#0(hc_mTf9XG^-%bA` z{V|4(z?$L6EB^k?a>MwiCee`Ij!1L;GVycsBy zf+6Zz{KNMr{~{p1dnx^XCO6Tk|6`g{@=nluDE&6ht3U|?n|?lhefm8;edZT6987;M z(@@Kq9~rtbke(+S0}tg<|3>G|qIC?HGd1corr##qsL};3Xfwyiy0TLs{qFRBaOD-f zuJl{z*ZWbHP1GR5H2h-Hf5!#S7)nWeL`*K1%N$BeuNrC*V*d%gym@5nWLbyUBTL*M zL`tSy#vV9&E=p%)84VmErK2N{yvwc|Ej_HqLzcftf0h1B+#gw&NQX4<sanJw2ElQlMw~u z=L>M0X#85ISfovV0o*s~!|5;6^rSTFA6orh`U%8JI@32{G#toEAVHM)4wp&w9HR2m zdk|$vQW~dY@V{J8^AGfTu3jRrg-FMIB+3-Cpb#*0hMtOZrDYlagP*Y+8;i*<27P%o zuIEwXYeY5L{X`nG67--gs}(PC%+=|i5oJjKkMu==-c5gsyM20j7y+NA{|L&B)>90( z=2QF{qZgp(j*{1qS0nAm$WOg_NJtg?re>8d(xr9XMl`&Y{zzh} z?yOOj4eRBP{yln%Q92gm7qn!RI=Y1a)g8^tNBc*X!~fx*7`bv5BHv;C-z1#42#;v| zjs9ms572Cn*#xriHAW8i8-+q9v)d_o%>6B5M;A8s?r39ZY$|~{vcxfYkHRaA+HZ6U znf`ClGa`+iGE<+#|AXq1TqAvTK!=G7x~_o4LX(Z9J{t!bz8srkGzK)2wi=^dB)?@1 zBx6X$4ojo@jb|h(kw==8!t_hjlF{wU)GAx~Kag0oP)`L7hkw&FNp{G1ouE=up*9F<(%)3oD`K^wlWkYOKdliZ&B7 zxu!w~Kn9@R1eTf65cZ=P;p4>PJgY< z1Pyfe38+4xz6|;~PW34^Q2909tv-}~Lj5AWi>Wtjvy`wlM{(iZD?9v4@)a}M_Z~cK zOyJQ0qJ_F_u?mpLMp=WY%m(n^xQ5TkjMA;B{ zMA-f6j`U|pmnM3&uJrZbDfp*NO23acb!^&{^gn?8D|+i`#Gtpd3sJ&FxGu)^6I}GZ z2KCLyDD6Fz_7zI|3SLGE>bV~M_i{SLC5qDCM`@o4-oOgUMV@cqjblcAhqNhpYp4t7 z&35Arf|=0N44rZp1r}pTY-?`umIY0r22K@Zc}>wu`?N7siKw60UAs z7(0H98^&SG_%UYu7&Cs17eB^}ALGT3@#Eiy3vKaVitBP*SL3=4*A2LC#dRmH`*A&j z>q%VC;`%eLJ-GJaI*1E4kslV0A6m%|ZC3%kQjrVpO#r_p;uN<@>5stuL)v6~ExIfH zFm_Z05!J>d2G)VJEQ^v7gZ_|S^- z0{o&SKD4A9Eh*0xI68{9+=nZKD~78M7xI+1x zit5Y9@9Xk{e(=77>qEqE#C0aFb8%gO>&Li$g6j%gzr=MtuA6Y(j_Yn*J8?aRyr0Q3 zu|EOQRG)?K+dw@i^noHDsPs|GK&1~et%uIF*RjB78h{kY!3^**joaD5@muSGdjegapkVk(D?=bgkj6+Veo&|kk*Zh+_2 z!^#`%;YsgW48a?52mA?N##n*89%_s#Ho>w?!?ANypJf1C7bV~@5) zcuR!*$u4GJGWd=`pSD1ksn9KEj7c+gyowN(;LpsSY-X%J0*D#FUmpBK8L>?yo2!q!rl1Gfp68T2w#KeRSx{F-cSmZ{rD@y zdnSKFop0fd1FP~r{;co{`xriK|AarR22a>xXp<7P1hWmkScEVv_Xs=iPUQp@@{c!B zx{&51%qpeuGCc+Gsp_c+hw$b^F5XUAiP3c$-odoNcXSI%xEiM{y1~(30=`DQ2Ke8q zzg5aGAN&sQE$oD!z6qRu89wB%sIMsH;Q3y>%kUcf-!1T?{2SCjv(}6mz6Jh=kP>)? z5(mjnr?0~h2d&IO2XnAMslZJO(@QSB^JNr{avN zZuq7J;0rpSgqY(|_#`if@9;|ek$jwwy~W?+FV6mX)!YZQ+y~>h59+uN8n_R}a~~wQ z4;r}-lH3PP{EiR&C&BkN=KCb(dnfaKGI<~?Co#w8V-_vPn-RqKQ^-dc-loLyO?dyO zAKs(H_tTm0TbS=>FyGH)zMsW>KbzhT!aEdrKS;Tbd3`I>d%>6KPKf>I%kCJ+l z^e9VuC3C-uxgTRm*I3fyEa}xO=`}3rR_>2l?vFg~k8<^0H*}vHpT|^`QfN*IL3c_B znlyteeT&mUze^l+ri7q5B?SE`A!twuL7!%DvJ~i6iGzlf5cI5splu}tohu-sROW^L zm9fym68ge`QaY4ro7-mKiVP?Mr~zYKaaS8qGoT5C3w7ffz=8{N@$KA z>pn?`yuU%ZrMNazTKebSoWXf7(qkF7@Q7H-yUM_=BS@4n8gf4b3ud+ZLtxJ~Ae1a| z=NS;yb_u7u%D`P_K!jVwvDX?nP#{a$Za_N>=-#71M2jfp2_yDt1NxHz?J}U(4d{Rl zg=XoH`yCw$%?D2x;aV12jc`42!u^>6eWgPl;6*H@^H?aICtt_mj6WF*Yeezk&eQ5C zLm0qSNjOg};}Uu-;k+mv=`KfjtLILHlW=u=1`&c%iJNOcpj6@(8PGBV0_8H6=o95- za8$<@uHzgdcAEiRY(SS95aqbqa~;AP4BY*&ZNXREKVm>n>QHD4c!1Io0`fddkmo(m z$HZOFH@H4#%r~At>$prTL7qL7j-h=zB=R!u;1Q)zjzfei%T;is0F}*90YM?OEreD9 z=Q1Fl4pD4bh+-Lv5mZ(;DmKZucHk!y#OZn&H(gKX^_P7TytnL&vQ*iR4QRFj%`>3I z26T!60ax}D&bz|Etu>&H26UzYoohfBjD?66S>6>!>@RibXq0|;j63$9O;M<#vEc!GN6S9wA6rB=@8|hwF+{q z%f@*(Gw$qBxbt+JS3tkPjt%6>yMr{fgf225;JlaUvEIuJ=qdxc)_}GXT4lOP2CmzH1`TKtYFtJ<4M9tVZY9XK+JK<{ByNiVK|@L0 zHfS!$4DRp5bwB5Y&X5pEoxqXo9RX1e-_@MsIwO`S5a~dNgfb;Riy5bzb0Cd`9x(3Fx2!K}TlcVw~$exKQ>|o^b;fY3LOf}F26f44|h99L%iQ`BA3QzbSOx;YKkp?UdNTcY(RT;C>SxI zT?X{J0TC@+O7KGi*I+j(@ih#%(un4*^0yA$-Dso;D!# zxr{}>3H!=V&^weP2)+c-D(Hv&ijs;l{096V>JZ`lpAn9suXISH;~bPTfUAVd;B*z1 zBhuC4zLMh!N)SY&tYS6eS`BCtpl-$u>Nrj}%g8a8WA7XlJKu<1#Ieit*oyUlwlMC= zKpmc%gaYjq=NQm71G?CNE;pb{4G6f3XF2cH2JSioy1{^MHK02U=zarw#DGAHEbq?- zZjTNHCj{z(CsA#|W&7`U0G@$nk=wkyqWI*5OP(U-FTmvdF zAeRC83@8RDiJokyJ`x%;FdgHUdqHSm;(g!(T<0=$iUBP)Ac|eVv1>=+HZtzaQ8?DI zfyaQ`$+15%peqdMmj-mb0a1>dILGayaCbBAsZqG+4V-|!2&4j}sceC7gI+db_ZrZC zhTa;L_Y*xfXeKClAb580yx>KEiwwwZKz;-IlL18xXqN#|Db-xchX$^}fF=MsiDR1$ zh;UOG*Kgov8qgdAS_nwcx72{XGN4rkv`&XYiVg)g8_*?a-(|$p;8nQpW$0Q1`i%iW zo5|Q62J{dhXf%WeAXgTKo;D!pHHm}nlhEr1gdD+lILC(u?lS}WN{2#7BJzfyp+W)5 z5z0595(9!}l(Cc!dMxuVl*sJ5grLnr&|?U@aSbvw3(#DK<^x)UD{D6(YhAkyF&A^X zO95Su>*~-^AWA|1Lf2)i(u|F%VE-0HX! zX;!7_>$Xp&S0db+!ZQ-(9DJKcRqjepK=@pG9>QJebqH5+_*8lg;ImVBvV!Mil=A8H zOoUIRHy~VvXMHFEM4DVzq8EY3$V2!vzN80OqmrLC5#~LOp87th zeSq*`j(tGGGYfnz@MVNA;EimQiMOy(qqc{`R}kKU^LSOfl}+JYjDHSal~k2$v^2uk zwO0}D(teBZQk>eWDlg-0YqV7R6npOhoY4pxOlKfm#o;-o^8ue@T88j8(|HK*p_Iz~ z3_r(IU2ED8%wJ6ZhH#H*F2Wyi$uF9&0Q`*UDuj<}M(z55!v`7jFozE?=5>Z& zK(C>84u8Yp$(kFOTR6?{G!5`wjDJqWGbx;HB-%yzUFts(Ud1W@&hT@TUJWqjDV%gi z9Ha0?y!oyw$8dNXzD7muqPE|!-Hz}%tr6k1oboRm?qU3;T+WM(d4@uG2HTMGcR1lk z#qKj;b)yN=qHZ)dAnY-niLlH(17W)v5~B7n+=KJXRLx;}l+*ka;dN$up7N#%GJrIo z1bM+dgy;*EzCwX=h{I~T=`O~DtB74hAyd*roCk)|^pHepy+n`pJ&M&XVX9c()B#RE zKxx!XL`8IVSFv~etE0@s9EzPA8v;xzE zh;8L^er7HOTxX_QTFnv0Q@dKZ{8pyK&NSC?yIM(F5KHY^ZgwHGo6#4*P`e&t4*!tr z`XSf*L$2jNXgsPviHC ziOO)AK7XVI5Po zl}otQ+(qfl_{tqZl7w@#$3fdI%)_nBfwwvD+nlR_;Q~}<+sUyDId%oZE4VF- zO~mcR+`7dk;`UEhfE~-jHmZbT4}~a8jBC1gnaY&(bG=hdUWz5F03o$*D$@{QOJ)nVF+v!e2ecQl5uyt4aR|>~ z{1wy|Wj&RQ5FEpqLEx}*3h@&dvO2CXZDJU;C}(g9(>dh|wold*Z;=wU<5VQjjMKcR z#`VNo_=pkw6tx0&0+sM1PIEJS^i<`?T-TM92Io0)$Sw7-ZMK8;@VUfKzEa~JH)1@3<3~8w*K5H{X zOL4L|V9{POk8#o%dPk*GnlY!q-;3V#B1{KiaGnuK!i(Iqo0!}8vkYuvp1;Vwyopo3 zNV$|v%&Su7ZK-JsFduQ=k4!%ynB-jfC-?7%TpD?u1J8LsVoCUjOWw^XcXKcA=9D*> zI+1cWr{B$KZqP0Ool7(;;kox8;1*5e+M_%!qTHtr)&o)It(V9C98OcmEv;j`nNvnM zwvPL{j_SgAq<)ODu85kJGfe!M##-)7vP;z{rPS;^DqUQri}P;9IbEvyYwqi9+HSyG z@s(0lJwbbkU~bE2-1g5n{UB3S!=pG#qYU+8bf{5Eqb1VEqNQImon=gClt*`zsU>e= zRjr^BaLN;n+0U7uA9F3Ab1ffpOUJ_xM^$UJlM!n0U4y;HT9+*o=ti0s;NyqT$FPq} zIA0+-{~=@konvP(+{N$+!y^pWGhENPE>wPnTo{CGWYs^WUN``T*`H=>xaM^H<8T`Na{l4qz{XU$J=T}$YT&kS4i|B_jKkF&uIF$Iy$OyJ@UY@#idEB0`AgnO)K>B)#jkpv zz^kOu=)PT{^*zS9yl;@dbrOGyyk97Pee$`$Fp3JzaA{>LopM5MDc@Yko-Gam=wZL+TT=X7+~EU2~2-CZxWDZxn^puW(*R zNXzFk%s6Y*ryjyg1WMs4MPWII0#nE^&I<+J#v$8d@Y^8_ysZ)1Im8YFVCrQPym9D+ zHt`Oo3H8w2NxTp+-$P>Ptp@mtqjj)Hq22Nj&I5*E3&IHUVYIN;E5dg@RIO48P%L~} zRh*Eg;#5)9^dx@U@q00Rr6+0qxPvbtPS}HYG-OW0xsCHt2WZot7AXOnfl+ZrzpByh z4RJY#F=QA2Nn*T=$p;2|Hk=0IhpDDFKn2xwPwf)-IcUddjAabQItG(F29q}iV;h69 zkHO@R!8m~7maB8nitXG6m0E=sWm+XLt(-!%lUf4^C;4g^@tZjx&1u)cJJzR$)EG{? zOhO7L+t4{EJs0m@&}n>h79XF)hcsp*&3*9ven5K! z-&;l;MxF|9J$f&C6;4|HKA#{RQkUT!?+@s0Zg^LcN1?(#Z7P?cp`QEjb_JDD#AUeE z_n4Ls5ccys;HP4DNyE9%Hz2%4yAk0%+D!=W!j~ine?8;>i}Al9JgA^@Vc+2FaMoYo zkQ2Q!n7LE`4j)joTYaB@F^eI-1DAj(q&LV_@RE2;p#twtf}sZiQ(N)4N6}8lK_djB zef*=EwUg*}_6!pir1WoiV(SS%IKJbL+x)6R7YW3MjYAt(E*XE^rj<)h+kkV0>bpl) zFKyVkaTym2lt$k``F|E@ThuFQTAPpPo!TwaQq3~*I%;p zRebZL8rm&!?D6B1=bt_A3q^T=(l0t5y7r9APMfBvej=%M$(o^c?M=r|P?VjR{&H>0 zhc?jSxCrIniQnAitIt^GyvY9nx(KTDhHWc`hn9N(%{*ICZo2~Utt)`YU6+HhAE5in z6>BzbaX-815yV4f*`LH@HFS8xpJCYEcsumjTWf~4tTW{jS8lu%L1^vJn&Hm*^B`VN zB*3rDbo$ZmO^$Vn^5hM03d&ozet6vq%a4DAc(ezq*F-NP;rP&i#1XyPyqv>7%E1?90i*M5aID{iWtk5O9=cq9WVg-P+UmjAsGan+SA6U> zZZfrnJ?hLo){5eCpXtlou(Kp+J(LqIEvd-eeRZ2!;M9UCv$Hhd&a-LrTyCw@dH8OJ zy*%W0s{c%F+pvMuGuLH0pVU9K6DAhDw5j7SHt?_G*x37Yym^wu$6?d~FY}4`HQFYV zo-Ymq=S41ebpEMh^6$n$nVIr8kBQ%p!;0xMI83j3Z2o;XxWLG7I#=emXs?}^_4HDg@hx@wX0Zwh z)(RR2D%+=)=d}K{f@C@1Hh(T;S;e<}i|QvjZTWe5UTcXj+>$!85@P*Ab6ZVYr6;#G zUn{bZgnPXXyLP;g@T+t4Z0;OK-0Q?w#w}h?QM6254Oxd46S96HUIZw|p&8IUDn65r zuNc!u-uV;Cars?kBGYqZ6OI^n-GSy%s5u})r?Gd-G-!>M{3M4Cn$v_|(tVjm> z-NmJ@QdhD2y8qb-HG9GNGGl~ZKBiwS+P+iJAjOpa+B6edsu>!p$y)4DSyjZ^RJ1aj zWYrOFD~+SYq(!0tbu4`x+!Ox&!IIL{10`N(RKq7MuCP}WsSUnp)R+3&mbaULJvN(~ z!f&O)IpAecoW+Gfw&`{EIK@@rI(lQynBjOdIPOGO97k z*n}`CSh(Y(96O=!s^CK`Q3T8hZWXuhT^DC>-I%A z7u}T?s~i_keVF@_t+BDb8b&leJ$?wE19=q3*JC*$seZ%e+TIIm&;g(n|#-l;7vaEqy6Zqb$u>7)uCw*pT( zN#I|@D?G&A>hwyY+Q7f0;NhH(H_C})kxmcozP-G?&;z6!MLi3Q@^;~)5H#kgw|R%RuG6NBKcOp@mvZB~6ltm>Ct*4l23!^DjN{`l&oIrenGlmsr@s+2fHsP^ zXiq#nPUO;l&DvUEUt7mzrQ3;KnbwMa-S%B&q}Pl}eGso(h*DN$;Vs&Gr%5`^SAZ9S z4!p>KWtEP<5~i}GA9z9k4q8hTyaeq?%4))5zYmc5@Uv#7_wZI@9O&9ApGjoJBK z=7KT#cVm0wLfPXk^EG4Q_v3RZKR4p5$Hec$q4d8(d;|3Z&#$7bZ=RUp49%|s`^u>q zER6@kTCqL#sFVz}f#pl!xea>VCYpCeDZAMc5!|I&SEL4|SzM=CSEN4hy-b}n>k8}} zCm2|Boy5j8V2P*b*BkYE&50vocPr=M z#GXKI>S2<^qf2Pm7ijP~^Nk4{Q+K2(=FM#v13&C5jTAotxph@SZiD4spT$}hC@v|{ zr*6GHS$y0#Qy)P#v;#V@>AT2xSuXYyb$vWqKFPDbLBN}NRXPFgNd6xkVV^sa_X);} z)c*gfJ&6}2VbK^0-oq9GncXxOupt9WrnSx5;vvo9VdSVkow;pkN5|4_GsmA) zRahAE<`(CA+k?s&40`g4M~H~)eGzAV!x~%rlJjQFIB!Y2%avPRP!=S}c&RipMY z-%oECS5oYal;oEb!dz9aCuHiB&J|N-3)Om_NsdPgEZUwI3&s8E*pHW}=ZHuy5Q31dbkRRQ`8c4HDWocN-T-u|~HZ(;e_pre4Rrce7 zh0qk=hxL_*4Y07Hgd=fN8?UKMkW-fxpAc$Qr6w1GtJV99-(K)hL2W@$PN{Yac5%cSgQ||J&#~M;;EPsb7`ivXt_m9^|E-v01fvn_ezN>lQpq?xA!%p< z8*_#57$eP`n(O-*bZ}}s;QCUehHVNuB;KMuDlCokge7qop_u-TgdjP z^atgNU`Be9_Ko&}OdrSQGH8T6Z`Uw|a=sN=`C!vxM^sZV3E~qqJFyo>)+xzuTNUdP zmo>*?@no+Tr998`c|&R}-{$p(QhRgkw^`I8oqtiEFRT_+6$J{;%c(317M!PKF>q4B=C>ovpS;u38V0DDcHg{${{@|((l4YzeM^qqtYKFJT%(d z>4}8L=e!gTZ=1AD*5ouJxFF{nvXO{O1*Sk*(CaI6W;r!R^`&Xiu%Qgj=aSEgsdAkusx$xoM=P)zAHvVQ;Cwz>?ZpB;>fzYf~>OAwh3G z?kRs+Sy)vIg5GHmIW;wdGP&UF>C?9^NS@caaPs7ZEiDwbstd`4)J;kH?O8K*>grw! z*PypaTfe~4bUt%!j|wZwz|-t4@UO$JqkUA7p6p41e@(eJD}4sP7aw8LcMD!H?Uw0j zr3XBEXOH@lGMCF+gZlM+Sn1*8rqp|*@|&$=^6$pcyyxolm~)NzJD|z0M*e&7;L2>o zR~qqp{ri+_lwauaywVfxc@vNDGwmk}L*QS5)n?$$4x>EAqrNv#9$6X8A6}VKzPPqa zwuj>X!0~7=+t;F8Zm*t?=V^i8O-|AVp0tp_?^oW=>IYiu3H)F2@zckR^t7rcd|YEZ z^IPv%;JClf!KuAhyme;;;NiVXUGYta2F+!QkHyw<=Upml_io~$U* zPU541=e4eAC*x&1X|*f*fmXYtTBFIG5JX* zUL}1Bwq3(BDR#dzb{$*a-?I4~#TL2P+Qqubw3zktCyOrVj~}ycP%EI@)Zgv-?-KBC zzGX?F%_~wrEUBC_cl?xO9b#?QC_VxwM_u9ykiTVyYE6d6%o%iS4wVjieO_{u` zv*ZP$hODR5?V%=rMKV~DE;4F5i1j3TVf2d9#7jb&P@=T+PX;e`Y#*mNi!S?}qLRDD zA4KIH3z|L_E6-Kj{_j^L{P@2`)Gj2m-IR|pRf#>4L5wL9NHKZQ&IoUZw6%Gxw0}=q zB(1H^|7K`uFO^o-Ozrbeo}Q=;I6dWMun%9D|5!(~CD9ZvuPP`m&;L%*^+!FPnox^M_{|G$z!)snq z`T^3joPNcq^zSJ@LTTdJNj+qvlXWdN%V=vXsqaca1mg`R`N++~3$F;+s&A-y0wy-H z>iX9;ISRsFXJKqcTh-)P<@Ao$Ic@%)^?h9<(Ta$tFu$mEl5gC^K(xDI(t=5W!A-Vk zo!wvHiRK4G9_($@R^;Y}oBGCeosx(LJOOuQUV*pR9rFg_o;-VC{6JIJ!lXXCh#id` zf~Pn#%)py{;1S{D*joi2nrN5uf%34_RP}7lP(1d?`ME63MfK)RnJY$E&U3$Vfb*=x zXGLkpP1N<0^21DBMpe{eE$H&p=H_{83M-t!sJEb?W@=;8KwX(>QlZPPs_usVMCt<4QtHV) z8&7I<6?l`SC01*0VW7etF0mf)obi6G!+!V^Tg`-lc*E?@NVM~~@rhH$IqOSYCijFN9EUle zt##2kwsFU8n(nQt+7g>G-WQxae^S@{c3)Yfu&B7%nT+eEIW#r)tnvBRop9fA(>@3` z9-6S`Fws)mcu1?Z`zz0@`ik4OlR_ws-ot>jX(1oZR>cLG(o2eIPi!5hS#VewNdjY8 zqHs^rxrwoOf`eKie}uK-IpDvG)K~{)@YsX+fk-_e3rl+tzmuugjKUw3_G=)$A`6eb zgk?guzQ$)~AIw@q)AW$pX@S779;G$(BwuLY>^XC`4pjO*RpSP#$4zetXb)lnzWn%e z2M5nRUKP{P<;iK))zgx6M|94BugqrFihtl%?ZK0w44&3C0{=QbBTMTVk)E`rz`v%i z61kiVelMJD#qKSYPufzXe^a?2t9)8F2>dJ3tAg|>T5RACDj#R1ztF(%rYBtnJ(=|T zm1hk)-!+Xl(*IR`OYClf&UcOWl3flwdUFpPVnumt5U=OU#PhDRtnXd6&4}(y{BC-! z(}>sU$;9thUe2ORZ$~D6pD`~GzkX}b{VJRd^>R1!JR~nn`g1IjtjVmk#QtjJTZKz*O-Loxl+3dF?-N%2h|OhV+4F~l>_yzZP-&tj zT<;$J_^M`VB9uF%9#tSoSmP%jTkfulAMp%}rR!;_ z+CyQlqu}sEE~n}&(RwWoOGPk#_%Qs3$!n;H?dONt?|~N?DA|hL924!pi7MpOKwEUJ z9{52}U9F~to?6rIEy3dAaNcEkKcQMfmfxGkd7NR(HSSAt15RhadS1~lED`PDXdusi zcxQ=I1qrh)sk@ynro^5Xh`OAq0Nx@fDOFW$=-zKrUw4up1XINlzh92Y&Y{ozq#uhh6ZD~OXl$<6V$E!FdNLgfZfkj-svOH- zOYm$zn!CrATkH0hKH|Tl6dWzjbLOQU=u5rHlZ7Xw1|4O#@`yk6xmh&@z!9#6IrtV@ z3|*nT#vBoN;-J9qRZe3L3OsR;@D}YAekv}ov|lJv?}FbSNo#R>HMPOOA5m{zCm$YHbpLnVvdP~jTlx}WQ zJz+7IYL)JB1*sqAtFwqTscF7gfqHIU>P5Q4)@-@cnR1zf*xavjJDuu>7CEbO`OL46 zCBOd3Efsj;m%zWKoSTIw&ItT2bf@9tyktiLO}Rx~{IZJ>v#jDk>wl*L7D_b=QrfPzy)O zQzuh98}l4hmEM7^$3eUKs%r`y)39*tshc@IF|)32W@7xzI&fc5^ff3&OCce*Du2bt z4|M$V5^vFdw~%^;Q=2Z4*qHXaRd|U3c|O96QWU!gt<&?w(NE8!%+Yx_X7aAfN8VeM zpDR}yCI43@cBu=o_bAu1w}z`W{kv|8b0Jz4oTJug0-F4|zxcCfsOmSwPau!be< zG}B6>q&wJ~7X5{FPduc49`;V^7b- zZvG3mU^aV;3R=IEu@>##R)gCHKP}qLiv*Uq9F}RVg3Edh+G_^>7DZ@FL6_OU-=qAJ zWmMGpzES6WnI4AYy3WhH=LxxuYi*F*LfX}+IzlX2JJM6mqf4=zfsN;$Su=2bmfSun zX5)b|!VQ79^Hb5x#@o_tAx=FmokH8z9Yx?V7+Um)rp6AO&MlMM;L4rKe`L5~dP(Br z8spJZdz6DxLTChkWbo%MNFtAG(`N9eTv2roGxup}2FRDLr z!njk9(JUoeRa-F79COzXR9DrN+e?bv6@K5sj`u8CT-1Xecj=O>JzQ5zjBS ztZ?yxYQ0Ke9K*FcX)?vBvadHJRQuV-QQ9XBvnXPu|F}zB_ye?tE#F? z3yal+uJa=0+D4uQARbj^YwKnt#vfBxcg*<2j5?ZAFb>pD(08@0k;Syni(y^uik&PH zW7<5EU8s)lIAMI5r>L~ZZmY314aDMo_1?<-0=L`eZ(q=I?8YfQr)%M|aRFB}uK>nW z-Hb-mXR+j#gv-1&-f8E|f>suMLhZyStX?2zx8h?6QvU)EO~iQUwDn@_uOwQH)DKIW z86KcnrNm)ZZ2QjA?l@iQ7vy@uC~dp)5OV3WRHlz~PoATUZ!F*bZ?n`vJfFS%sI!!s z4`b=v>=7<@9uH{|$W0~uoBP@F$4FuaT(>8VMSNVhfdgjhCirv188tI?nk?AfXq{qW z2i`(Wt?J*~_V>#Z%|4+%ol)^Ln>b_l>LTt*-8^ z8`oEjQ&93tOUukEvhWt|aXE7wGIx`2D%To^Z>o6iy#p^a$&sMNFcRQ#Nc%yo zsYsWR7K(p|%2GR_T&8W-l}qZpfI8wnx*D=&>x`wY@}sF0!5>gUqXLghc~rcjj|$*n zVK5$+#cV0Zz|vTu)EFx(WVT|L>b4l?zS zb#OPAn8_O>>tIygTku|wZXLu-unzRt(}Z;}-mng0CRhi0>;^sLSERAWwx$VlGQnD6-8>Be|o@qBY4XZ(xeT!iw#Z0gnWa@jAOO4qvrom>A zsqf3;KWqkxy?fB01kddSHl_h9dQ!Pmuh;Z1N5tNuoTc2VkWOwSYT>IZC6UaFjPhj* zIa?|XuS|U+J=%Kz+Hh-_Jgts%*1EU(%_7d;7Q7){W@m2i#NoS6A zlpL4(A;%Tdy2R{gixu0-@{42R%0_v!l5EAua*$lQYw8hW&Ui*dz5rw-c<{akJxHN# z%2BoiOenQ@-r4$?)9|9pIf2xFxJ{MD)L&VzPyMH2K0Nc7lh2hCH3pr3JkRBA)8mdJ zOJ2}Pb2;gKT8rKe9a{#!AzEL_xsd0gXrINB&Qv-{42g}NWbn#5nu63fH8TwhjlT&(uSg9- zSkgs=wc;=?JRM$4S?AruBtzhxmWO`B zQ&8?Tqp#%rXWX-qSsI^gQs)oO@DrzsYB4RA6GT^GUZ|m63r}7?C9QCGDdCn-s3k1! zG~R~&!c-iOlwz6xfv6R zu&vl*GRQqAu)q5iQHb^^Y3@0J-N7EGw66aq3wy^yl=cFYYh#T}X|3A5=`*N*=wrI7 zC#&R}e{En*c>beO>X-&BYr>dv3urt_O9`7^q&#G_Kc-Rp@$yUomG>%2*+DBkF3+Of z@+`364tc+l&KC6F^CqQz4ip7hV+-tk)K-)}NxgzRsw72JI{c~>_>zLM?aDjcnrDzV z%+e zJ$ARxnp;urRv#sbWa&7bglrU1`mGa5wos>9%<>|zcVOd3c)%88ggW419L_v#21hV9 z^_{FX>@&vWZTwK}LNOpCYNaPqmU=1YZ_a98$$bm%bk<=1ipL>Y&zdJzQdXM!`^A58 z9)8dD#M~!bUSF8{cSh<(4fTPxQ@BT4l*`g|`~&$U6!ZA`e?W&AWvM0KMh7Gk^(l{Y zyN~cocwBhU`NCQL21GYHsK#_N=suNuY)=P1*Ui{#aA1vtJP=3wH|RZx4rGI$p5@7v zfZM3!^{R&z@*v)+eNqO2x zjnPK~m`0y?o|Aq(y+%!D<$4^C;WZrjq2Q>SgE)93r2FU^BMm+T7BB?Urv4H@g|{rY z+hVtP%ZdwjTk?OUa~+Pk1%bjPnyIRwpv=0&q=}7D7A#aiGm7bC%**(Nbb8_#9T>j4 zD=bPsa%MU5K5M3#Ju|;JH$OR0H)hH?>g01wXI_z~rRrPeAa&G?BqVU0oRKi|EF!8Q zJu8`$J6Zb0{OCk${ulG(snd_Cput~US5w`_5Zx8#MH(i2>-=~mHRvuU>xa{rnp>53 zX$^yqSt?or^IS|@hzDdA^|T6-25=gw$!?(3Si_vnSUjNx`Rb8L3{owR=sP52jhi4$M27tL=fWGg|k627I&3ryHjV)tHyCy`O? z^4BHePej~7UtUIIicT8StEv0i`lXj|S9+2uz}lLqCokW*%(v~@TDE<qe!hGrFu&(>SF2Tdz6l+MrJ$usEhY|I19#H%7c~a3I z(~RDWYm7%dw<@p8UR{1)x5 z0iEhW?GuCQ+l(DYk~!K5pj;O1zAV|I^@+gVy}&4gcLE5Doqz=_L$EUSoq$2!3CNBm zZr+OV6U9zI509To7})x}BfUq+vcx7}8$Ya;`&Gin5i{YDFReb4J)BNxnjY=w4urZx z#&8QxsIHsfaTqVGs`ji=mO81uv!ScbpXYD)8>6sg;XtXUHt6x_@2tYY!l7K?OM=Hc z_R(0c!gm*=o!)eUt8w-&##jxo+FjvLuPF4mf{xURLiP9Gst;Vj5?6uxMhUs@rots^ z@LSY_;6Kqsmg-Mgf=BBI(w(BW?m9lxTdzuNOaqps@g3z6UFuESkBGfl8CLcgvHyNV zER7@?YkC2FDeAMTVU_kB>6t(~o(W{0ipq_4?1GGb<>vp@GlA5Xz9=3F43i~K4@r-F zD4h=Ub);S-dL)*t@_&$3L^}__ z)9$HAeedg(dSZG}R%+Pfd4x@HTGDDF?S1tAQThc~;+c|RdEoUTRVl-BqleLc>@l>( zu>5d);L;|x>N1|LgtXGL2k=5{Qc5>Ta3^BZyhWA86;l(1mF0oPK(MPd(tdJduy=B) z-CSB&?I`Z3f@^u8sUk4BIW%eh_=%g3FQ_ZBnTjhys@3ar`$}^2+|dbj$^KYjxV*5o zINy|yWva*PDKE~=bys%OH4RqPP`d=Jqyc^_Ir%H*B;!rLkoY*}9N;n9x5HEQc_SZj zn(~3u{7$3Db+^RE2+MiSRo>=2D~T^go`?Aq2|+=oq&qs8F7#O@Em^WF*pd~g$&wZH zZkN_b8G0={{mxN%Jw4eW=W;taA2dt=R?HS0JljLQu1&l?k}DeMV{07yjzTYEGfZ}* ze10SzE79gavyf)2*Ll_vZ~a5c@9;&!lTT{vUDOnA@|(sT*V4MMv%R-S^sOya9IPq| zPpI;`3mx9LyS>3%6DW$Tqd~UNS6B$&PW8Ul6DNo2k4Z#Z;|_B{`-zjgmbQo8qN8mN zyo=p1sWLCu>vWY{9Tl;%%4kurz9Q8FAMvxpc<(ag^^%>5r>yE~d{*7T?^5z~BaUXl z%9ytq*xyp(X@eOXHUl5fCM^Z$<5d8sJ>TUh@N`B?o9Y8m%@Hc|1nJs}4Vaz+h?>vi zue6r8H>!{MqUdK?Bl_rfEE`Ycy&$mO*^9E$V=VJ3sSu;~$5}N#Zl+O7&!27B0V&4~ zsVlKuHnykSyv^L)sy^?FY#tfeL@OM6Iv>$IV8g1j&t8>c9o;!`VkaP?R+LC%YbVRx zt@yYG)_D?7wkY9YeIJ)eP2)gdcdR@b_6{C1txPLvE|KOu0AntW9e?w9BAk zr}XL+smVGNW!xgymEZ!~goGtO8{z#xsn62rsf@rs0LvgG+j16bV#1RR2|QZ`!bYXq zGx;*{x{WHz)v3tD>ozLK8}wx2bsJUWCrc%}e%(f;SR31@6l>9L<7edpOS)WO@5|yp zS*QYg_cVk5yh0!>EYxYlm7DQiKRu1=f(=SCGm)Q0NsrSAG^VMN6^*)-imQ)WjK*bypatrPw~LV zYO>HoCR`Iwc7}5Q(|vhyr8k&kdY~xkjON|1<%Bd3{OWGEsV7%!L65b_np#?^nOvpn z)j37>lBg{G061AAMjsdj|I#>0g`cS?U5rG$3Ch>w>^p`52+)R(norsjJkaRjL(g>L6+_b*Wd|^UGo-uGGES z=T*)UxJFKo5g(|f2Uw=DYfl_d6FhDOo<^v^-(mQjwgvtcK4r@U zjh9mQpsj`QF0CYP9?ho|FH4F2UyId>CeMUZ=`2Ug1EwbCnp7`r>mRLGnz(Z%jk(R@x zMdlbLGsq(>67o30!?XgpAYrrA*t!>Z+>-X|+c+2+^oU1jMa_|h&R}VAQ-6KSv2~;q zJDQ>ipHPY?p0Xxkvx%o>Ii=o-O+97daqh+w+to@^l~9B&x8#xLIcvuz{WfmSY6iKW)ExUi*9OZxp~{j%PodgfK(lVdXLA*#9=8zvBK3CpwowIaC(vr(cS<~-`kID$AG(7|$APqWG1q`M2Mv5&;IS_^ zU|PjgY&VS<>FUR%v#O_3IvQtK;;qVxAE5Fv-nOgv8}VC=c;w@FoQ0}hsLs+}0dL3g zPPe|kFXcCQvgU3)5wNsno!|Jc<0kkcQ|lWV3;b?RdBElN7kC>@WmVyjuPUToIDYo{ zx@qI4ba$2)ddrGF#m0YexYj$Nc2<2&ZL%>O?;vYR#kp&EXNb;=#QXVU*6c9m#ZxZz z*xG3g^@FvwgY^y5YEO*@1CdA|7}cuk2Wn~t>Kg{fjT>wTj~f>b*3>{=T;N3=KE4sx ztx!HaFZ1qCG@`HTaH6cZxwy1E2X5s4QuS^Mi*o#>z&w&enALp4h}31AW_3;uLA0u- zeW=OY21sX);m;%XM~?K5Qk&+TK6&!#^F&zN8Hse()=Z8>CfgSLdev#yEm&~fX{&y{ zU`xwsvyWTV(z5Ef*{8LT{G&biG|5gnH;1iiXfu}kHaYJ>@qFm7ESOeMT2>jn)aJEU zL~#6;`h~mIUg-_F79lyG{zPTrIjE+A-umpZ)Y42vvGI*j#QlI#r9Y0q{9%o&PxHXq z?hBOctr|BW;jFGI$}h9It31Wt$=$)8C9Pd;7uwgdIqUU0->se-Om>!pN{e#C?&46f zH&)s*)ZKRB#7pyt3UEz*ANEiuR&{vtA$6tPcM@Y7Z@6X#H%<1|IM4EOlYrDLXk3?9 z>L?x`47ODUn=0HDQr4LKir^wud4)xt&Ap(^pR@Da@~McxVcxqv)|Kz-OV6KSr^~dW1i` zFhA%aJz7*8aN3HZlWSwG0as~7batI?)!(qAg{#HM`jH?~ zTq*4o!$8-uBk{@j7N(O>NsyLY{QntuFBtw@!_8 zoZ5l^4LuEoep}Uqc>DiH+;;%RQCxq|%=O-@?)FY6>D1-BQ#m!;>b7NDmV3c~n_O^_ zalx2kI+)%<37BAN2!s*{gcw5ay#_)xhER+{??w0hW_C|H*)k#d{{QcjJ-4&7vv1$L znR)YO=FNKt)YTnOU)~ryDX%OLD9bCUtsR)zR0dV3ysB_n6X!v4?L3Zi8`Z#=>-RuE zK;aR*7juPX-h4RX`{?uJ?NvKfvs=(gU{axNVcpn6T2L#Y>X@h-_IPE_!rIuP)yt-N zRDp;usNmZCtWwEW^Qw3Wit<)cVJxmjIPac!&-2|L3YR| z2Iz(lx`Mh=okQ>S1{S!C-cTUu{@5SVX!C6WYg!-j4SeqA?l1W1kVZIE!N|w?DmdOC z7tF#OVmaVH%DjGoJ7_da%qw;~%s#!&uJ`)Nn)9j}g5E+;yY^70&*u!fU~9HBSQBt# zt6Q%L*nM$tQN84=EOR#+eGUg2D<#dKP7|!JWP3fnj}OSaP-ew4{~Kb2;`_t+;LVYQ z)$4O=3_7jH;Vp{z9e$U^tI;h9^m9Beb_Xp>3>u@lyvASFVl)~oen)_6)Mj7E?X`Gp zdY7hG?Q=T3LATe~V`i#ob-1jN3p`>McOz(n8-?ekt5Y7gGLq(uJk;n4mYt%0thDq~ zZ$zWL#uU^Gf>XWYB40^K$g6ki?z8GZ^!Gq?Ku%=7g(HT3E;XY~6mz=oRc>CL;tLAF zVd6c9_y5k5boiWZ9S-wDd!!&7as-^lkd--9PRrFUpU-tcfu+!%$`4oPS4}pUEzUfL zL#mSOzQHN?9N*>xT6|UGboZA~!ZL zEMJaz_ds!5aWL*s2g(XZ7Zo)Zj`GmLvf|dbBN%kJ{eCwufj1g8M{6PP1@tz3Ni@oh zZ$}G?^j2GS80lYntPZ!^Vf9=W_P7JEUL78R&ZvE#x2_hOOLA-HZ2|nj%5v#^DkpI1 z=s^NkwsNv-xz$O~+8p!_Cvb9d1*iM-oW`8iLO#wk(mpst)zP0I4|hOf##%qllU;|D zpOI3JloZy<8lexTqXSAxbEbhdiIfad+QnN9bR)Tg=@{5hrU5C5%qZx0{T?Y)c}qe{ ztH^^)nI`0^%%tHg>l)-yDS1+v3fS|z6)9LS6JM8*(n@YaiYA+a7TE`j1_Ea)_Mqu( z*lK=M#+lCMIMYtr2>Sv8XWEIBX4*oVa4Jp*r%13_h?EjK9?qH`24vt&$0H>{+u_Xf zE~Efw%2y60e5;i`h7@F?laW$Md!X$FI2>m>1t}>ymi7ube36giBO#@gW5{t~>j$K8oR2{tbY}*j z9eEGNDY*~k>?Dr27aHkVPM@Q|IIzyi^1K!Yo9hqyERLUl@8|8RBr~8YWRGIThE{~y ziL*j%^Px82+$8(lmGit9ZiUW9HK1JdcOpj$5DM^hxQD}t(2aG!LY&ri2?hs(_%TLdOtTex>3L&Wah$d`BmVZ(zS%!1!Ll;wx2DadJY@*Yfz3m4 zz-|V{-wW9CY*;KkX3Ghiu=_2;w!!c8j#L_&`^n=*EE?JpXw1@}ahzdnL><+LlR`RH zBX*7%X+!GckWEd3qOk87=iHJn8SYi;lVhqdE0UzRDx@kI#O%#DT^lJYiwv9-e{)!= z+cFCQH!Q=&3YyEy%L#udd{2YdFt0g2OrJC&T-p*JIEMqKYli^_JyG*?7gin0`2L-X z5A>q;Z-}5rHl6gG2_@KmtfAkfl)oSz_@N??oNDI2J5vFV1f)2&G*1DlbQKL4C*C z5z&MyAh67@&BbzM+y1enup00Q`4inVQV(1{So%+LO9nMQ@*-`f|5Dx=CV$@)Z;cnX z;D2#(%Zo)V#icDpMJ=VpEk(TN#^A838{7cPZ|Gr=#o`b{;(XsOT~|{q*P9RHHlm^l z>a<=w@p88fI;|T|gjWX$PKau6w>qeSs#UFKOt?*!2`beDt6rxMnp;(BJaF2qP9{Ik z(`BC<^twFxIdjo1ZrcXo||SD}rHlpDN_alQg*0`63Y?eS`nrAg8#4 z{6XHph2qJZ{I@v7dffQ^`Bs;o{w3n`hIkm5kehx*HW{>!WR75Ys(gN8Ur2<*Y|_ne z6l&>(I={tHpdDDL^PL>>n$4=QR(l@QV}82dYUi~V27~}0IK_$-j4NbiY+-JJzGw04 z23D$61rDoEM=#XIPqtfCV_`SXmuI*Bbidi`7n;`tbPPFxK8h93(Jw#?EmKR!WOm>l z8n+n(>BVEgS)T%d*7=%_M4wwJS@NXo`8?&vUzR5%WhW(@kfgRbh^#;bykbPQWlY({jsOQ|J(2pR~|s zp0uK_4(lH_f6&t~si^76X6<&({K6x#iE8Q|J74shsGynPrm zby%CYhcs}!1>$Key%lpV&+8Ch;-`YqpYaV|5NJE=YQdy+-2Rj((Ok7oJz+|O`AULn zS6-e=8z}ZMN6EN!DCDx0lvr3}D%Hp=B_%Qc6Vz5aY1Gnq!bgK{caZrKMN11tCkjoP zup3iVYkp;Pd1I?xZEtN{9;wK;Tfz=LH@87zJ_Y4(&ejFj4k*SH-GkOF(3L5X;-nCW z3A$sf5(rxvcsA_fxooAy7S@zZHp$pHvWS-kh7h-xj!Ua`kLO#g9^g|IS>DuQ$3fG^ z<&>*$6M(5QgX8-=SybSnMTs8E+s=#p=K_s+8^7L@ z=TQZD1ap_46J~zpb>jv!UhjZL(U7p1Ax2*gRG3l}rKY}pz_GjYJU_j!z+=zJ@X!ac zGMEdDRq?oR zb*IBG&PV}Iq~m60vMet zwc0L@0t|=x3)0qz5+j?vKe@_=S-@l1Q7UE!kXM1dmL2eupas%*iS+q!smRmkn|Rq= z;545h9m_Q77=}2pJT7-@SYHrD+u^ruG8i^n{f+|5qCTyrDCN1vYPFfKcIWvV&Xa6g zbxwUCG#nNrE8crBLW-W2hBZpTnJ4T83!7h_Yb-VnEaVly;NKz$LNgMf=Tbh7nK*8+ z;tiw(I1Ii+@sws%7|SjlW+|%LBjN@TXJ;fL!RPZ!Ul5lvUQ)m#e-ZG8K?8(H$cwik zQC0RWeBG7#k$#9ek8uqyg+r#iTuPFMQkgWxactUa`Ia*V(ww1lhOaE>GPtxZm!`|w zJJlAn20cMHidGQUcUWiU@!33O+=qn56AgORl|g#K#9pS-gdCnQeE*gXoY>n%SA@OZ zP{`}$YY8QpgP>LQ2=oV*U^fE-8kWkkE1hET1~xL~F&^^C&;d_w4?^N69l@-@&xCNt znRj+I-O7q^7!)#3T(KR&&dqO>v{=?ix8)agmKU|oPL;Q*^E@n=SK!l>Xo4=Ow%FIU zdD?<&j_%6uT014(Ge&X*wdPUfYiIPGHMeZ~`46mJw{1#c*W!WKg5^=K+GDB+OUVK! z4#UhU%G{ zsoU((jNi1RqM)&$c2Rq0zl-Wae!ElSZHQNns%UUUCik>^EEb<}%#w!G^cL?_+vp0Z z0e0wSSK6Irq2~1mruAC&m=z0BB`u?ik~3?o;?!qzh0NNpFOeS^UFXT4IBvAt>~~oB zu~}#=-_J_Wa7O(C^KcgMe~y--#XhEA6@7wv)SxM zeaK|?8GU)dSrZPfs$W0rb*E4A1lU5!_49+tCLBl87r5+tl}aPUK8%;o>zH^@rO)~E zr2!A8bU3qtdD&B-^T9w019F;k2e}Q(cyOZf#+DtvfeYxb<2b?ahBBTO$7Fhg;KCZJ z2BJ0Jjsx!zmBt*}^(*J6Y8(|aI!fkF1|2EO_vwQ+Ypkx^nIE){>P__a7$RfqZ0X|i z&MAdW^P;*)RHqBJ737t8gC$j-MF$SNu*xMlU5=yd1wI$`PV3N2h>T7<#`P!>kCUPb8xnFqOlRvS1!|x#4pKr? zpfag|V$aFlCvf}gCv<18P0OcLR7_dk)U<3$C9ad|mbNzcHK!}xTC*`~wS?`3O@;C1 zd|TLRjT)^6r@tDWfKKSnnyckty7~Lf%Q&26%^cqN>ILnM3!D6ga=S$hWMcKP;`%V^ zRbz3M8q63S)WnVX$KO%DxHwM@O~>3d2w74R_39QgCe(AhK$s&|HqW^cV(?D6O^T^6 z$~-hJ7Jcq^lg(_H87jjTm6cMJRy zHL;5SpHx{niLSI+?$KMyqK>FuZSa}Qc8%7p!D5jn7_!<8-4@$cIBv9R+q9n!z79~0 zm6HSr0R^?_rZb?2^Eb%CUNTlLa)gT;aA;r}W5#g#Y z%?_W>@u1!B@GWr$5AZ+j2^df8I>uIL_jtMLK+_RwkIU{*o!G7Nd%bx&H-MX$p1HypG!=`-rzDGCn)F(i4PVz zx%Un+YZ6!bA_qhr{P(9fhT@$6RFe1U6|iW_=e6?s%HZk2;$egD^?Z?phi);cEIEmgEjIr&U@&ilw zZzfBuGqMq07hDr$F6$Atb2!xoTOO7`uv^dfR}_{ez78_Exb-TT-*Fli)0UZAIy<+_ z>^%amjIE}DgVc;hCs&PK5Z9Y@Hci#aIR^Nt@2RYs98npJy{Ar}v9-7B$Z1BK{&A*a zQlzHQY&7ZgZB2AK05kx*3#|6y?Wr&+>T%P{XG!cz&m-5ZH`X}3?3 zeDuZkrHzeC+uN5lHZE%~tPO{23k&!!H4i#-$#GjU~HEl(2x^8Nf*=lkYavNLdagof4>=soKR;&uJE;Pcbm6$6GTgUpfeqX1%qgr~Z zvfg#L&VMn#KS{YiO6Q03++XPzc6r#1m9n_9QudnF2AiTt2q+03SZylruc{_3>>>6r zobXpTOd1~hW`oaQc8665v7cQBll@|*2iaP-HvpMWtej~1G=Mb@_!6XVfzw1WOB_RB zTPU}bzHC=i20hrC*;oAG;sw}pzRaYRJB4NMDCj7!=q!#V3ypo-BS^&E6#h5pGzB9&RclW;HWKKW)v0%8lDq;cH!zZdx+hSMHWw9{Uo>m$M6_7iS{_rw+O| zc4b>~^NOyiD{FE*AVZeno&%v}%k8G}y)=0Dee4ubpJ&4JCHIw%)l#egt$@E%Bhh3& zLGBHh_3P2U^PmC7b&TZ7D9=_6{kOj6=*p*f37>j@9>2_^*X`f+9KDJkw&8^uIPllF zq2q1Nl=+y=ht$Vwr-Q8d<@(W~bLR3%UPDzXF0CNOB{8A;=!140pQqrC%VnrZncK?a zwV0`K;b)@IDoY6Ux>v-af0X}TTiQFhz)@v2_zeC)ywsI)`@6bZoaRu#=x{rnAvp6+ zRXY!|cr5OKJH2dt>G)7EQs~FZysI)?knkflQ1~~==L?5@KIypZ_Q36dfL*82MZLiy zt4>3y&1Hoz42><%=ZhKj234z1A4^6`CU&-j$1bdQ*=>aYW7IjFkwnnj+};){ZpGFj z*P8!^jYS{LgI^AeG@L&x;-`zx&miQ98ip1ICkOn~$@iECfmXwkc*>7WReY4Bx^Q#4 z!lI4Z;kEsM4X+BXbsDoK5OCPI-@4+V1mVXHLo1uao5Rl!mznYsMRt3MuXy5^!n=e! zJEkhM!)gW`+FdtzFZ|o_aZoP=ygWp3%^t}@E~ZV2rXef zG_T6}-U?JygYq3Dxco&5gWtVuNjflb%le1>6RH8ZIGcFtxf{ zo&(5hGgynsVk*~s_Eu>n1=#2o_z$`30p}Yo|I^DoIB?7z@vP*ioi04_oX(^4VE%hl zZRO-qQX)j%`B+JU_Z}?D4E%BEIdzw}C%9W4DHzO?>S7ge`Xo<*OG>&br7^V*hpnv2 z$Nl)|gP6V2>$1XP~ zf%SPzCH}4Nf$6-C>f&rtc{Pts9zCFIVh{dWe*`kWa=SUda=Yno{zLBss9WrU3k&{y zWXE3MFUnn_^XzNkA47rgd0Kb1Xm4U$dEnKdcq_O3|Aw`it&;2R(9MR> zy6pISPBudSHnKMWXUG;BcuP$~HXHar3SNxEovy)?~R4kTE{!9GEVqE{zI6jFE8@0Yll$Oj8U^yE-sVQAopP$4e^b+ZO~JfLQIv;uQ=AQlVIbA$Q$ zL3Cyx?-0IxR1!*kG<}d+#R`L5?i@Y3wD9nNicma0tbkv78g*dW?i|?Zd~aYUPB-FI zMAT8>_8cAadildbqzt^{9Lk66IOYTmS2B(@u*qT<4C*S%3XhNil+ud0GKwg8e^?cb zR#rx%RoUy2IR(k^zq!5l|CrxZ5D%d zU^J-7Tk8z^73+oOv(j4AMZO}j#(>ZHG=lYWDQ|@ves$>qx`E5h#+cm{u+mO5{s!){ zcxM!QOitb7OJ3A@O#UK^{$3qiNd5PCMAmi!ltb@bHJgHa)ohwon%io~a0p`G6 z9476=MiKwqz%@F$&}_^f514v*Rsl@Cq1a<|4StNbyfLH;JYp_51lIcmhuBXM!0>Wv z>s;7>p-XfFr&}z3w^g#!xfZk8GVptAxM-;EKFoLITaTY}lE!aw7Z|mh)p@WdpgBgP zzjClDb1;x?=MDP`TH)Sh)@hU?Zyx%N+afsze$Nr0bFDad4U^o><;0hlIMsd9OU72MN zr-4EE1%vo7RJaYPU?cvh==aAkC*wYLq18h%_r*+#FL??K&}79{tE?x+CrrALa|~vn z+XE9Ow>ZpZwLakV=Ev&u16|!?>=v^|=Xd&|Qgyy~ZZXNvrTQZATr!HsO>{;e8gN*> zdBxFKZ3Js(-$f_K9+Z!HI^655BcuQ<2pgmzXr#6g7+l3ezW`TSkuJ& zpg4SZV_ZKp5VzOH9AhS>i!+P0~l zXlZ#0E5ue#7i@8h?FejWzh2p!O!ihbjEi6Dukv~0J{-=mMI!b~i@OrOg!j_;xQ41J z$>fwOo`XLSSw@x3Zfl=aR#9>lW&~yV`DJds%YIc!#S)kQs?zeZS?z7J%PNbnvb*&1 zQ(oUT=#xP|owqAcQm~#}bqS@;#KHq=?c{K}ARcqt?f#%UP;u3$(ifeBURH5?MNOteO$1VFURGD-8OaaZ+hxxC6lx2YL@iXs>7>QRP^)vVXNIt zExJ5g#QLd9M>pwIcT+VRZ}b`>IDX9L+3l1`KR=F*;;U6EwB!KJv_Fb@4)+0wtpVPW zNALou$|gUYS8#l>SS@7Z`Nq$dVA1jM$BM%2`=52;B9*xyFOXl5=a2pLRZ&=F`RU7W z5vUGj3t0ERgqNin^$YD2aIwL4%Y3Z(LfI#KRCHaeg!-PEaNM3mIRs62kb4NU-}L4(beRT6ENe!@J7I&8e)J zI7X_7HFU;Cg{4UCf}Vy*q-9BKd3$kjdwE%VNlCl0`{kiY^aw zagQfwh9i_WPkVVhToJKl7mk@aHoC$Pkql~Sdn7Eca=aQJTjYtS!eW9ju!&C-!a8GR zTxYWcFXHDKG+e_$)`dd3KZ_}TJ* zL5R7(iaZ#7GT&k3y$UDsqkKmkoC)J1m&dW4D9@#fy%=(!uY>W&M^i!9vj)S{aJ=E< zS7X4TR-ez<`S_(&AE5bGi_J{)gTaBf%{Gg5V5h~hMXi2GrFv1Vei7^otMQerg>F%u z1z!fY5!w6KNy7VACF?@^2IynnMr6-k<-)U9CF?=@H6mU1>D3~9dZkz+y-9TxPY)8N zW4B{XpL_F4vDx%x)e5BliSX6_o5+{K)+kTks5)ja{YLn);#NduIQ(4DpRg~W!)f)0 zWc#5wopQBm6HkxJmO}~I%B4a!hi+ic$a5yXR;2V!Sx(8hqcxFGM9L3KA4Nh^xF9UC zIlep?T=a&Vk{hv**Aqqztd_$hx4?F-GAA8^xK^=f&gKl+D@T5)A;W_7>s${W?HHBA znSK*6Rjn~(^93!2NfxVJe6a@e>RqaU0wFETBme+p z3@&HPL{(}V4IyVJNU6av$>?XZ9d?6psofv;7`>($CM_tDWHaf>>e$wz$^Xv zV7WyhX}e&To-lkoh<2S>NH>W5XeL}T8G3{F^r5?bc=qKZv22VQA=vQ$N9HMv7yR_W zJl^&)E8wM5;x<*}IZ-+deZlp{h~L~ZE^WTZsm*rmFXZ~Ukc6+pu_nC%t_k(N5L^Zi+H;eg847&r>6pxuoTCjI zlHjw)U(M0hj?B>b$^IF1OXfq~ws=YZL*Da8Z|Od)hf{Ag-I{rX_gvpzJ@7*223azt3`4Eo311xy%K%upT4#mjAqEqM7oZ#Z|h8 zHPFE^OS?^WxH;H72Mu3y?{)<(jT6bnlq zM#4glV&6e@67o~-k6aV)ky}&;>?=QUeh?`PN$kBGhCqoEUg_UZoiZdp$s9l#=x^oh zwn!GA^jaT#AN&;?0$hB8F$!6JinCqhGt~)v#Y-O1hRZplV6Z=c4EG8 z3ZF-1WgM}9C$mRBoPiSYK^do^F4mZIW>;6Sv{uws=M;;tY-cefoJfQVIB_oE@Mu=P z`Gjs&jT=$E`D9SOY5Z^H8=nIX!GA%%F+tsGZ}}$k3j2)qt1cay1ta?j0POHP~&Vh;* zUfK48IDCZh0J&3zR@~Q;m1GmV z#hwpuBe#(UA<4c)c9S2dkwWJPw!}JNd1*df4r?~Y(9`HUtQj65rn4*9jqGbxy{bdi ztD3J`r8-=7qUsLSBdX_BZ>T<0eW}W*jcTttuAZ-6uHK+NMtzq067}`!9qK34uc>#d zzt#{a(fKt6no*iYO{ZptX0hgO&7+#HwC&m{+5@!9wClCUXiwMPqf^5ea7b68YtbF2 zJ41J|?ke5Qx*fWQbua1O*6r4Pt*3gk-lw0UpQqod|BZgT{x$vk1|4{rWGFM#8d?pL z4RZ}E3>yqL8ul2zGgcVujUC2b<9y?C;|Alg#`}!V8Q(B|Xeu=wY+7mBWIDlgw&`-y z&8B-z&zjygeQf&HtTX$}1?ExaCi6J+Z1XbnCi5xgi_BM)pnv-Vm~vVLr9u}!kgu`RLn+m5uIVmsfq+xCrJYj@bA_6mEwy~EyXpKo7f zf7?;y=yuF>EO4xHY;v67INNcF<4MOWj&~iOI(~HOoGxd?S?)|b+nv*$i=FG8Cpgb{ zUhcfX`4{J-F3DBqs&#d^4t3q_dd&5bYo}|EyUcx{dxd+W`&joG?u*@5xo>vwa6jz+ z+Cx1SPry^;S>rjvbCTybo~t~!dhYi;=Xu@pzUOPN+UxShyh(4fcbs>+_h9cz?_u8K zyt}>M__RKU?-Jj&zEAyC{ziYNf2x0>f0h4m{{v8wuMM;YCI{vQmIO8g{u+2V@Lb^a z!0y1;L3PjtyVS{GOK@UvR&Y^pO>lGYq~JNhOM}-3|C(2jSCzLR@5YcRlp@&0nhCU2^85#&1!k%z6JSyB69v_|_J}A5*d{p?9@blp}!XHQ` z$t&ecbEQ7%GU;~dKI!L(KH`o`#1>S-M1*-}+6`WjfLBZ7pw-kI|*j9LY;oXIg6~0*bcH!>AuZu{L zsc2!*>Y}5IwiaDfbY;9wV|mj12uvC@}Hcb0x!`fV92vy`=zO)8sRwz#an?8vgS%Pudwx$Lg8 z@5_tI4=6vg{I>FMDkfJPUGYq1US)gbWtAUQxvLtg4zGH&>Z?&>M;$rp>QP@NMkVSK zt%>f$^u+u`Ut&#SQ{vdfsfqIvmnN=G+?co{ac| z&h*OkdFdVL=j#LY*EIMV<~6+6IHU2=rh=v^O|zT&n=Wa3xVf--LG#U{W228B{pOgN zV_qHe-qApr z=i!}qckby5cOBGqdDk1`Mvpsm+_~c(7;hWDV*KyMe>4nK;a@pjWlaHHx`{YL_XL=fY z`g+dk`9shBJ-d6poMM>LG-dge^QK%qG#Oa|Dtn5$B2I@k5BALjq z)RlP;;Rl585Vj(GEQi78yOGa_w09BSq5jPKxIV+c5C#LM6A!>eJx{*!_+XIaxT;f0 z+e`{n)xZmZy$u@bhJG2o z>UX4s2YjoQaeD4Aa9p|(9uj$n;ijI0w4VRnz%_8#7JOrWp&R&jXZ{5o`Sl}yC3~<( zz6TcIM1WuDJ@5j(C-WJ?$p|mYLF7R`%$D|mtM17>P2b^ZSeupe{}p|DPi|lhXj4j{ zR`3EIrt&Lns#-~qo+;ACL!XJKy@9YtbqQ(Z0qsE!hi3jh7|*8h>zvG+s%1I(c-mT& z`~PKFiuSml@2Pkn-vFHCP(p7dtvSI$Z_Rv!`zC}0;?F{R)&FY)yA-GBe+%pU|0aMx zw~i3%Mo1ee-@i{wp-+>21O0&5e(g6XX8d~x)geSu&q6pHZTSj>2T|Td@Pu#>;PwzV zTMC<{C*j&a(kuWvPr{hnLRwfUR%MHjR*1BEXycuL_gav572=Cw!}Jcsj|9CBM*4xU zMS3FAdHnxVJ{BTw)gad}!NV^Zu`aqvPw6$|V|8sG@i1Z@3#dmMTvqzEkC~)9-@bEtFKS#QoxJ0-K z+FdVUJ>vu7QgNR6FzWMg^qWy6tTH3tZqUF8c&8%H^L|HKD6BEiXD}{EsIP6f9!pBp zZ-AF@8A^qB*P{&WWDFkzDeq_NGvBjR<_96aoSpduebNL?Tbmfq-i9a21919K1laE^ zMSeWz;}9QnK&upNdP=Mf=_^s*uW^s-V6Y%=!TQTqe2)j7uLEfzgzFIwK{%4cRcKGJ znrCBIBJT?%pY6mIUdUjfBgGcror6fPDuFhA8NRiYpNbn6ph3p9$Oc16P8@3~qy_=Y zzI^>h|e?_=Hl0( z>e?xsa9Tg??eEyt%3z&pNp})x2 z;JHRFA#VKgwJ0}H>%PJDF>);C9OF?Jqj1V>3oLb>jh*4k=oR!9`V#$seoVh$d8`s= z(b8-#TgU#2lfL(|r`Ve+waTorshp}jRiUaxRi#R*YE}112FWJ*q%c+qillOBl$4f6 zOVgzTq!rSk(vi|h(%I7a(nZqmr0=7qs67hPTu~_+i^ijgXmzwP+7VqF-5fnNdPVf= z=A5?_o0*h1^t1Q!H;*gBzLiUU~T$UD0pg75(i3Bpo*(1m6GTs zv*eUQQdBCG%A_i(PHL3eq{;*9#Ep=eYoNrNcKhUl>< z$yL#7QIZ>?e?m$A8ofKGBnO}*8kEF=k^}}zvIHf8@n#0)K>7g6Lo%bt(*y7DnEW@; zBwsVJfwwczbjhakYvvD`>oVIjzs+2n`Az1$%-I9HAO73FcVoNoH9~eNkP&n4I&@d< zuGl*}-?<+DPJZWkLf(1ho!5A-ckU9ubKjZ!PRBdz-)Y(T^v-RB>^uYEP=w7pSMTiJ ziGKaYt6DC55%LJaE_w)m3ymOR#2s)w-V`IPDt)n_orZ^w#fR2}2FaeV>5ud3ft?^3@%^nNy` z<~I9?`tNF#1uLR&WYguV`Z=U3K}mT+zM|bUumf2OPOoog3)w+zGV8&q=Tl($u#?q5 zGrJD$>MV9TJCilB1t@ozNTdk8stX#OJ%n48TSiup4dgJa`2L>!fm{zu2lv7+_1k18 z`6qcFR?VlfHa3l20KdAAlCPd^Kl|3fmYT;4k9zj95N4@%EyuO z;IHl!awR#7oeKW*EcnQi80kMC*U?+4j(h{H0V~x|Gqq6|PoOaxrVC(WUdI9?9-{^hx0s00w&s)Sxjo?z>Vh(N+e6@i3NFij2D0PFM=aDiR zhJ0EL-d_MYsfu*aI`GLt(hVLzo;Hz5bSz}_F=Q(3AX8`?Sqw|+3+YTUi%uYm=qxfH z7WQY*8DuHe^Ow^D!DUy{1!N7Ru2pm)>8DG`db*q(4oU1tx|VFFYv^inJZvRzgG_M} zd?8#y&nDN>OUVs%JGqfwN&ZN$Ah*%$$epl-`4*$))rhaudCZ+(NG*Ptp6ytMqB|SI99BVH`ah^5r%35^@e?)lcZV@Dlzp zJg0rao?uV2&)JjU_|L$i{<0E1_Gfl8bSh%h zOUkJPNiIT4X^6DbT1a^{q?#6y$&hfT(N0KZ?c@O3L)JoCTSu3X)sUDsf8^ zb>vuhML2@?ldI^(S-Bipyi~FmXL$#G;$!k zFPuwHCg;|Msr!TJRnY38I` z)Ms+0W+YpmI5ahIC&wAlqoi^iCneFIr9eNa@3MD%ptVi?L6(o-BkV zjwMLz#XJ4z7kx=GQ4$w zvk6k%THG%qtMIioh;0&2M~(3PE<9O{T&sY4l>)gGQfw6OaPGbwv1x$09$_7LeV@R& z3*THLu5T7h|Lg{M`+o zujfz?#@je}WQHuEUg|?DE~WwU5yqQ58iI3CiAE@IAAJ~AW3+%gMhj^XMy6s~fcUW~|1;BU>~qN6d+JdN?@8F02A zXe)V^w!s@p2lj5eFb5e&$AgPcAfsqEM!rcH`+DdU?CMUX(=d-o(CHY%X3&{*7M+dp zeJ(WZ4!|gXAVwxg{@@M=V_bTkE~Ja-V!8z0ua=T!bQyMpp<6}|fh4dBUQE`|e!3RE zOV+_R$p&~VISia=6FBAJ7-5efx1eVKh0(i~9*ObxXvp`+&||@sV6z)53GadH|42`w zCu2O`N>9Zu^XcGkXV5e0S&*L3fqmffNIkgI`50whrWateybwI0XExm+Z3OVF5 zA&G3qeBnxZ6}=iGO9RH9YayFl2j1L-Q}!QYG`x%6K>tW@q&GnpzXcriPp~w2D|}FQ zkQMaL^mcj&{R_R5{uLa2CA|wW&EM!fupoUO#^d|RA&?6mgoN-gM(9V-_qKyycA{Mk zP=c}gG5R?8<~VToC+Sm^(x>S&;QQk-vOGth$GE$Wz9=N5ztdMR2CS$5ps!&>eI1g< zoAfREHr+|z0gs)4`RDiaUHTsVCuRyCKzjI)euNpr$7sE;kPUPXydE4zKOvjwr}Q(d z7;MB`;!FA!{hEG*IsIh%9sQpEKzitp@CP)7{tSOD87P}k29E+*t9Y06G7ZkvYjI*i z58p6GW`dVF3$tR)<8WpNzu3$i%t?-9F8J~Fzy_=j-h7W@0TyI=%k%2@^Z6J*R!z&U4Pw5Y;(k$@bSV%208q|;jRPgX~Mf`nQR zDYlU{kyFWR=o5^_xG@$oZ7U?tcF1;HSO@DQ$FeSR92-ZD!ALU!W6eZJq+Hsa0_k!p zB-H5`b1?Fdxn!Oghvu?*f{010#nE;XKLaxOSs3f?CSQ@S$u}7H&SmF8F8&Q< z>kHXM*yp;K{T3(f9MDJOQu+-T_pbs!z7jeZr(pi5gkrQgPW2T6H5 zc&7=cWDX(=AQ89V#LC0u&+uY<2l86+D=H zh|{`Pk_XsT>}qxmIgwpU{zi1{_v||IE&Bt`F5Uncic3^Cvs)ll-AYb^-sy#qsBR}# z@+mw^-3cCjHe~S#xq$oz8Y92QIC3>sOz(gsz74Y1o#YboTXGG#7$^Jmm~s7;?SMz7 zyV>6`$Gw-`$L@z$rw7?Xm~X*RaO1lfBJ$vUgzF^oNTvN)mhF=rO_73^JRovsfcI94queDyeZx#7d2?b7T_A1~dQi+tVb@jr<>-yK| zTKkvvujyN9Y+biv&9a4yH*VfKO?I}6 zbkt0%Zi0gF1Q|i?gbgcJE$K7y29T&}O;jh6rtaY%OeCvCx$v2^c9N((?W7#SRtv%+ zSDShgiX!7Z_=gzVHUbS$eAO^_Zshg79PHL66YsI$`=xL@Y>sKvYzdWaB z)F_Z^hMgGW&&2?erX=bxPr(mppy1N=_veLe(jt z>J$)lN_vBgNV2|5Jp;{CJ0k~;Y~Hk7pj1*JQ=L+Xx=!uP+&Ze}b(Bb39c_6pF>PRq=FX`QUo&CI2Bx>Yw*Zt9wuoC#@XiUu!#I9;!uEy!b< zJwy(}>?JGu*7dDlv0giS*}8>?^_k`lJ(S<8SE{?KMmtwzk(0ZW+Nf4Y*Hy2bC%_x$ z?bXK{Qrd-s1VR0a{tE_^mI1XXjkrw^3hAOhpxrCqZB_b5YnEdx@3l6_@3)Hnfrh8R zY0X|0!Y8VA3l$^Q8cd6aif>#zSP<=!!3L1*Oe%;b6oe8oLrS*S zY5PPuwSBp{m0WE~vD+2&+Z8n16=>~pu~Ug!W8dI+6$&XOZOfK4t?SFJ_M`$bS*zMdpwyv*l&8mfKmaJH;T_q|}yDEq7)pFsJ)$Qt4D4~o=wV)K*mV#?_mbX@G*T{Wi z&BC?)>o=_HU%R|dyCw%*jZFKb%$>odvs_z-m`aOo0LWQZ)}ZP`2kg%!l+sfMxwe8t zhtdK%6p$SXNV$E1TPxLGooz8ni)!!E^bc{64y7oaS!hbG4yE07CN2H@auBrL{@kVn z6ZwKm)rxO{Teq3|_um|-cq>qM!}^w4z~PcXSzYV zNp6XohR9&pG{~hl$y|E#&_lVx>XmBls?~1J<Y9n?kCa=OZzvjQ(`L)Q}V4}akxk_ zu7_m4MvnHaShjqFoN>(x1t7-?lAIXza20gLRrC~G1=hHV+QC)S3a(=8!BvzFS223x z+92b+X5;F00%sB7IP-{tGmpwR%Q23#$j5OO`DC0$RK{6k6gcykjI%QQC)z}tNF>^1 z29S`4u~Z_h;G9-)PAfR46`a!w&gljNs4Kp++$^Xn$Jd*d0y8CU7APqThYISPc|^xFA?myy>&+3$jMcU7q0K)AONbvxOBy` zjd%wVn+#R%A{cHH+Vy>l``0WH$;o83p>O>LF!c?6*%YP3ZMBxg{j2)dtVTIEEL^u) zy`+E5GIi(1b^S6xnR1ZVJJhRItXrrgDrIUD{1gxCv}^m;<9+3^(o)()&%=nK6jc@q zQVPq!wN~G^dhLeIcvD6*B_oweh$~1_D@K~?M4gxp@IuS+L{54`PP|%)r*iW1cs4(e zXY=#8Xu`OP>c%x~5Z!Ze|LR2s(Gz)$~&GB zVSa*z$S>#@&?Z{TkQ1wE8}Pe{ZN~4>43PN>8h)>0H)Cb&PtXNbVHIdM->pF)+y<=B z(rE);YOei(CH(uL<)+4(&j(n;!g>$Z#Tee=5C|abfc=G+*)FW@pf4;f6KX%g|ys1N)pd`pk875jEtU`Med4ZHIqHXnC#>>%XeZ}C5dE5GBGywn^DXwFNjUBzK>|^ z_jsETUtoNRFx!X5_YucY8T=k^swdR=gYgGKt*4p(i1-fU4*Wi2yq-|U0)bn->3!4t zX0PpJbF?&O(_8MbrAlL_owoTFVl`Xpt!0+;Oz&G>HJ4esO?%8W)=8!>Ec4A{%;Qbp znFcIYOUOLUJl}GiHFVkB7{a^@! zoJWC%GK(h)6^CP#c_p(GRCzi3t<3wV*BzM;(5rI(GSF$^R?%_R6*%Ib3{x-Q%3oJB zVt9%0^lQ-NoxMvZ1H{{z9MHUu1m?=qT_e8*s7g)bzANHftL|6-<$fC_{W<$0+VXdq zA2MIb_j`X!&WAjE5Oxl}^KIta%vR*J@s#X4oW8t8h%0c$8#3luK1^V3WAG2H?DOF@ zGWck($o}ZESGK)wMXjM!9Cz%D5cHq#Kyi-ak6Dc6S2Eur?WfGo@FBu?N^|~D>fM=# z@KkAqnQfV8GP`mzC{aNCP{8Eh`G&*KaGj{ZmXv`-3-PB1HoP_Jvk(!tA)lZl$Bd|D z3=s#<$mZg2<<_4JZ2!OeQf%Lp+_zDe+Yp`+Uj)`D_d^+sO5lUWoKo;J)O(N^yDy#V1`yt{kw-5*g4;BzCqJJ?r0P$~yNGfr@%#J`+GL{+E(KE7i#L5=^2o+IeWf8{orTTX?S zi29bF4hOdH-AH|C-&}Ia-VpbBm&-7F{h`go#wA=a z2N2S(ki|97UgrOfm(Jk5Mp`ODuKfOxkl%M>w>dX;NUHy(d4NT#m&QmP(i*8->Xl|m z^QA@7a@=i@jsP^>usH&sOZdcGB2Ua0U@-_YiV88KNMIkHNIeREb+|Sov>_}&7>|&B zHUn|Y07UvyQUpEiX6R{;mwNDT2K2QTNK2uyy-He#GH;Pi1Xa$FE|hjl+oUU{>!h2c zKTA7slKm0sDd`32C+St`E$KZx{Y?4>bD~;s3rfGo_()j||E1r9)D-*S_@zPAAf=b_ z`5XNbeS!bJiWyH$W*gqzhItp>#CadgKq%@?yvvYh55Gg&mzihiH+YZHIYfuM9iUJ# z<}VyN1N0B*OFYKYWXQ<-dE;OByb)T(m^VVd3G+s19c_ee&vHI*ga#1ijnLr1ypi6|XN>eUK4XNneasl) zgAFrA`Zk|2(p`MU2rU^Ec^j5}Q758|xC2hX%r1eoa))y`97`22e~5mkV6SF9ICC-$ zXQF1}?8_|ZRH#wDUC^Wfjd0~9M2dXJ=?-clMS=eoD6SXZ$fkV5?^q9NLqmgzjYeoi z;FvAM*O8jpiF0dT@q8Q~)|$B*H07~ooVXc{6Io+3pCk2MHYxK2o}hl%DF|B;PD3~k zfuE@mbpvR-0quuDXl;O|8@!XD?E$o}SX<^tKzs}Ma=e-l(2Chuge-n&9Rf!KZ07DE z;}A|j_ygJq&en?h{w}i%G}r+e?1Fw9fjW@)M$V&$tcnKNsKx+shT!zzMW65~b&l9zuJop~MZUo3?LDaYiKM@>7L(n5w5!?ttgeXEBp#tFu1i;9vM?gvQ&O_)x z;OX6j2EIZ!Fys0L0=PopL)>d|{RGc@5oRIGM_7cg9AOQ@1_b{8u?Qz4oPogMUJSgz zcLrf%9peA6_9pOg6vzJf^vv$ep1QRwt!}NftNXsK+m>W3<4Zn-@g;m4+nB=##{(xg zKrkVM5MDxfffooNU@k+UEN(>U0vN1!XT#dP)93bL174}5GV|R!VoCrA8I8i41vNBs0+=&4{8sg_7G|h zq4v=I_yGkWP!IwIq37`fDj2^LPQiC5YIX&FVEf5V!w>v|L$HwG!zpTZCw@v4N9YjJ zgW?b<4uRqjQT#pp+>W2S@I!U%!_R*FJc}Qy<0bqgolt%q%4R2H`~9h58iD9>^Dt*Y^L`b zd#@XB3< z6qndnf!3$6e*Fx$BAmDv@e72%#6KtQN&FAO*YM8?AFAIWJPZ#jCnVbu z*#C~=Uo!53{1J5?$E^SZcbDHq_&)xn;zmdkR)tiN3Tcui(iDN)A!G|IGHrqFxWm{X zI^f&h2R~F7{7FY~6JUv00(d!Yj=I3LF@)>kJ?eq)=-CK&h#e^5X53p0fTQ07e2cgR z_#cWN;)d^o@X*YLH|EQb-@nBy3*n~8D@qP{eh7ZvhvB2H!(Z}`N**}ggq6G*eufw& z@Bk$alE&o5Z{ncA9P~2>laxICOF_@lH(tp#izXN>k!oM7)N9o0_ zx_%PGv|RCA0Im($FVeiv>qr8BQHnAe%iYn^$mGp}{#wa&cOnb$h=npV*xuyNCP z5At}=;_>cczPp(3+01vrBR#}?&t<-QneQ6--i~{oG~V->`yS?g0dv2QxnIQGFJ|t0 zCyja<>0ut}CCvR&=6-}ny2>Lx$|JpuM|wGrbc6e&g8ReG{gEU7ocUXNcQEs~qGUpf znh@lu2|=pHahdx#9i*#?gQPbhNKX@j{5%PQT#e(*QXpeZ93-s?LEf4Wq^=1;_L>kV zHSa{g6M{@OAxP$U4ikceH=(08^e)Q#1an-;{J??=I~Ap{pKygbetaAov!H;- zf+#lNuwo??j}4?)=>j!KSB#$yN=yF*I^wuMz=~yD?g_D!x6H;h5hTl)3hnh6ZhD)P{fy%;a=&+&nuLxWH8$yUfO|vY`z&w8e(D+0Z361X|2euC;MD*w9Tj zbgK>BVMBXtXs-=DVnK!Z;AshdDhlfmwh$)*2W;qtlOakM_!Xyn)sB7Lf`W+1Mr=?I zyoGdcFAL)|t+w8--2*tmr@bSg^!b&e%=-Z%y8Y}{rWy1<4mvY~iw zR{(cCxAGbr`nH|!dp2~t4c%o!&x8LjG4C->3o$|qS7iMbxI^3vM*$rtC_4pmH;R@66gBggDQK|_Q+fb7YVO+>q%8^~eIqX=*bug}X zQo2D4mo1?!$U6@|mt@DGnUip%cI+}6T4h5UY-k%p@8J_HuPThM`k-d zwQwQDhIAXs$8HJo1w4k7Dm;&%P=yWE*-(oOK{m;B3G!+OaF=pkNK6x&ZbOhGCXPlT zrz=E`5XL0J6&5bE#)gO%Sso}dq3e+LYR*Az*@o`4 zAxKLz7Py>+obFy5_mB;}Z$r=7&|^0Av<(4gmJ-LsbG&G$`;849v>{Mx<~?TP-m#%L z1s_|u{3;6yeQrTyTaYax3xf$|pyWcfE`%&7AGmz{g})EsX9hzJHiVK*9L2WTu@dUG zW2aeAepSv{`ILX%WQgq%C*l@b>DWrK4u74+HLT0I3-IxrcM)#3p$ly28XF?qMVxM@ zjl05z2zNin#&Pky*W2m7ZA0I)q1$bUXpyDtvvK=v=vfk&|7`EuiFrCxor) z2!|gkjPerhBMXx0^0M%E8pjiq&(M58YZxlAp$b5CjBBxRl&;gx(a+HINwKr+*u@O3 zuwyp@I*)Mq8To2{DDUI^q`ZSRw8MsW+0dmn^q36+mv<-UjpO1uuC~)%XG1sI(9JgV zeG4L5%<}HFaCr||P=N>6P0OgZ0*4I|E`@Oc8;6fPnCXfwC@)Um(^h$T&)Lw6HuM_{ zIwfD;uwwI$+0Z*S^u7&!4#)-GCdDOYej)yjFvqI^H86A*pd}2o*-*C)QS3C1oiPbF zhj9xh;np$kd%%60q0KgQfel?`LpyDVa$LbVu9<|po^iKN!rf)#B=p<-L;24m#qsV{f$~%CU!Y?6q-^*wB+U zbijt*2J{x^_>~R4YD2Hv5T$#M(pmo%d=$636nqNmIU-?v(1|;Z-(~)-;WiEu& zlY5d=L0mkgS`*hwNn@0nAS}rkQwaMh#OaTwsQ!iiCH}Mg>j2I4&-1VHZvfQqpYC7d z--tBlPOL!q^28Xzug2iPsysF^1>q|b^AY}jViUr1Ied9y4dAcFUO;#`z5qpO1`xhH zv5xRuZ3I2vZzO5H>ls5ymkK=kp=$mmEu{7`Hhp5&lv8jAL;gfn4A%LU3F3 z(QtfuZyta)^-8AXXEbKSEexMQBT#IlK0hHgU<_JW8(+%MiPoOK6}LDgR65s6p-Lz&CISH#jN)mpE=AjH89| z)QSdfMFZE>z-8uh`3kqFgj>`=;{hqDMXMbp2o=XnghiabfkqMD=OEthVt(%88oxzz z3K(j|F3$T+PETjw0-gdy^E#SqVjZVn$Fb`OKk=UgPy8odD-byELCMNf)H>xFZrwGM z9x&)fNMrsfM<2teOSy*U^B6hMjcIa7WM*Zv#k;Vn#0 z2Gcprl3^d?v$+>?xn0>@%jJ|y&F8$?)E78ipmsriQ^@$6n4hkmr#4v1sddoxPwv}Vgo9%Ksd;98b4zJl=;jIUsx)J|-~IRAvx zya3s)lGELx+XekodRCQ z5?oKKo5(aVSmO|@A@nk)l-3$xC~qUJWulQ*H?c7@-xkKy(n=}}&fCR3-bI>#=%QArE4Z|y@bnO35$7%CwnVvvDlVZ^Qpj+a zTVa?OYDFoRU&`e~S;rXX*dVv4lv`BFEn1{q2A+>=|Akz&oW7LSZV@Hg@Kz7amwv|g zGY$P*<7piB(`t@bYD+(tR?2$Ie`qx0;J1U4|ASU@QOY!ISFye*moffMYL~K&b%t$J z8d6d?N^}BG`o^8w%?zV_qG zoHhV0Sh)cYHl<~FW;wRoTqS-M>`jU!W1;sK4+8E!)m=L_U;3vp%7vsAa_A>u}$NaxYa+`be zBIfY#xHnnq;;YNln-?)R!c0||C{+H=dH=5MBberb@`=Q7Y2?jIX(;dCx!?cJCBMlj z-{d}hlT$vR%|V(sIsKcQ<^lC9pm`hKg9e`a`lsBk2=hF`+;%bdBFxogOjQ-Pu8J`Z zj?HI2S1}K(xYZGvi)UPfdndv&WRy9)Lk*+e2&KfESm3b8<}!D3xR-0Wl55||{VQk3 zN8HAbxcpI0Q_X22oF>BKDne=0L2g})X~<&Q=LF;3wyOT?(WSCx*Ci-cm$mSAuDY!Kzb}{C9j-Adhc3Z%2 zVHo>Rz%>llaIW2WEepB$Cf`BEALKL@4AP2CpzKLvkq@a-GtHlef*=>X&g4aX6pD#T>5S za1DnWIXsWU9dv6Ouhr2`lqeS3O!GH+7*U(ZBNV>{>5}xj!~ET*==?W&hJV+XzpG6A z67z59d>kJ#e`lG$1LkkMkd1qWG<&H$6|;MW3C+V>d>a*a-{OHnaS5(u7m91A&O>{61!z2#VID}_3@LmqdmLcFZMYka2YijJNj=ywh;pDMOF$DA97%_AM0e;$8gz=t)@tuUhrBBcW+6C<(_i!8NCM~rn z-YSWqJssJHh!@m4%9RT~UV&E}B~IXdVITY+3q=GkQ#N1}cH?!)8F)i-Aw2t3%-O)? zH{0l)Hua4!dYKJ5Hp8=TNSyJ1l}msw^f7v_kHapM^8&m8g}5F+H`C~WXB6FzpmCIf z?_5xu9C#V;S-ks5Ym|yNIY;4%eJR?Og_k(tIZC0Z5OsjemT}o7Tn61h=F-IHcsr3w zQ{mw?N^$r{@?M3X+XobiKOtPNtfEkzMj@x$#u)hHA!jzFfj=0*XLHz&_Nee8TaNT9 zUf;V91=7uBCrWpjFXz#_dHiY~(m3oik19^YKdmHD9A=(?#~s~a-hfw6&*hhz3kAM6 zD*j40ui+s{UV#exrD4y7dQ{Z&C~iVf87W*w0N<4$TK{c&sp)jQ0S)sxVc-`uRQyFJpNO?5RW;HowafG;5nNwSWbo2TyVh}E-D@6Ih4Ag zX#<0wkALO6igE<1gLr;u&g_w=@-rG0!vU4o3|mj z@j^wkZ^Yk$*(0?LUwU^@mZI#(kFsI}68O@jtyr>qcH!?eD>sd8X=@zmK>QBeZ}hC$ zHg?`sUq2Uvx#wNPFI+ozJ}t)B1Rp8EZ`az57q0osdz;ch3+lV`{}woz6y-h|nl0;qaBXpdDUZ|NCF?d_u;V9h-vo94ezeE?9G1ATmUq23 zE6QCyNWtTq#&&GcT*S*CQT>IR$2P5A_m{6>Q9p1E-1~Bf_Q3=0nt&0-Y7&jc(I&-+v!b>MGcv$8R4Sh=JevA2(&b(Hibn7uu&1wGt0ta zZ#W#A6>c>g;#Sq+jGg8b|7AF0W2!^_PB=V%bvUeEtdIW)kMODso#S6|;%l#f(dEUc zX^r}%62h{86@Wi~8;!Jz7XC)Gpf9oFL#$DuwH5`%(75W+Jszk^!mr^w#~%C^KDXgj zW1roXB&K=Pua5814Ugf_)f-(-Lm%IuIoK_?Sy&{1lHn_F3{N2Da*T=TsGE zW*ZKdVYoAkV%;>tA8n`zM)EV9nxXD;ojBNaObXT6h!3XtbeG%d5BkOR81@cxbf1X} z0XcXi1CmPMOXm2>G)*%8*m0D;H66jnra|bb8xfy{1EK1ioa&GnmXw6DOG|T3(SXS_ z^Oo$&P^dCnhJ~fsAH_B}%vO$@!@ z(#04<-*EX-V>kXTMf@@_ew7+U2{E15c$KBS?@n!$wXMzkSc&CEzrA$KE?xGjy&FlcXVp}x`>{Q#mwQ3q1cmp?0MJFtfu|~@Uj6H@-Bdp7psg21Tc!D5e$(ekd&|DiJ;b& zKro;iAg`D{%H0iIK;7U{oz7&VVr5(JIrHo4=AYBsc2w~SprYu4psEhXR?Q^7nVa}7zV={%Dq zo+O{dABF-)lCNQc9(jJw#=k7E(pq@CoT$q9lgcB>C(Ap8@J%!cF>Z$Yw>ZKYrOx+|d8Qwfp+_iIc>~rdvPnVqAoY#HUz&pCG?r&dt z>9iHMESEhozH#Jh3%j-~YH*ETj2A*ENnNzQtXiG?KsL$Y0Y2wTmlmf{bFWbP%_By9tHL-gV{v8Ec zg{;*Rm(!+02*c)tqcES?_Y3D0U1wlbP8>;7x!X~$ zm5)|nJi>zb#QaX}@Jab!6V=KCRy!OtUa8)A`A6|Flgq7m?N2AgzoG0@&bQ-f1(tNb zsa%`D2di8|eFdW`&Yw8`ReU0V@zf3;o3b2khh09c%o6{a@^}LOX=Rpp(90{c>=#;@ zCH^fuS{|>LR%XIuWxlB1F5PNfgs{Y0^zjp47)um90eQCyt9m*{VsFBlZq4U^H8LRr zPh8)BA=h?S?1wZizi7nzeQA-L;>;YU!($jmdLb?DrJ;0ZsvNDS8plEeON04EUPG?$ zcAID=l>DK7q?M5QhSor8HJZ!EDI1ZY@1W^n()ex7Av$?3J*%6OhRATlj=Jc#EB27f zN2QZg+={iKjd*2?{~&7ajz+8znb7I6n27H{!Z$ZV(zmWD zb2^fo4qYo9tzY`}xpTk1v|+5&k@PIl@jNPbR&DY$kL{W@Yu8wFPLAewdOVuQ42@j3 zbm=$d6=bJ<>X%deGl$2KZ>4SpllJ92M zmrknd}eBM;Q0DajK<%6;f5=qf=PtF)7f#ybR(f`u#ZRUkR^wMDH- zS@|hW-G^Rxdjr+QndvDW?2)AWq|K2-HD||mkK5_44d)m6{k$C_Dqso}ZjI!nq~eW- ziJ`+7FK>Vf99?2R_b^Fs?#+jPs^69Y}w|GjA zHA8@aMZ)Ja#J0lN6@~$+@r2KXw6iM3ZTk51L@5G#R!xw!quIKa={&9+#BW|na1nk# zs8L9Mz#?OX|j|yqQk`N>N7he7N5$ov4(mahf8+Z@jBHmaCZ*V zvVBteW6ICX(mT{!?etL?wLv4s^d7a`jNhG*Pgmc$kmL;96x3qmN@4w##&2sY%aRNO zBRAdNXvX_%8kqFF?(`b|=T&`^A1v@lIA+nvyl3o=B3%j@#^U2dN@_MvJxoo+Su*A(h3 zkGdZl#0BDRuZkBe$2Sr|>e|?P2mTYcjANTi1Q>qF?d?=Q`rrcQ&M_QhnPUVW*x4`l zLKZz3w{bEcV1vTI#8#KNL59Pv#p;s0F1N?0J6zanyP&zF_Dx=|!=YYej9+DVJcjz8 zUbp5LzdA#`)P-;C@knw!$h}1F2sDx)59OFq&*hQCcJUObOb7j)VC72lE|~JP(@*oEe)$CI<&KG zKGA3fpLjIGhRl52J&EpjUcG|qH<5=jwc6|wcseAQ^0q;R&VFV!!;=(N)=T3m1at{A?2;euVm z!@CwNynI+Jo3m?0Q`3rFbFAMhuA4LGx)mIPJ6so!CbBA#&7qw~lf;v?SK?oX-a=fH zcp7CAe^~j0tyjLp5?;JKb3D_gm|?TDw9pYmZ3D4@k4esccT z?075xYhr_Pk4?{UJN^Mk@}DCA!}u(m-;RIVj<@o^0sZU(JDzk5N%xz|iwW(dSs?MR z;1ECF{&;$Hk%PI7CrwJ?iI24VmF=h9udIjhW;<#B`$h58jwhgbG++lzb1WCn zDfgN)o%FGYrGzDzu>Z{r6!*w2a8*WSUU82sZP_O%zO3mryXAP zY>PhY_ar(qaH?dekGeFM%Vmt8u6x9J7Gc)!L;{)jppa$Ctvr_xisFQ^Y>f>%2}u|e z4v*cJj5qu8sMVK_jFaMN?mc3~Yq68!X-xcCYAzbN*GI@!9@pzw1INMMsZ8Zz7}TW! z82cveR+i5Xt@=_|@z7>`){p*--RiXL%bdW{!2T2{hQtY^dyJsiG?k8)h}>@0-cuS6{>^aJKtxeXSMHR#ebZ{w@-p2n zpZC9)d{a~n_RXlPY)eT>Z?wj(^?TCz?Q^@5^Rkn2!Z}{3?LKFlqs|oyhqBzROmT}f zg00_)BiZEIS?Hk}I!S6YHCv1r<`LRJK!EnvFpNdXHYCm98G3pVlpq{4q1!H`t`e>( zXEYQ|iPV%$uW=`Pu#r*S9z7{1Ew8#U^Fy~!6{2if<)RrMz=TWtiQ0KhH8bkHuJYOK z^^I8uj#@AWHCJJOS8ZFq#~1q_mrsm2%BI(M%{5|=V-C<*I*aGrUFMv_1D-aX)@zAB zjNj6xJ4;2&Y&<@s&GJ^3L+hbT|0X_~NOG8LwTbDE;mm;XTrbIOnf@*1CNur+N$KB# za~x?`;fa5U4>3C6(CO5%`y_?2vB9!V)PRecd0H1FCWni`R?EH?%-2}e%9*vLNp&Q- z+^({-I~wQIh?WJlrPDi#n^#V2-&o*DcDtNujlCuHbLyJsRYr!pN}5-AL`#!u=&odU zOUtRs|+Vb)w!${Jz!s&%mDr=@hQ~U)rgUvOwh)YZ@b9JxeDzpRQ zs>HvCRueu7t1Ix3Sg%6fTyMt{*C}3C4{wW?_PL3T5SH^isoctW&L_^8u?DQ{yP1=+ zF4B->S%TaCWTA7JSO8_q@y#QC}s!#tfaj*AaE+?a67X5 zi=$J^)o@c@VP_<-ZElm6*49%vV`Y1m>cM|%?$q-ZG_Ng6b-MMQ`4#!~IoS;r+0`SS z^hmDiak~5tXE=%;t!yt#&8X<9D{8F>X_;wR{^->C)v*yOLflioVoqH_c9K_jIfM~u zE-a}lPDw7R>Zw96V@#5+@D#=*ES_ZHjna@T3Kr!sSLWxmfy0*m7mf{g3Nw_sy=qu1r~(q5Csf>}}&AE)@1GDX#`F+zB@!C}mxuMgw= zM~=_)sb-8FiGP{qERU(SN%&`!T_%3kB>ZvZRTIB`68;!n8RT+yPr|>WdX~f6s zTC^DUVyuE}o5V5^H~T=u@>ruC^@!U$c)`%n)iz3>yWsg zm!&zKE2m6Z*-3v<&0GV|bi$~TF zkF0K%UTks2ME;_|Wg03>X*J1<1uA|@s4yk$PWQyeUiVq;w5u=bsz|9Tf&Yr`bQ^}t zJlx7ncZT2}f2t8lqqY8w!E__5yg0weiZVYHQ zLgD$N>UF}M?Qz%Rn%8i8@OLscB4g~>>iAj;g_!7 zpldLk=;PlM;@3o2tV*co=?3wMDdKT|)STzH%JrHk<#{XdhhWR&Rh02GKPCPZJV<$} za$XZF*n1^^VjQ0ce;W4cY^uv_bu_)j*#KWbsjmoXbUKOa)^s=+sW~p6^KXGa>9Ine zQ~b&4cX^$$EGTp^*#9l#xKn)^+bSdWaIxqjKbAFy6K9X=O-{q@9q;xzoh~t09D5G_ z?d-0uCZqOzvu54U?7c!zv$`}LhmHd0khZ=EKL>lXw@Bsn8#ISOr*tRJ#`k_WQZBQwY*Ah9${B=(It z#I;V8Jr{ET=N+=Y;F2a+QCeJ^T1Ak5w=}99evczoEX0evgk*SNOg^l~rqW;Pe!pQj z$9H;shRX|m(WzeOAP}{3`OL5PO@2u)3yG&SMB*)P3W+Cm_${Q!{cBs(MN=R|h*c&HqA{6h-U0h0G+svw|>pQ#F_gBtr zFHG_Nylu3hVYIDvw6S4SdyPkUOlCK zRbSuQDQK~z=uS}j4{q_j=4vPLzc%r@`rtX0R%&gIiH)caUeXK+buV;`J2`eUT4&{n zqMv?>D->Ut_rezBy$7=WR=eckc*z%rk@Mlr=h<;<<8c?`f(9JE&ryDC#j5cZ?8OJ{ zY23;dJYeQBaM5WeK6PT^HJ@GD13TWQHo!k>6~|H=tXx(bPRt#M=iWlO?@>Nc*4Sm| z#bYn|9I+28LXjCHui@$_U;up3`E;#M5jrl66;QD8I-Ffoz?qn(chc=G8-{?=-hx+hpRS>O;zZo3h|N zWva~|YrLWNR(>pJwl)m@+yrQYC$?zM;@{rkcdR};-2ImCgw;pHv zoXY(Ews8H}P;qBnvde9F90g-jhA+UJ@2Hs56U9y^7M^i_UrBRz)ku#okk&P-YW}?X zlB>k_s+!c?ilEyI1whMbEFAx^sIRVlMyWfkd3tG1>-@H^HQm8T0P%*ith0CP?D^Yz za@rS6artvQs!HmzD!Z}^b9$;JpTLnv)QCuyl0}?}^F~z}feZ}egLsw<Z)i6Art9yFa zOiA_l0v?wynm@38cyRk^l2fQ1BMKJ@X%BZVJiAN@HVMDi9--$GwRZZalpD?Tgty9x z;?p9)pHv=Xn_15T%2zFkhgJO%oIF z&pQTfxCxJSvmL{9;td30Z;D!)Q@(@b3mnAB>(~}l>13vA>CxB+I9V0By&JkqdKT0* z&uU6a^;GmP?={?z5bBIgGpaIr*7r3pYHD26(mdMOG^)9hHO+8s2(OU9yh1U3`%v?l z1JUY%QoVKKl0Kh`I~`t;JZrq|$zd|IS?O=s&_UF~C?ool*Pho-xbv)mtXz&R&b zEA*MNMSu6RrF5+>r7y76*8$Q8ofM=f5l2?#uEQ_N5LU4(u;q&_wM9ex} zeDe#0ktb1dL{2SS%$e~Ck7#(5@JPPf;)D2vGz~#`Xc+5E?F3jF|1$MnWv4wa;;Cs~ z{P3h)G%I9T4_s(!3E~sd5*UkX(&tJo0nhJCexoKU4`~U{u%?rk7dnEKcfHgR{%+|A zC>c6}6}wgH2p?NI0%D;fSg{wfhOola5Kt>L1dfA-u-?=VfQ5!&V(&gfY6u?#OLJQC zVh{OF*_4y)ka(ut<}@@16Mv6s7b3~{2`)-nr9XVWsW|}qyp4TyovAqh3(di#WZx2- zK5eCmji|tyJxR81E+1$9GLIEYwLh+qTpmJ+nljTGiFB+N7r>+`^Qde!N2Tq%iNgpf ztWV+PdS{>_*OQi=;>c_WO&+joH1Rbrt)6uYS`rpdRn6^)7KO50P(EBnu=vqnd2U)^ zZnj7F`X`NJUnUmGrj*6I z6aEbH^$py1NzzPY%SqZnqTl3t;H|ibIe%->bj`z zoE@d!%I$2ZoYzbc{bZL+nJ)g>G};_{otPp@DhDen1}o)Xsyr#Yt*)fMF^^gWKO?Fa z)S&mkg##0R6lDDynK&x}udC0lLTdU1pEyFc8%_~N#zRJBP zu@CM=-|;5`4<)4C`vRq~;trMmlKLa`f-L=B6(2HZ>^tgS8~eboK8Dshii+RF?DS*4VBKv5M{I`arX>s3 z+w|h7)|yTNcO$mEwCxr1Ts|ZA1654b{VwB!{J-n)eUy`s)MoKdIIrrlBXuHKj~NDD z0K_P9dptBAMkfZSPEfLhd$k##RNH6KBL4aRf|fLhhhMPdgu&_5w0PZUI<;Tf$?ZR3 z&hDHvXJx~PcBKD^<-~ICxr3#1uu>2ged;@)9R7)u=f2gG*krTDkJZXBkN5(19oOWS zryJhnl(cm462?;}_>~gKOK^b*XV>IGhgV(hP)d>?DSWywElR!p$wZrSl({J>dgeLe zaSa{VukP_bs=@q^(ekK6Ki}fI zSca1~=X^LYU@cbX3pKWnx>NRpl(kw1#tN^@fuI2k24?xcSY6Dm*5uVDlUJLvwxy|l zr@wD*q;pm&4eq?IdMrBx<#g0l%xRM51jHTHX=NfcC_R#|yk zRgCW|OkS1bAl%d#?r$yxN3os|)foBfnUnj)dGwSEuaSSo%JKhgjg(Zk^vo_L!t#0= ztLC>_BxXmtVeDyG*c^icFlehW*T~=SUpytw;w6%(!KnQ+%cirLntSmi2=(g`bR#6X z5>b~u2`S1n3o}?u~(IZJooFT51PU2TqNwn}eFI=Lf z18_F6M>6JQ-t8)_FHF&CR*g(_sJ~&kP1KVYFzoEKE~#e{=McX;q1D*cE@SK@K0~iw zM|w5!?H#nN&O9~56oG*F0<|cisR0-#TiFeGO5o(ar&fxB(=A6}vH6SopPG3sxdKaT zN)aeYrnB+VxN1w2HgJ+Ba8d6$LxWrT`nC)Xozq*^ytH%5vX+)*Q#zM6e?DXTz`*ty z!xRqpt)4Okwh;=62V}b};?-w|Io2Mv?ZQPWZeI$*W52wu`4p-5;?AVB8c->Zo2kk6 zeI<_`E?uL2EtP^j@nxT)SMJ3d3Y)pK%?Y_^Py8yz)`__bzE8G318H=6qb~;0nSkl{MtM6V*XJWDyUK0yl zY^@vh?8DbRC<@y9(9@}@5d+R;(bTD;pF`Hn7&_DIDqub)6@j4s3Z-d<{WM5|ru;brAcU|v= zcL1FR%HG*}LM(Cfr?zlbi`gHNa7fOVDx*es$rVQtU@sIJL z-G3<|h^9T@@3(9T;js8_OAhq+2D=C8)|uH?z?ywUueiv*I{$>&yKxWe7k2Et6Jlwe zeJEqKC(x6!WJ44Q++;xyLtd+4XdI_F9XLU$`2T-8ApY5Z{*2Qun3^7LDoTnyN{f=p zN9a3^(w5?sNZVL%&&sy8l|4OYwb?X5*ZQ|KrT#yqN!vw~#lCe4qG&ZC3$bR?WZC9Z z|L;zc|CMbX3cYKbUnmd@Pc+W|CytWClWp@-zgo#OCGK#nMHLefb{lpp1h`fj8&Y$c zhU=3GJuW>prLem(pOh(9pMDgI&KanLkEhQa>6sD!RQi6AUbB*=(=8Y+*g+4{D4lu& zc5e=0%b^>A8jL@gw%J3P5$S#_dDkwnk2uIiiQV?3v~r()8mA>Ir%d~(1{pZHjA*-b&=n{sWb49)uGQw-dJ*s}-Br z2w^eP_b4~85A2C~_Rfsw>9+F#Yv#F^OFW-AV%PK3MRp#xrO4VJ7$9qV9r`Msmh{+b zSdW$Uk{(N`nI6_e>d;?_>Gy+9TKy#6N>4fn@K!!Zmt5#aA$YJ%B_CMYg`1wOwB^=~ zX7UE*)dL-bwIZ<#*$PRw;pn{~=^4xGjeHvbLwg&%bkeJbTU!SqK?>_nukKtmB~X*% z)}782x4$sGYF=~8+%gZ2EW_n#-R)JK;bhe_mn@`RWGjV={JZen>DAG$aB_XM*OAn} zqI>!|y}HMV^IJ_fT#d`R2S@ANQsLCFgCY~;%!4gsWHqr1lsM&-^NJ2gwD#KcxsuzT_eBytt(!U9L z@V`X`Zim-!`$cMXmFjiTDVA9ydI=jDjHm-HlJJBbQ2%OTakd2wMkD;h`9ziw%}Or&AHb|jSilTT^B zj^N|_Jd%6N;Kc)X>dp1OWZLq)7O%KD9@ z`aV;ym3WfnvW<_fhAcPvN49i{-FJq~RqZSj3k~%QI|ExUjSW`*+$OI!}qy~S?XTV@*!c(GiO(AQS)foo^je9$)7Ee4i* z^iDi_Ea)5vJav+VIUVOPFwXGhi!3NxJ%F-d(PKNPjxtebR0~>Vo2NhNVXieuzt#I( zej&aKFIS&&Q}R_@*R870t$WmOrC!T_ekbFbP8F{x3PX*R>V{vO>5SdQRpadoc$~(X z#R~ixyG6XnH^e0!H-H21Z>P2k4Yu3~YCGurFB|^=Zic>T(0#;c_^Vjnsk4jz#rqQu|P(dp2A0dWNg# zHhERp^ke-vLJMFBU`OKgH}M)3?wW`JBht}fs}THr8Qrcu}3+;I>N|ASWRO6L*n;~|D1TrE}P_! z#NVUrOsJFOjx7D*bEFic(n&%{?7mScAqZ>Hjaf8Gwx$N~_ipgHAn|4*oh{gIsYRGR zkO%P8{u*=RF^^YMMZYt~k5PyQmseG7c)uK{zY*LW$2`Czxp?bF z;zdloGy@t{wzx`6Mcwoc zCGPM~wlBbP2VGvCd5NjfX_XaIqtU4qmD8e;uEvJW&W6S=wX}MuynLv-W_o4i^qS_u z!RCg+L5!Db@ER_Gq^FTCipOa&14-Orm{4=V#rCeQ30{Eo$G6;G=Mh^?sZF-TsIGlp_>WHhw2k_Cd}=DG~))* z$pmz1qW=xI&+{GJt?+yAb7Dvnk?KYIYDg8Ye!eE~sTF?7gG%c}K(`CQrT9Hxa>Ta| zc*v|H8D)+H{Ij=y|e2X2+rf53K7bDAMrSY7+4b zo|fWu=zF+TK#uTuNHp;&7x>4^IfIT6>u|aldrK2nGq=vfC=yehUQk6{a;__O88!su z9`ACB2Xr`$(gQ1~p-&Oe1@_SkYOPk&;&(nxiyRsqp?ue5j7l&*6`ZZ1x*j1)a9tA6p6 z){)v^5Oz6{?!-GCun_#y>B&rPTiHK-Bf5#lI$qIwP2Q5HWJIccO=& zDC}I+8p>@RUfzm#X>n1YdeMv~Peyvr?8>^~=G@k?zQ)h3NU+lklTx|2|YV6Od7uG$8u3n0_S@H0VOLvTK!6g}uW;HxF_v1{tmPV+E z+B;a5k93dIl#@IdORd{?EVI((GpuZh$Owp48b*HW@+ni6wHCJ(syVg!jib$=Jg+wP zzZC<~=(LKF{;4B34lk*hSzR@==1j5N;cFPIYFgOXxUi{epx*EB*Y!o3=hxTIZ;tfV z-B{IMTH0ULH+Sy%;rV@4C}5zfZ$9_VN9djBaOb8G6mvHuX{Mwg7wOhkQxk!JKgq!- z{e<4>+A`0N&Y4x!v!veao0)CYH}llzxO2`uXXWgmn(RX7O-@mrKV^vRC z^}Lqa1_%pQ+ICfY)gWFpRKcR)+kam`A@IoTj~-Y((As}p}w z3j}Ucd5lUYy>Ap7m-;f+bLBHoiB*tfER+ifSGm|(CG#9-sv(J_q8=W^{C3VyuL$L2 zlo$f$Zh72))GF3LdirF!CH zB_^t9_Jc!Fsvi|dsb~=)iAs1mF&Icz;{x%TKy9iU#bf#xrsh|LL++5*<5x92tVbjM zI#PjLE3%3_!OGmiMo)_0fRVv)I=m&JPzY}<`C-@6&W70p_ndHOz09e3k|V|WHQ1a8 zC2QiJSVKzbeh|&CW}H|zf)kec1FRx`K>Y19e9krE?)3C$4b*)lZXZJ&qNd}nE8S`e z{2Cm(<3SA<_+y`uRfX39WkM8@4*dl!r*7plc$4LLRz4c!|YQ&6pdK9 zD)82~*_M`;*33!s(1_X!i>=@V6CMG-D!(o_EjKx%2+w@yR)vZ(L{>Dnt{_lSl$~8v z638tn%gTy|Jz3GLjDmo!y2`SHC7GEe!R&}zbyws8|4=9^n425S3SC#2ot2Z5m0d`D zZh#Nihoau*Gqo7RnJ+ZjFOnyztqH`;NnmhIRd!{Li7j7tT00xOTO&KsBC`*9R64HFAoDGb_ zE~gr=O{@pTX{8anML_+9q*;86Xck{rcH=f2)6AF#oIZWVX+Se$s>N<4ramKSW=xj2 z7N1Ob3?+kRGuLiqLfr=pXl6`4avf2B1`HB0rc7LmPgS8VqFH>Kc!)F~!UK+JHp{sh znCE~2&9WR-69H|iq*jGXJoQ-)+W!tS&M0SI5g& z9psV2`9D@yPtM;=`5o}ovEt9A{IaKszE&B3sU5$R;RGscmO7c6B{Qjgge=sREH7V#%Lym$8j8S-rK&<{NU$Wh&b|Leh+HZ^` z2T{UyAMCr*PoPCR;qN$Y4x38xqd1+hMR6EqGtFFf<;i73v7i-6N-A@0W{VFPvCN;A z1|FdC<)?1Xq)KO;vT@XlVD+EC}Xa`6CAxniF&Gd9jZ^8bIdze~HJ)Tf1 zbvLtAbfm;;sP*DEDUWmCbJ;{KPVU6xuuN1Cl#cwiyFHdtc_0MVL7O1;6blm{XSUMR zOB`MYeXiOWmE{AKHM2vwAMVzi`72vn&S)-dpHb;ZhMy~5%gnSJDHgeWj_T8@8iy;Y z!iK})Of@Q+>zDMEPM;EX7&~d@{B=U(iKlog=0UYuQVB1juzZb0I^04L1FdL3x}Ed@ z9BDDj>2;$?tPzc~)EV;>l#%T*E_DTFq%y4KpQ`j#QUTrRI`UCi0T@ z?i0$TVm{5k))N?Xraks15?dzlF4Hj)5wDwhXyB5v#(L~s#cjpb|HmwVFf`Zw;zp5$ zmkauM4nPBo;2nWT6o;^_Elp9`alv4q1t1lO$N*xXv_e19GZ4eyh=@jXfciI}e(Lz# z&paa)-kO#0j1;53)taw?V9USE|FQ%Xr-hhM4i3GtjA~i3oos#Fb8p`>M~cZh=*tehGi4H#ylOCcH^W-WMlD#BTQ3G3kEwDz`Vy zKfWEq^wngK$Cu>A3sn_~_{S|Yc&|+C6o1q5@qV4X2C;_Ck5k~%y(XS8d>#YurcKp- zJN+(@zcvZ4CAeIlf7|P#-g8b|EI!x5bfyQZl>L|m9$lrL7jM%w{wb>PxKr@n&rdbq zSA2%H13Lgbso?tjD(JZE>#V>^*e`B_rMTG7n45ASo+8LC-6>Koz}&diU$r^~-3{c(CGR;cqPuA=gJ z^|wdFJStMWL*cD>m;(Q?-KKM*ODt6n$d{t|nJ4J4W?z{PIx)kCo%*8lJG!=x;I#^m zXZY;)*0WD{yMCgTU334Er9Zl+R>OT)y=>QikBr>5E8@WB3~kMy=*CBue`@Wzn&dG# zrXWin+}i=4u^*{3ymo5{vzsUZbU$;R2mE__xR)?|#2 zFzgvv%OaMcl$|;G2(E>wr-L^7MY{x8WHHAf4h8Pus&htiGn;C45~zB8Q+j^Wv}h_u z;nZnaHC1YDb#P#6nkL-Oi?o*08)iqR*T6dEshJ*~-88j4$LmaT;$$hKqN{$mccoYO z&gvVk>#0b0raADLNcbGIiSNRm0ehwF32HuCj`t^NX@L8Od}=Dt2u2W9tcqadvEhZ5 z6L^G7^Wc^?sHv?~Yifeirln~rT)G@$?z~BKr&rEs725M&Nl$m(aPKOQ@U83}uJ5i) zbEe@H!_-!*0fv_vP})>po=fU7UXkt>^!aXc&qn7w<`kg!2u*MmafnUJM?5;;pn1~g z(r^iZjDg1IPy6W@ib^X(Y3;H!fr*4IwcRKD+#>hAC_Sk-df(~ud(HB+UX&+aNDf2) z`a1M4U2S%d1_<3okk@Vmz94i0)VzbRW$+Cv#3+uai&)cGWO_%Z0q1TuShezNvV7>00*xQA#!jrKwS~4wyy*D>B4}#4j>fN zr(9~5`!DpYOHZv`saJlHMwP8Jh_8!ZLhn6OxtH60=3l7Y)xh7#_$A7Z8NUQuJ6bav zlsAEgbqV?vO*kxV+?6o>U}eK(_=Vebg|1%#q3bigr0voi$qnuY@t5(S2g1sDAZ0I% z$v)2sr9rQv*WR!~k~W1>gIGv0sHb_16G{`mbUz5giR*;oRHaHhDf%%&yhI(&Zb23o zM-(=K^Hg-}Of9|0rK#PHbg%a71X%r1y4Q6mHS^VU7_0CbZAE#wJwfGhX{RayxyqEGgohk_zfBq05Fka#e7{I8{17I%D}sfUlyl4p^J3#bpa_yP&E3>#%<;|| zN=k?EMio9Ubh_3~*KFS`c&7Q>pHr~Hn+yY4M~(Q^v=Myc59=Ll9iqK9R*%0AlE&wN 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 literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/style/fonts/logo/README.txt b/nationchains/www/adminapx/static/style/fonts/logo/README.txt new file mode 100644 index 0000000..33d9d26 --- /dev/null +++ b/nationchains/www/adminapx/static/style/fonts/logo/README.txt @@ -0,0 +1,67 @@ +Quicksand Variable Font +======================= + +This download contains Quicksand as both a variable font and static fonts. + +Quicksand is a variable font with this axis: + wght + +This means all the styles are contained in a single file: + Quicksand-VariableFont_wght.ttf + +If your app fully supports variable fonts, you can now pick intermediate styles +that aren’t available as static fonts. Not all apps support variable fonts, and +in those cases you can use the static font files for Quicksand: + static/Quicksand-Light.ttf + static/Quicksand-Regular.ttf + static/Quicksand-Medium.ttf + static/Quicksand-SemiBold.ttf + static/Quicksand-Bold.ttf + +Get started +----------- + +1. Install the font files you want to use + +2. Use your app's font picker to view the font family and all the +available styles + +Learn more about variable fonts +------------------------------- + + https://developers.google.com/web/fundamentals/design-and-ux/typography/variable-fonts + https://variablefonts.typenetwork.com + https://medium.com/variable-fonts + +In desktop apps + + https://theblog.adobe.com/can-variable-fonts-illustrator-cc + https://helpx.adobe.com/nz/photoshop/using/fonts.html#variable_fonts + +Online + + https://developers.google.com/fonts/docs/getting_started + https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Fonts/Variable_Fonts_Guide + https://developer.microsoft.com/en-us/microsoft-edge/testdrive/demos/variable-fonts + +Installing fonts + + MacOS: https://support.apple.com/en-us/HT201749 + Linux: https://www.google.com/search?q=how+to+install+a+font+on+gnu%2Blinux + Windows: https://support.microsoft.com/en-us/help/314960/how-to-install-or-remove-a-font-in-windows + +Android Apps + + https://developers.google.com/fonts/docs/android + https://developer.android.com/guide/topics/ui/look-and-feel/downloadable-fonts + +License +------- +Please read the full license text (OFL.txt) to understand the permissions, +restrictions and requirements for usage, redistribution, and modification. + +You can use them in your products & projects – print or digital, +commercial or otherwise. + +This isn't legal advice, please consider consulting a lawyer and see the full +license for all details. diff --git a/nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Bold.ttf b/nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Bold.ttf new file mode 100644 index 0000000000000000000000000000000000000000..e105b89f8a05d7df0df4d4ccafa17a4b9933ac69 GIT binary patch literal 72836 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 literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Light.ttf b/nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Light.ttf new file mode 100644 index 0000000000000000000000000000000000000000..995eae9bbb56ab240ebb7d3232393eea442dd79c GIT binary patch literal 72964 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 a_6dE5fPA#i88|$IK$MR!7yxPf68{H{z=rSu literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Medium.ttf b/nationchains/www/adminapx/static/style/fonts/logo/static/Quicksand-Medium.ttf new file mode 100644 index 0000000000000000000000000000000000000000..2c9f03abc8fc4af2e051e650a4a1f6ecb717b8af GIT binary patch literal 73176 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

E?OUWrYdWHzuQk!< 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 JiyBY2{~xpFXea;x literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/style/fonts/texte/OFL.txt b/nationchains/www/adminapx/static/style/fonts/texte/OFL.txt new file mode 100644 index 0000000..8c54d91 --- /dev/null +++ b/nationchains/www/adminapx/static/style/fonts/texte/OFL.txt @@ -0,0 +1,93 @@ +Copyright 2011 The Questrial Project Authors (https://github.com/googlefonts/questrial) + +This Font Software is licensed under the SIL Open Font License, Version 1.1. +This license is copied below, and is also available with a FAQ at: +http://scripts.sil.org/OFL + + +----------------------------------------------------------- +SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007 +----------------------------------------------------------- + +PREAMBLE +The goals of the Open Font License (OFL) are to stimulate worldwide +development of collaborative font projects, to support the font creation +efforts of academic and linguistic communities, and to provide a free and +open framework in which fonts may be shared and improved in partnership +with others. + +The OFL allows the licensed fonts to be used, studied, modified and +redistributed freely as long as they are not sold by themselves. The +fonts, including any derivative works, can be bundled, embedded, +redistributed and/or sold with any software provided that any reserved +names are not used by derivative works. The fonts and derivatives, +however, cannot be released under any other type of license. The +requirement for fonts to remain under this license does not apply +to any document created using the fonts or their derivatives. + +DEFINITIONS +"Font Software" refers to the set of files released by the Copyright +Holder(s) under this license and clearly marked as such. This may +include source files, build scripts and documentation. + +"Reserved Font Name" refers to any names specified as such after the +copyright statement(s). + +"Original Version" refers to the collection of Font Software components as +distributed by the Copyright Holder(s). + +"Modified Version" refers to any derivative made by adding to, deleting, +or substituting -- in part or in whole -- any of the components of the +Original Version, by changing formats or by porting the Font Software to a +new environment. + +"Author" refers to any designer, engineer, programmer, technical +writer or other person who contributed to the Font Software. + +PERMISSION & CONDITIONS +Permission is hereby granted, free of charge, to any person obtaining +a copy of the Font Software, to use, study, copy, merge, embed, modify, +redistribute, and sell modified and unmodified copies of the Font +Software, subject to the following conditions: + +1) Neither the Font Software nor any of its individual components, +in Original or Modified Versions, may be sold by itself. + +2) Original or Modified Versions of the Font Software may be bundled, +redistributed and/or sold with any software, provided that each copy +contains the above copyright notice and this license. These can be +included either as stand-alone text files, human-readable headers or +in the appropriate machine-readable metadata fields within text or +binary files as long as those fields can be easily viewed by the user. + +3) No Modified Version of the Font Software may use the Reserved Font +Name(s) unless explicit written permission is granted by the corresponding +Copyright Holder. This restriction only applies to the primary font name as +presented to the users. + +4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font +Software shall not be used to promote, endorse or advertise any +Modified Version, except to acknowledge the contribution(s) of the +Copyright Holder(s) and the Author(s) or with their explicit written +permission. + +5) The Font Software, modified or unmodified, in part or in whole, +must be distributed entirely under this license, and must not be +distributed under any other license. The requirement for fonts to +remain under this license does not apply to any document created +using the Font Software. + +TERMINATION +This license becomes null and void if any of the above conditions are +not met. + +DISCLAIMER +THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT +OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE +COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL +DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM +OTHER DEALINGS IN THE FONT SOFTWARE. diff --git a/nationchains/www/adminapx/static/style/fonts/texte/Questrial-Regular.ttf b/nationchains/www/adminapx/static/style/fonts/texte/Questrial-Regular.ttf new file mode 100644 index 0000000000000000000000000000000000000000..b3a45b411a91a83af1b82693afedbdf18e8063b4 GIT binary patch literal 148120 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�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

rY>b%Iyy2=0`& z3I1(cofk?=_xs_qM=uIl`}<7ArlJBkFsCXHZ(J|&ZzR;`8f@eqO^^AZ)j5CzDn})J z2|Uukm%!#^k!7RL3NU9K(k&{8%jLXP#uZPp4jW!E>XfTYKmr)9gYR9%)8cs3@4=!D z(p4s*01gP3r_5_D3*>>t+Bue%#wDP%*!ckko-^)4%Y}%1q-kOnaPhWC^=P1V+_Is% z??-`%n#wrY74`0m$9`Dj#z2!7G~BTSuz1n%4##|a)Sa>w^2EmG&Fr@E%RKCt#<9zZ z=E0YB|EBao{wR{a1d1}iZb;=SDqA*4@8ka_{{1n&2k<@iz_dVzb3P69HfUXxA7IEV z!+FRiLUpIh=T&6M`t8zQ5zjBPHud>~InSSDLA)3FtHFpY*h7`5k0^D_J?FF8{1p-S z+y~BCj@Rw1LgfnZRspbG#aq{AYtG`Pjce03XLHe2&~ZNMW7)k1=LFVNo#j39y*SDE zD0NO*#DvhSc5qoA)(tM7i`XhrPI9d%pO&7$_i*v7wD1hke(7h4Qg^;=(!0uKqshh$ zMlA{{hKNPDL@$zH#V{SgGctDEemtGOggo!yEOfI<9+@5}0^6_`8*tP>0$C9RvUYk{ zr#CJColhs?IS=B3u6OxflHB}IX}vzLo7`fhK~rdthB+SRJuiC?{ErWvGK`r!F*; zub!bUX(m(`n#tw(NROZPB$bo40hx`@7siw|Rz9Y5ebOLAed<@H9nv5~`3TK&x<1Lt zyc}|}@Y@Qie^f0Gh;nt_Nh89%r!}H8)us8%smpgwR34D;#JPI}IeFmG#DPmO>F|Bx zRyjqtoIY*4MG`s+gpkRSHVSZH0tu4UYUI~KUG?bV`RcN(E+$GaUx3yJa0LWqP$Fo( zhSRpUm|J*+MCOtZq*N297;=1j_qTQKA5xfmW}ssxq?=JH!nKibaqDbDcqS^}I@1_# z*gl?eG;N8sOa*OWXP~2!{jek9tQ%-a_3*Wc%AJ|DfsXM3^o8GTBeDZEddR$6iu(Z-Bx1+p(?A=I&Mn+s< z0K{!PNTxTT8R{UW0`v$-s|NXiw9D>2!g`pjQaY-@@#blx_e`r}{5 z$N$O-#~}yLK`Q8go@D{L(;;1(JY@wj(NQF^Be=?e?vtVEG#N~rfJ??z3+M+Zrx2M2 zjZ26c;K8^&!Jx#heb)~FEi0vOXT-ZRID!&x7IP7FM7@>~V=#JPkwTmwE@zx$ha97h z_|{$TyKjj_Q(vqbt<&GLYen)dy;r`izn|X`kVLxnmrnon<;?zUPw!9S9lU&HDSQ7b zPfL&SS+@b{Uxit(kp>g}z@X_d>rew&(!l6$O}(;=&~f}vb}mh8%WNX>h7+uqHT5oQ=4 zPra2laq;N%@zJyATDqf)om~s9jQY;|Nb(B^d|Lw`9{!MgPMPfa ze@OXA9;+A+=uO|Quy3l!SzKr@XzdaAEIhq)m#hrS2&_hrNXov8OJD)sqdJn5AQ36h zDh{JZMUYXw30J7HEFU5f_q0)3Y*76k?CeoxroOz&t;6MBcD^l13l0>^3T78>G#=TQ zo^pda27EoNQ{E4BESPCMRD?kAN(jVg!9^q|fS+O-ujA*JgG3 zV!FIb#?2Gyb`G%Ston3riT0OSAFY`*fN5I54Al{RB1b18q>#ITCE@TwoRoxB)9DOL z5QH#Y6ek7131ME4P7T+SSphO5jG0u&c>a0DL$1YDZWI~iiN(3Y8-{0i8eBFbyK~by zC(eEUjl)|;dKd5dDuSbu`?s_Dd)Zk#W;X9bTvSWj@Ncbc&8Z-JapQsUUsDe?4XRzb zDN$Ap|7Zxmi)(v)0+^f-pTQ?dv zT{dADJ{-(Z#x9uMa8ddSj@{y>4cx>sdSYiR670>=$a)ATj9Xc7AqVD(_`VybS)Igv zYViCcxe(S&BxJ<+)rF&sc14A>WQ5_wl2Le~jwg}mqx%l|u}haSvBs_Ewbn#pk(!B6 zZ%@cMHqg7NL6_7eE|x!qo0mepk@|*q|GoFNO~j*}f%K(Jnf*vo45v0oE3a^BqqY1N z`q&`d{t#gyo+ZF3b2Os zAma1sIpDi7RdApPR}rpE$)r|pU=6E5(5qBvIQ96wDP(N{OyvxBr*!OdJ@r~Xu21p# z*1bTbv3{V=GkpEP$M_v8GFd_(o!rLHO!)K&`%nNJCl2TR@D!Vj`KJAp>j9^vdOpJG zh(*ge=Q2LMR-DlSlSi-)GY$-7h@pE{@H2Hadk_9TJ9-8FjHViZjrdn~q*3mXUKQsW`gVYGR#84Jw6>MMfY)d08|2M@!DH&#kX?T) zHnH%-Mio6VHs;v5bzrN4*>;Td!VoPmY{GFUo1p zE3hoVnvI?i9GItsQ-1xsM9qvwI!Wk{#PVjRGpa8y~+^ zNRtxv4^sikDSG>~LK+Q~`(?zu1IJr8KpG1Cb!lyegws!RX+u1}OaGSf6XcCi++>FCCW$CenTIV{-zMG;^j&`6`+5=Ux`Y0WZG92;rgMK zhs?|Tm~)O@P@u~LKOlhuX|?fBIC8*g;m2W8VFA*VAoHz`FsHCs39vW6XQ=#5VQ-$= zo9yh>4Aj3(g+_bu2As^Sfaj0|5(=v-1ajn$MC<9Sp6E+978f`=Ldp_&&(=3E@PmPY zL3RY7r<(?;&qmaK1$|z7i!h;Z)!Z1l?NMu>NNN!|0+L7*5FTLlsh4c+&(iHvDwn;E&t{m3l359XAgWHz)Xbj4e7PTP?Y5Lmu^fYJ;k8VX^njGF*337~k;z61v2NR*33 z(o%eCUo1Z2L)Josi#Wmnw|Q{+6~Rm6+)!;Za$$pgc7erqrT&DQz+Q+>L^}EsN}0>i z#s0dK+Q-g%|G&}&S-+2W4fPdQH6YarjpkT(uVG328ufbRRT@3ki6goY&r_f?K-Vv9}Fb5ExJQgCVdGX1Qn*Hoa)dn|3<(%`=`oIoJ<%s&N zpUPb)o79u&gkKo(3UJ`;UlqlG=)(NnnbeU;v2M+nA1HOCxR5^cLmr8)!&@f-1wT@cd{>eNfK?}cn&oaGx+`{;R#obKy2^;p zYDpGXn=2iPUsnB&7GuCn%vJgFeXx z&#i|z=ynrV5vC`QzRgA#CbgQ4u{bden~^TN!>oPz%XpDLFcDyZ6>0ax?$qDJyn*(g z;(jZ?S76J|dCfcDD=_#E+a%n{-! z)xDWrKCQ{;*pImP4|62y&!@|~a>@^|SXMhE&x`uYY$!d~+3lRv%2)O0S?M&b0kbAMkRP`$c9}54H3$e9&&l6YF4d;J6(ad zg2SKKB2?u}B{*U7e10K0XPd0W-40w=20S@Sjo{|$?t_C#VV=$?6WmZ8*)x?2u;*3d zbVySx)LpNf$9KJ0u}N*kuDwTF|%YC`D8J2rNq%>-K)@V75mJByY5$;GJ z&XOi8JZ;*sgF1awexGJ>hg3&;{{mb=}G6t67&Avb~iH?QM(g?Tc;g3zg*q!G|7lHUQdg>duV9rQ19vA^d1@T-PBix{9gO~F*-dKYhhpd3z14c-n*h?ykuo{}aaxK`E zIBp09m_=fgQAwwTguN2*p@) zYfG9OgZ?UiWYk+XJVRN`vNm6u7)`N`&KYau)@^GqUro<3c_QxO>qm4b=ALLzNtbK$ zDdsOzKR4$b=2dNn&uhBU>a_^_P;EnPfrenk z<-n;g=G<$`X$PZIoZ6}a5%C31o3&V*8=)(YNgl*xXS`&q*O%upmB4_NI zERmhMFJtB8IsCMXlF=(zie$purK=GW&Mt#~x$%3_O^1!}ECq zd`Cn%9-Q*vocC~ri6g+p2wdn#8UW5yF0I}bjsz|SK$twQ2&qkAyw~m?Xc$h1NLXV* zVtxVehB{$G$Tn9qv`teE9e&NWxii`}wJ&sH`i%H))6~ew(zYwxdbHiKF20*6>IA1W zWZveKR+*->?DA>ppG7$=Qn)1=6e#K+Aj=bWf@p_0F_jzS!^iNEE+>i=^_LmECR7fo zB)grH+2vffr}~63LVaW_t=BL}yuZZRMSCM^Mq{!qWwi%s`+Im!=C&o219R=9Xjt$z zoW@Y9!G2-%B{!o+hH!NYL@z}Mqkb9gwI%}-#X6HsRnU`xF8s;4-5O@5HTPSm#@O82 z8?KRx1zr&9O%-(1UO-2n7h!MWUJLvHw}tgoucNyy)jolEpdV%UtAX>TJ znKn&lcaG9r@&uQjPxh$r=&5~c`#3tJ+OOpY(_I}CS8-!~_+8wbE?Q~+w1I0Gu^=LH#q$gD7%B4(bC&R>{41Z#2HbH5%W-p(K>IRmhQ`X`@f{$ z&dd4RO|YqOG}R|G>rox%8n=n7b&c|7>1m<}ekR`szE2qZPM`y2kc2EP<2zig4IdD^ zF?-UxmM0+^)Xmh^c-%GqT7S@8<|(B}3=711LL&2wEJE5Br(8SjR1)?>a@5p+U?h1# z%?8iy2gfhn5{YcNbeujJ>mNVVuTNR^{fF4m#J;YP&czF~i|uK*7L=!+>pyRJ_`Lpg z=3+5D7hO5yJ0SfvYmR8giIrPsCbj%vPW_YF<=o05R+N`x#rbnsMI+A1dCXT-8c&Qg z0`5+^xSAXi+7D7X7;JPU8` zb?rS%?WvbP9FFH+*xoeL#3CPQAJ?vI@9OqPoJFyTDqEs8>(chRP*p)3g`+*VwjHCx z`N{8XFOevEo%jNI3HXW0)01(%8>m}h&BefRj}Pu1}? z`uoRH_peB^pOA0={O2=sLi8!dzKobVG4_KwV?0UaSRwM*?g!7bNVt&{yo4C{GB&NG z1E__cvm79Sy5QP2uxtuBW@3GET($E4SCa?RVI@E#oqj-`(KsXmNcj98%kBeC5z5LB zDiQ02H^5&Djx7_%mZ!*7tSo_9fJ#t}tObyY5p3>rljFpd%cFUY4wlf~IJFP-^6G{z z&V_gFl=qH}VMS@TeI27Mms})^8<4gp&j@QPE=i|^J9rVYfY8k~WQ$;utIk=aCb~nm zNfR!+k;+*)C1~Y|bya5-xv0r{4e=aK0M%{mVtY4UJw7;?WOpP72gk2Yz1&UKm!8yF zrip77ijYPe*O}t)K+*g)lgu-{yRmWibP27L$Tm%wC)f@_)qT=$`3kPiLOm`0L@nPW zU%{ze)Spk6zbRkAJd4iU!&fjaqi9u!PvtotG(<#Bf1f5XU zUyE`LkpME|6b2VHN^vKe*K751cp-y^4_hV~GyTF#EtZ&vgDKX{!{b+P!cr&2`cC*G zin6ecrG9{Q#Y(edscXgJvJ)}xrxX1k?VQj*V&28NAEf@J%d^IV@=X1#tmG)iIMsIe zIK_OUyj?4YCiG0!ypRkj=;3#m(+s_l%qPl6a_Eh;d0x)fAXA^TGQLKl9P~qUMC&PQ zS{a}7E}|a^k#31d3C{jfDCxAbxye=4f1&h(k&^lgd#u-L^M?5qh>UnOa0ALb4L zu?%ucvg^QTsiz{X;>K#HFB?*q176bWXHHBPPfB<^SvZrD&Hg3O5Yb@Ao)PK;Fc7EC&0}>cXOZwxjnQJnovevie^j}#^x z@Zw$qL1Y{;!QHw>_Mzfh^QNtBJFlFC|9fb1=Q$6ZMa?v$M)cDiGz%>_Jh!a!`4QudgUSsg-V!a}R8N;39LdPyQ&!IE4FZQk`1W~SdzL=g(b&djU~AwRQ!#WMe2$Kb+jG=%8nY;0h96v8kL+6*7NrNW%Benlf0*lK zW)bDs1IPFriSm1Fa1<9_4@XI zzp=J8E`O?Pthgpv(^UDZ)L(I#=1r-uKNab6suX#QW5L9TUQ_dbQ_qia>i{nYX9tQO zQWO0kE^Y=40&&&g;S?ZFfJ_qrnZQIJ5U4)&(lA7YTToY(=^@Wnhh1j6BNCb5w2%X{ z68EsmTd&!=Y5OKtg&$krd#iux>&)={z~%Te@I1P_`G#~~iMxqDJtGPGVm0~{;ya@Y z-6IY9cYL*0mrs+7iFQ_(FXoifs*83wYM<(3)kA#MMfpiW^Yij!+2xR7@51|;XrIp! zED!lOHW-YQbke~=!C^L7Q%#t~sfM)YJf;KB@|hyhzl_6e01l5K3L^$wID|^0%U1oh z8@}*o{woot+x{#lSB{$A3QX%Te@3WNd#By&vGS(5oVwc)4Otm-na34OYe-jgmL%r56yyhXo~(j9#EM87)H^|z(Vsb7?B zoa$%#m2-m!&@;3lBcwS%+-NN?FhmU)1yjy-`bIVeC4Yjcw>9eq> z@N6H;nl(ny!AH~cbG2$ua)XM?*jwpxzTWG!&Fk~F@8fO5!yah5zyFl(*D%oyI1P>CC^CB6AzFD_KbwzT?tSb{4nT(uQ}%7DeMrTVazr&E1!m}EPTQfMV# zvWE6246s}ljmGD#&V$_(=lArSKhd}7OunHe?4GT!pM8qbe9$GXBS*)OSRc2Yq{iFk zTaij1|F_P!X>-lziSGrh-#@AI6qPO7+F?_EazD3{Mx;|>c5MJi8RW)>+L{WB0eUXT81cc5 zL?xB$GUW^jUXhVc&yuRI)q{8#_*!A*yN)cYIKZHB!_z=Gg zxZVb)61BP-(~nEl{66?zBmRiHju8EjAt-+*^33OR%{zcVHEqp!_&b5{2s%XfNUgpX zfmz+Pb)(^qJ}h3y;~I@rS(CW3=9fOtM_^N#t-l|O>G$4TZZaj$c3n1sH7uAp&+42@ z72=i}CO2=YV)Ht>9AUmSx}`Gp83R_7;`>_U$FK^6d`Dc(3i;{`V7FzLzlWoCWPW!6 z-M3_i-x~2m^$?0-Vo=Bf$ZgUG`OM*fpnU8E%0a8JOe1Xro2jhud+Tt2Sw*-q3^A3i zh2Z)I$JW3Mg9sv#u!O)Lpj=P(^-JaKN`CQDXwsilZ(=%bW>e>FG&aw1GOuL=fmtw@#A){caK-AW|ys5LQaR9s9{y6O;$8mMb>H6b0pnO|yDi(8O|6sZAW zaCZ@AfbU(KKpg>{V_CI_S$}5lW`z_v7Xclsuiwyt3o?Q;>@DQk$Bh`x!CARA6HRne zs5zC$-q6IExvvotUI4gCvlLVe+^@~gjv(KC*bc#!|AG|wMeH}jgUIX{4d*HBdbX6h zJ9W4G#nfG?yV!231*Z2DT2dr)0@h4MInvotbE#~ofDPR23d)~4y_f!SsymiVT}nXu zM^1Ip%98Zkm7gheic6=1LWel5b_CC<-(!1si~4e0`mOvR&cMfSrO`<852W8J6XJPf zzvkm06@-5m#&?h-&^8i;%e4a+s+l9{9(u|gd7&8aI(2`nf4jQq1IZKAXMRRBOSj_9 zXz*6ipuF-Y^z%)n2XT`@>AHt1`4#a33A{ubyqIt^oK6?vW^PcOslJ-`#m<&V2xU-L_y|f5gUU zj$~7oerU>?^Lni1KVj`gu(tcf+OO=-`u>}k=YG+K{%EER{ZUbe@9k;nD&(`JT;8)H zKcNk39*i}(tB0(j++C(YUltlQ9 zHL;Cw^-0bhCfQe*Sj#BOvue}T(b1w_VL}Xr{0;)|t|I+lhFSye=_g5`t1J2Z^UlRS z&R;fVaT61%aXD*K5I3vG(Ly@nAjCa&(x0X_9GHbPA z{~#bSfGd4?OtQ@rBiBCu1$h9WF9=5r4{pE8vBA2juWyregX600gL9Ym^<6smNg zUvG83Ecc?=x4YGuXau8qjgRI);tcmwJ0PCq-cQZ=&D0XlG3BIhp`a{-%qRo^mri89 zUiVgNJ8)IG7vQ#P`C5+tU47TBw8``AZ=jZ|6NR--o3$uk%Zo+VBW3bpaRF1>i;nR= zCr<_NHHJm%0_D!)>H=iYOqE3K>Te;3k(es&1*uZ{EMT9#Hm}!?i{PALy`4J-0w6#n zyTb>a4*@Qe*(raPojZi&a6|2MT|;dt*vtEu{&{;b#B94MTs!3o*Od%iwfoV%#kCdI z@=fk}cc-gy)}8t_+xJ9GyUUD%4)@K>nEKog@lUK8?X0upn}L~}Pi%&f8jRYAMC}U1 z@-%Fl5VlC2fG>HTEbe4A?g1DBvw^tvJrK_Z4KvtvzS?Rn(ATJ!8s$b+ys~S;)T%(= z)-?K9s7u&n@<=*gmN^-VZ)|AT7^h1J%dcEaUeJr{2Aek2So`Ih8QlQnF@HYXE)*lW z3R37yj>;~`Tbu)}z!tRf7&y=ZB$-N8lBvu(%``6e^&jSPEYx4cv7qULo55m0ZE+20 zK3BrfxMo(JA7eMb8(pV*po^rqpPcVFhZs z4N__6+&m7JB~Z5*4M+3~p!S&&6@!6Krio+Ih*nnpIb!&Cz40~%A44;Y~+D~8V5%(EVj@;iFIAF_%48R?0?R;}?ZZ3r{d}EWF66tH$We6o=o#`j=BIfX{J975N zHAu&RWiEVw`u<2kK69yK>d_NKLWe6@ueidjfHMQbkIZv4w&Jo?G#|R0{+mm!l{XPL z!i*w@3GfATlkysVU1*cnI3{t2~65vsHX` zFD@>!9sBmBpV_yM&iJ9_&x`#u3?yTZvM0W~(#egir_1X$aARk4AsD^Lh5{MDob zz@`h37{dU|6*nS#8fFX48TCp?J+=rrBAhK5j6Op;=uoiUS65q6u7p`wVopR3OqmFi zHeX_oQf{M@dWdbdSNMhjm24mGYiXN&-_Ft!&oCrX>rhXJ zYkfiKS1p!_9nSSNA4-3t<13SUyY@QW-*(kl_$F7Xp!AUIBdzaU3&J`!;~Fg}yTJ9n z&Z%l=8mZ8?U%)o+pMWD}^5XPQ%M01>$_nt#tCo)`A6K?xbp8ciifjs1jhhnvH{M8! zM-}w2kF#_B^>s1ry6177kQ^h*LH-;ogBp8~+aA@uJJJXy=lem}6xDZ|@5_PvH%}#x z-9$`ev|4aAPIdb+i*PqsD-UwIn^#yOp~6G{Z>7(6<72iShf2)|i8LmG&g2yTFL596 z${keK;e>$u@b_m=2>ksSxvOP%!i@bM?nfSUg^j@0O*2+x!62&zR7?{V5jGpl(^D*D zp+f}hJ7fW3*e{9xn&vPY`RVvif4#V8k)rGQO~?kc^j|cS$s%|^G{#|A9>U7@_&l^Z zGlzTDU=FxodmuZN5qyez(q(k?S~QC&x&-);oXS!K$TCKT8ft5*EA2MTtfUzgjQ*3& zN?B4xf$Y$1ORSPCN|C-eHn+xl-+1%ds*UBI>z!3{+e@Da)qf18rA^H+DV^%OgiK0> zGy7X+8!OXaG|{#vtx9vot~qUg86^#xKaH`a9Idws&066e9)L!$h8>|pfC^-4Ke zsISMXp|djm#63tSxJTHdM#VkW^3CENexOM!a?1B>Xih(HP`Tw>$vQ5^`ZzFb`kAb< z=;SjQT-BFcl8Q!GWTv+0 zCffHVXJ?a3&#%lvO>34$GBYw5^CQZimUlr9A>QR(v>rqq@>P)aPmZ?qp=h>i{+Mnf zle(FKxl8hFWTcdlWlR{!1pZOl$oQh>Hzga{YLzlJGL@9pq$+J>JYyrXmWe!#95`Q;G6g&o zZwOuP5Old9>2my`rC>&;{L9bzi1G$xj7gDR1%*GayrH&0H^lwmyMvXA-+MFN4x~%V zzhYDT^H}(0I+dO+xS@pQLcFcf0vMz^ z@-I<1RImg3MX+(mnq>BsSCbXg+tgHp_{65xrq-5b31-LXvl{ju?J zxxbAn3pmT0_+d?AiNE3Y{MG|BmXK3&EHT~+|GBcDw&CreuJQ@_=_lmdqnWU;avD$xe`OV|e}zaMW?j^Rlr{3OXn7Ua!yn>W9U-Hm{36I`vczwUuM-w{I}R6QlBXje$E?F{MOvE_ zEK=M6V+EaW5qmS(<$T_(Y2B7*S{JZJ&7cn5RiGI)6ugL#rRFUp03UaG2&KK1VE7PX==zxP;Az1ERYc^%rDYVMjj-iPna%vMS7>H zWo@*3(!Qx`|HjJISaRFO_VpswI3+XQrKKx=tFC#fsjVTt3HedO^^s6jxv$7pZZE4a zTBl7D=Z!!I8awEX&ymLC2ijM>Oa8%=-=*;J%Xhhr-leDyZAJJ=W%<@jxsDm2a>5M! zLELXX^EqJy2s0qB+n!fWSO!_XBd?q=1G0Q)Zh2;o!9P!?OVKWxdt}R$^$*kYMY-;C zm*t{enfl}xM>%whmzBqG*3I{eH!uH_jkEVbPC^Sz$7$D8A45OKwoHj=g#R+U%=wr-8cUpc7vm}zOAdPt-jTfzHh9*X^q#rrm24n zqZCD&qqY1NjL)r#X?e38f#uB^8QHvi9vcTkmKdR>2ju5V4`fEDnLW(L)d`@l{7Nfj z9I(m}!YWr`q&$!rDU1(}+6$RsLLRyLK&1=${Tc(U^{pVU^;h~!iU@gCfE;qLcHmf> z38Zk?C%e<8>3xq)%&C+VXp#+AYw8*S-M2TUms$y&!?1KcBp_q^ZlatY-nf;wndskl=@DP9|&HMTpI(~7eH&}V)P13n&UTwy6k0NuyJB%chg$$fT!Qz zICgNNd7ZbzlOFQcFn7e@NkSnV3ANd|UxO&lC^SM}Ds z(lk@cMk4)t(ytBd8HlW{O+QlMKzkF&s{R&smxdg6X!3xc%nm*Duh^jl*^stdXZZOz z9a}*pE!m-wzzYD?&`Ne_;@2>d)~AK~P+~wa3Jz-`HBG|XOKS0C+IoJfB$G|}2JUSAP%dzvcPhSANzv6l3&a`w&^M35jRDa4FWpK#Jf zKoSQDw2JKw>~3Zp1U14W?2Z~}0lOoplgPddf_Q&LeYhfE4|Wz{KXPI1_;Ykrcs>!t zrJ00Q1U0L2D|)WWV>Ycle^}dUGTV}MAX3qh4QM`~Z9$hc#jNs30)Ly_j+#eoMgCT| zhWlp6u>augHG*61(|CI=%DOlRU7^L3F+PguQa)`_#VSqB~YLRrp zR#67NDkco)@NQ&fg!BMN`#Hp?34X(9}d)PKV+NjU>; zN<-Wai6Db#ak}JOG+o*>!Ts!a0OE{w1W+NQ6_K;xlmLkZ`~<5p6?XMdHoNZbQ~6^P zZCy)`ZQ6t%d~nky%!u0Mzw!U(mqX`lSCAv9)z3}FT9{=qB_Kp`3Oa&C4{ zB#=$HrX^w;MMe(;Sawgx>=)})GcOh4Ob@72b8{%fl;-y4_E1x(sUhI=lCjC@$QqkU zSI016K?KY+JA{q2Ge%3i(~>tpP-Sk&l|@;`M{}3&$Ii&LdeZ z=2mSFFsV;nrfWC5bF?m#`eddYS`RzH%9JtQ#a@3@ST*@WjPM{(YmgoHIPZam(yi=@ zZ+Ag%RDz8(3gSkPY8M%xs2D6T4S^r904cl~5_-eX4UKvcf&*6OqM{5XEvYL={uA>1 zydH`fE{De+z%+x>&AN3y+E5_Y8SJ8c8NbjJa=OeyjWwCUW==5}c2}o4AO#H)-LM;G z&yN=^I6+&(wm8R2lILt_o33*^`on`8+X|V*@lSm4YsYsA+U7<&r_fboDu!OYrPEzL>2OicE3qs5C-^CAiD6DSFNoKiBOH2l zN^^+*JW6wWJegi7}e9e$NSe*<8IupEV zCiMX3k6||k6!vFG9${h=A$&y0?Q&L>7Zo6QkJ3_$<=4tHyQbJQ%^@2R;Ain3tTMi} zyVf{ksZVsYj086)n#ZkUE#No&L!B#LH?p^PV6P?6GgwvGRu>uxCpY@+5!OHXXsNY# zF}`N+0QJ<#N>~fxJJ7S`Hnw3&lP6do4B6W(k(wWUvcy@}HsG!sU=I)X)%aSft0T5j?_+~= zg$;es+k@2C<-e*GhzAPt2Qh_0$}`H>@YBpc!G5T{Wzkn?u@8yAo$MQ|MIAxkslL!+ znXbjc-o?h)QHA8`+Ut?#W zKQ3m+ETwn%t>qK)6v7|MIfb~FFJWWKTWCMpDBm5doBF%s;0pQ{8&gL_8^QVyjTd4M ziY|00#c&Pm4IG|cG*khC3f-l!VRF%sSF_xN?U8nA;8?&Ej|>gn%-Yp8^w3=!9=eq8 z?BBB|_R8x`fRp8{mGyFrq<-wARYg!zVCTz7Ke=^6--D-(%zL9V zK*C6Je?W*y6vCkoIliu?nJL|!%>yk14FNB*>6Ia@2vVVmHEHq8x>Q(cOxDbS8YBdY z<4U_S3+}qevAJxzZ*G0AX*maKd)cAbOgKCf6Xw7tHzpdw8)wEEJ4P;?n7m-5E&S0e z-gfE#oq0a1h;HfZnvF(hyE?Z-AqhiRB0mzI6~u?rF&<0(i0B)1Px7Q_9R4-#N+;Gk z1YHh?zq%TSzhQEF#Mcq^P5Gl8ephct$4CRt)UieW7%2QPM$J8qfB8#myuGQ@pJnhz zO9IT#VrWT?(kiUT0h9rHp+W&xuf|)`1u)I-jn#pQ5JA+0{TNBxHA6u_4k!#QrPQ_? zy-nUgq^UI)7>=|J);c|2XN=w00LrEsN2I;3ez4x(Q(F_Ns|i$Jfce;hvG^HUFu|W> zh~0TqENFdUHQ#SS{ekbVsE}8HIT1}jR4ncu{Uf28QvA+T-Ks)oXhML zw2IbaRDwW3eD;CbGG17uqfEPSb^-sddIT1r0xTy3vYQqanV>nCOrtX;tguihrP?B* zJP88y*1O7OW;3J}T?8OZK$WQhw?0V>fBW}s{o4zLbTw7y}10Y{DR14UP5vk#!kAPXuxY@bT^WpT5BwmwI_K)T%a~7t#ZpM6` z2H@=3m1xs3oA1FY3@LrkHJ(7EMG@uSDFodwga+lh5!)|*VEdK6AhsTde8p5`DngDF zqhhQe0CNoLk=fNLzTbWH=)!f^?H1qrzOVhmGg|PBNB=j^h#@l4j4ZkM-g4b_3rCOc z7T>ea0NXm+`4&7QwF)vk0P1emDZDY-#E}iRkJ&6{ETV5LAiWga)u>(@Kz;-ICs~X$ zF7!MaQyey76G*^ke#s@$c9asf$Iuqag8(fa^mDeq-xDK{;+;?;4zL)+%+?U@g z?@Q5}_}B1^26>$j@AEONtayA)C9F_YST$ZCzp$b-o?UEg zw149$i8vZ)h%&=aO*N3uW#TyFXCSN6lGWriV3m*<7KJ%bHYk;|FS`-;Lbkx{o8?{E zz4*iRQ98%8V$a*dUqPHY-Zb+!KM}q~o>iD%le293x4cfdng0YAxnl_)e zo90QH|A+E#>V?qEUxt18cFZmKnQs?!i_WA550C&RoL2F?(8eHyWe`l}5CNjx;bejm%Y$Eua@S(u?;HY{`M4f8b9-JQjh<092cjG#iC&pCP z0gI#wS~2zaVNc|JM(#yGpLp#qd6Hb8k@oatqGiV9E;%f9lnobvZj0m|t~FRB6U zK0$uyk94y4z{X3lq*}Zu+B$%GH}DV1dZ=+k84w51^$uhEG9t%?oq<4d_+0iL z9u2McqgwA@#C3n5m`Yi$hN%FO7@7e0;ar2LE7F4yMFLSzAIP^(Xi9O|;%q(kDP)*K zEQA60kWR1=H275itF14mr`g&&2!=MnevcOX5-9l_$jc|56y%2o&`x#!mwDOP(f2sz zKSAG5&VMrWuJWG{mAhTfe=@U&rTUPedqm|swy;$C2jdh1L|sj83r<^V$b$oz^ zYOyjnV1k7KRUo_w->yaDyW=?Q;WK1-qiCLNU`?h5(gYgF7glD-35pRzQ%GSV*r!#T zMD>`Mp^fVjiBuxNC->}Gdg%GYbI&E7NB>3J(EpQjsSLecE|r{{Cd4hZn_<+ykJ+Gd zSS7Xh`xFXjzI-3Mj+&i$>XTae^FTw2)c{8X&B%FL4ajr%EOwi2F=AEF0WXc0y6oJf z`m4;qP@6a~0B*Q#8c7YTd6kW|vgXdK6Mtp5)5J=j0*ORB<==%yk-~2#?(NLm z*^w(F|MQuj=Y3A&!Tv5Z4qOkL)PCg)G?#=%kXwG0@)S)3`8+X`gYiIN0idKLM6k9L zT{hev^g`msD>rp(tdrH!gL3!tzZ_s~ z;+eVIc>24?(JQ-Z9Ts|9(6?+#Z@gP(qd~qqJRa*R ztI8UULzE*-*J9zmT0J8TMnN0*zP{827ieQIBk$5rYA-;}A;dwFR*Kz#=1@n@96_s` z%Wv{H-cMNhu#^c^-YWP6{a_6s4x1tiFCv8yhkc5PU;Q&^ZitB|fkL?kgfV7h23=Cv zB0&O0__I6v3AUm^?PDlNBIX$>CTrbJJLa5$aP=nWb`4R$uaMAPmBB^Q_YFy5?iN~; zfY&=+E*qE%yrmT`>%QnkcWnNC8HC)y%Cm}Vt*#ccJ>m=wIQLcU0**TJf2`p_L3$8t>|4nZD-jWI8Qqnkk5L z+9~2M*rbTLNTgx-<;!FhehPQABGTsEcp*$xFtkDsf+QvaCtraHqmGFnbMlBY4QXjO z!crW$#;J&gsmy#}WnHbtLfmPqP?Sn%h1=%V$W`i*wg*JBR%s#3WbLu4o!#Uvf?GCG zG&oO?!2wjO*gH7@OQUE{)Y zUW6w02Q?|LQ-Y+4K|6`ONV6NcuQj-B_D$M|zej2T5JLGmd)B~mSWqxnK-yYvu3&Pw^Y0sdR|8=#o}&VS1b71*dA#VzHTffvd%gQzEB-%;-k zer)+!MTzIsTYh$xdT*|*2b;WB53_Z{@|~=N-;I+yJW`;czyY6LP!!l_+)95urUUL}AatpM zNHvZ~PSTf&R`>|IE3!9(AEWY5q=7lrHBZqFi3jb11 z(v#I2f(-F7wwD)2d;>bmq45veEKa*UhJfhR@$o z$4dXR{`~3sut#JPCw*cEWSH1XAS0)hCosGD@;{evP=1VZ!We_?7NJyFV(O{*Yz=Z0 zis>QLY_oxQMCs%~_9c>*6a|w=UH>!w$kOOpgOuwK|1dIy@`cq6=!AF9U9|?%(QF&>v}^5E@f{1}hm_Cl7(pnH&^)uH~+xNn@3&KilJ<@-yqJ z_@2I)o}nxsVwUCm$@U8mTZfb$OC7pY$XI{NVEw(&IQOU~U!?6fdp+`&LA_t?Z$1gy z*Q*))(BI*70%!MfadWE6f5j|n4Pt8X>|C_E_zc$(4vsz(C+Kwo{|4W91}mq~y>se3 z4T(Y+xtuzB`;~F-zYuaSPHjx!$kIq>cV!08#p#?~j-L#MX9d2XR~6(Rh}Sxb3$Q3; zN>deq4p84oTLer-gu^385RMw)oP| zN4yXWp9=PKFnmf%m`Vs?Xw6u&5|%_sz>LN?kkj!R?IXd#4(*S8S4sV29C}Mr|_H16Bcqzrq-_krY)6nPp?@Y#c?M6%hAoD(H7=S7X#+j12-cB2K)nH z?E3*$T*eB7W^Sd{??%(%cg3%|d4jA!7%oTj*B?lToRV$qVdecIS`*q3-q9cs>H%Tv znNjzyV(Y;>ggueoA+1KTu)KtyMxlQrUlk;8o#Kb=2rE0|+iF-rVF*%!uPrSpQVYO; z4QEn;aysZKu=UvB7J#RV?=OZkVgS0LV9*Z)!i|Ao09%Y7$QXdoC^{F;q#V`VfgM}2 zXgi2apZv&j1MhT+Q+$I1Ugc20*d1`Wx&!=!5O40Y=K{;RZ`kKNlmvdz^;UhC+pz~}CTWe1FevQ>ijv384 zIcA47&T3(^&8aVJwyQk%X<{pdz3&CGEyLU6SI&-?mcXSAu?tTHObOb2tbu8jFJef! z0N)6XC23$4z8x=iJba+h(!ytkbQv^xGgTJiHu(0doXChFg01{q7+}r7)px^XgDnP{ zuTacUC2J7|s={b4vy>Lrdo3C?Q`%zfLr;o`s`_#s)nJ(!Q|za`BQ5AFA;Obj4l)A? zqDVwI&=5Kz92PIwo&G8I&=x|-lpwYNY7gx^jKj5sjfNS4jxm|U3KP_l8FY-N4!+ff zx)x7Ml?%Z{78t~8S*?Z%&*Dn+Tu`vnp5*^khG*LfX3g&QhM}2;%ALJK`v-?kA3AjT zl@~|G!{M=}$XFyYW@%4_9AWq5rKxatY;kyKvG?9PKlK0`4h)BzlMVIpNH{@Ak`Ca% zCphrm=&x}V#VP?uP848KMZIQDp{offD26`WFq$*;U;k^ z1Ke|ng9ZYAEGp>igDFe)*`6ZX>}HuDZR5kjj;ca7AE2fR2*CZ?!^R(`GC=`UVHAK zME4G!Gc!G3z~&0(r|~oWg@So4Xh?GBut&EiliSl*??@(hV2+43ifAAb-Nbt(`zp&x z_R8{<#&|BuMI?wU-8 z3MNYQ7rNTZ*eCKLZqR2rXHLg#M_$=@nED5Awt>U}0_bSZCEt}ICc3xI6|4+a^aM== zq?S`!NCBGSBmv+)I!QQwG1`sUEoXWZY2ZBfXrXa}b?!(0^G3lsCxPo-y?DPS=0!m&Qw+O-G?;_WB;Ki5;A6M2BapD~_Ns-rueF6KJzXt> zt%Csl*4NeK6OPc~nL*%^=}RPq5h9nc(jeF&*y6u@4}I>S$L?|(J;tVvntCq?C=6a> zQ}UhO4)3$ut4kWz*d16QmS*_|;l%6 z^87)EzoJ3p%Zp`Jc_*Ox$TW%{x!LkUH2i*hLv_mDShXpYn%<1~gp@mIUq3$YsoOR_ z^Yh4Q`@+6c&WwI`&*hiz`E2yeQ+$p4_C?ORJejob>rBINTWfmhv6!$a&A+jIrW;mXpFNpIr>bM)B{ri0J89RGv%dx!Z5lnCP)4{{QRr-(vnrz6Y6+@Of#FJ?uRemk68qdoaDLCM zh=vvQ4k>piq8`t$XUngLR@^M>$@^t?2(raADXV7fC1W@#L}(_q37gPRx#wx?2qgQ-;8otw99+l()Eiff^-duwOs*6!H0 zN_(>P*=O3~@wR84ZB3@XKiLtTn22^zo(R$mJA}5e0WG2t`kNR(Sx%=cS$ zjg?>}z16{YcNJc|cn=(4#}_XyOrOU_3f7H|t}96YJNr#?%kJ0M?aBKVH=liWXMg&| z{;v1HSflr?7OR#bju07BWGz%JC1yxpi?Y8I#5v`lvVqFA-v$(kYA93pSFqyZg5BVS z5R5v&vbQwX)hS9_Yjb-`dnDW#tP9lny&eo!t-DG9y!>F=wA0Za_G3E&RD}gO(+#o# zP=Ji=iVGGZ*rI#PQPHDLrgl$MotdIupu%L*;H0B#z zokSLmHzG+2R04V@Mf%J*@e?Q-tqF$CgMsfV0NToRGSjbqu7LOoS#2w5e-k-Cr(8hf z-8np+$M{=WywgX+ZSuI$(U^XPaa~jqjW^AE;^O{=;E?bLEg5krOkvlcHy7b6*MMZx zh$19O1bqe%JKqbLR;74>Nc>p5&&KDEiGk{0t077AWZ@{EfI=XbTdUMY(G-B zf}_D~7LNDU+{zfkA;48F7BPmSGbdjF8WF$GT@749y)WbsAzhr;?L|{S$7dsGAkm1p zXzRjym1i|_KFC;Gt1a)yM#fr<<;dwn=I(CO@aZGqh`F0iZrcjCPfzbCNZ(a3e~f(h z8^Ub^`{IfCzCjvH4$4ja{3awSuxIb)8&* znsprI$NU|6Po9x%68(o@TK zEPouRyG`kb(^Fa0U3yBlfnzS^Uzgi>bV>DP8)#>s z`V{YVl_-bTG4DR`E{gInmDA2G>IVh>%cO?MATI9Z!@DgJni^KfBCa9$qK=y=QA+$U zRae3ZAdXEGj!M|mq*Wxrvkm+^!LU6VADfiAm(<(i>l2q)wwdv;>czU&}Ds!gcTg|fv1fWFCzv~?J8I><=U84x{) zOu24lG@ek*(DYdz9VX&cErKaZIBAhY(`I9e&27U_VPea4g>WD!6|90Hd>R5GR!cYo zkh+cz%}k`Q18$j3-NRz(?~zJ_+|%h-KEH0mhIOw^Bu_mx$^Jswo59@u;@Xjp^g8DsGz{NjM`0M!G_>iHB3a4pcbiUr&`7*=2n5tg?fKud#hD|9ys&@>v>yQ}@y z3ZK<(ttc-vk*7j;cDvxMpqO;ZeuG^fA+Wq-f25$R%d%e1h3(+&;;r@jXL=0eW9GU2Irk&f+wIx*2-X@oLWh;dwekab z^{>h;&(tRyDe8+oLS6^|7%LV2F=*vz&m4Ft(`XMe?1;aicCcP{6mkXjJ{ir4lf53z z$&@yH{uV~tq=3HwvqgY5PCnn86q7vPcQQ;42m1)COCM!FNWX?t{+6Y0(vyvV5(Ow` zQ@v;T538K>s}Zx=E6(|7$tvgkYMk18an3g>k$6b91TstE*OI(oTS;}rL}0>JBb!4> z8f+A(t`iN}ON2GIJ z7?7FSV9Ib^Bp}N&T9ks+%({=rm!2gfK+f-pp}bVJZtF>tx(397w(`d z;+xwo@DsHK*Q{%(+!Y(%H#oF^n0!Rza1cepV-|7=wNE!X8*}|cQ}u~3oI}!4L}MF< zk7!uOpFMHs^B zqKymr&=ezhg0w*~BCzl6okU_2bM;VM6Gp_5@qA!B1k}vrFgd>=yhP(>%}dnWUspHK zl5rBThxhCyCsA*=<|IOdb;e0Vz1Mn&nRCCGIbrq)8&)SVbMC{;`LdiFU#7W1xW?n` zS?qJ|3Yc8LrqD=!VwiK$UC3?tM~~omjk!)tzpC8Hpi*aiJ4~s^5woURxCQ}XqG%J+ zlVtF`h-eIkfCCc7BRLbuU_>A$Ks$sf>CLg-QFWbKJJ?><9Ik1b4nI2Xbf3DI+}vbX zYlsbBJlyieUVqCumey2rRa>p8+*9rG*$Vv0;_?>XXeu0U@U~9+y+zH*in8vOsTuS? zFG51BO)HQRn;@^>rGMXw6^ko?Ji*mgEa`6kIrs(kY6lN&T!?|jc1tP%MUra_%<7O1 zkaNfV*-E*eyId%#thg4n$_H^_lQ@*uKmRoTe$SvQWLOJkCX`pN7MbDke z74Z^aH#nmEm;nq~nl{vc$12wSi3|WmQx2On%ptE z3Pvj28e@GN<&P;1#OYCu1>RBRXC{0E@uQ0%;&zI~rU-OU%WT064$h zRY>b8!2)fpP51I@bQT~wlCJS>xj)DezZD6zg05Huj6BFd{(>S*JmT>2vsq`Z(D4l*hg< z>hq^rpV}kqE4R@8!tT9A?k@@(fnM@u*lhd6Nw1Z{V+;#$5MuF&YzMH51B-x(1v(PD z&5BfR7Bk@8F6N@mgbscgx1J#Iq46SzGAoYsw$LN;5vNmt5w#y^u@$Gh zoYkv_Dlq_J^-}cD=YRz{5HED#ID&izBI<|8+G>SF!+|a%&tQ_2#v=g+!skeA_jaEz z#2x5QMK*~L!9*mt3^qf}n10=dKG<{ zKK~n5v6p;!G{{)HN6=eN##F3VL>?er0DwzGeu7$FunsBZBb(Kl^T#> zw5we5B$~3%p&E#p_(G3fmdQ`SI+_7;yF*FkFPY(393J*W*T!3G+RWc(-|~eM&Cd3!>e`?;(pwwr z?U?XZc&&rzht^K4ke`Va62OY`ixu?@#*jR5@VH}(dj=>%+Bsi_K0OW0qf7BA7vAq+ z1+cP6<-ub@SPcT>Spj&h$W4yjpr8QEad3o_a|l6XtJDC+imaxQh#~a`Q+(c)lubFZk<(FR{ zXqO4x4x=o})qqx?&@OjiF4GzP_X?q1a*-;7*pGAXeX(-;JKiIODB6?nf&Q%Wyr+RW zT)i$Kaj1`(w!s^C%PMvA+oq!uWxs*1pnITSFCz+>&?Z3rQZy^>iwFbEoOvwO8fE?g z6XXl1Tjj*51TdBxuc7Rj!fL53;cQGzdjc zL?fXNAnn5WMnew=3Q2w}#!g3hBJo5bk>nN0TOQ?Q#b$L-C z&Xg{M{1dq*EDhJdkE|N5r8f_oo^~Pw)4!tB)!?$q1}oC1lP8Q0T;k2T7;;5-nk5xz z$jrJK?VmlPdt0KZKrP!j+`i$$$-|qc2B+Ak($9t4x_1qxt`1!|#SW#f=|^hQCufIm zN&jZ+)zfv0Y~R19FSWLuf`*i!rQ^uCK`s5DoS)_HYNUHFfzytx%GgBR@w_66rqwdS zA{j3?Tq8D9qnX4WL1;NBt}wMJ1*lJcPI+H5J8pg*vANbEs~1VX!UB+^WH#53%>7E8 z*ZWR$J=U#?jx$4d%lP_JD{E3!tkB-Kej_IPfuDPV0Et_wc3kaV+d&h&JIEH@sWa2f znDalsHT{A$G%D6+lw=Mx!H>y4+sdHub78v1KD#!+I7_+XdMVamrd01a;Q+9Fb&&BO z9GaNybQS;t13lM1r^vs|>&>epdhYktx{+|n3cf=l#Q_SnLO(c2xW;PB8(|^@A^5SQ zw6xLDFbEW$EqA4-2$FY8QK+~$5+Z%-7~44U>gHM5Dy)URIYI8KYEP4;^s8{M;jaFw+NXb0Z7;)`5#PEq(1id1J0(NvP5j|se^BJ9q zv2(s0jMWCF`ue56I&^k2dG^rg`HuK*Q*<)ivLPDX&@!}GxI3PaHT;(u z8T2vbTRE}}GET^*6*5lPiAiPw-Evs&jlUCU6xgRVbju*L_w3FHUQSI$8nI*kR_w-e zxSse;=rlmSm_I4!!ZP%(ts{imXddN%I(7__w|UvDoV)x4G&pcgozOC}ozOC}(cKZx zcrGF=-u2y)su_1bcKkSvQ8Rl?IhP~fn*mMbAJDvBACQ|R?w(kFgB^$Us!q4f6oPz7 zsly?S5=3-~RkcTMjE?|K10^l}kHGRIz7C58mo@t4GYw-py)&t=2$V>5d{F)@)tCjArE46aKpuQ6*$KTRdt8ovjUfMVhr$xdWIWP$<%11(a?#(}rU zklKv$#zNc^)i)(Rwpo4FMt;rGW%YcGSn-!FeV)%_PGD@*&)|;L3V4u+w3%UF!Vm1A zkjIO#%_s@Lc1aG?jsw6CQ{+Zc0vb%p^r}(jWSC;rfQtV+ZJ+?L4RN(I7%%FzR905i zS}mWNyW)zuPi?s5@TS9;lsEMA+d2lSs(m$0Rq0~(5g>nFPaj}6(^tn#Gv2F#fkK$? zhtD44%2s2_F_3o*`IK+M-b#@`KJx9sp$#^tVpz@$N@)qg9mxxFIw*crvzZhQ&2#Pp z<8*Zu=>Z7$u?eMvzLgW=M}^XXPKf`y5u}g4gy+`k5p`okcvqjO7nA`S-o)tq(r5VkXnnhj;2Mt-dIy`czK!pUBo1GgFpd#X7U) znfp*Ktk2>)m>Phqxev3!u5`saM6^^n#gA%}fpj%-B*xn392ty(YKLnS)4y!(NMrST`&9}ztNiRcW)Y{(vx z){hnoz4oqrtv^$T^ieSiYWN@;kJTZrg~m~iE6{$%%$# z7?UedwOqdWib6u|#6sA)KpRF%AWZ>VEO2o9*0VNl+q~tITQ}|4aml^W&eM)P@ifRZ z1f2%+9?7(a%`2z3EAMky`MRu+ zJ?d`pc4hfhT6y#G2{uDE4@14eR`*|t7^f1f8Dt06MLVr?a#ql5#@V{8lqoJpsk_V# zwW)-a5U9f_K?QXxyxpN8RWlesOR^oj;DUpr+q#?=9PAwO6-{pjYT0tnU3cBX`cAnd zao0UltbN_i)@#H+^t6eUQ&tM>RQ2e|tVBmF&&r*OJ_ncYTIGcNd=-yV6MQsl)p78_ zI*9=UGZ$}JJwRSYP)26D0n*4~*Z{y*>@0Rd8WgfZlb(A*S`5jL1c$u_Q>AF$M@}C+}l4kJ~=u%IVt#-wU!~)DInf~))KVTkC*r*vgYx7zl4s^ z`-B}W`+mULu_)iOllfuiA-RV4wyIo%5tQZ*=CYN}Hb5jpz?*<>Lj)cmIgrZ4QK#<4 ztHWA^m1cJ~Vp$N&k~g42q9sTfBLskhW;D!Go=bP%cv<>6cI~B$pSX-QPkcOa!`L{! GjsJgr>e@d5 literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/style/fonts/texte/Questrial.zip b/nationchains/www/adminapx/static/style/fonts/texte/Questrial.zip new file mode 100644 index 0000000000000000000000000000000000000000..8869291db3d206652ab3c7aa6460b8607732981a GIT binary patch literal 152858 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�xh22ulm$4#j;4GBXT18Bt#C@zbJbNZ64ej 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 + + + + + + + + + + + + Questrial Regular Specimen + + + + + + +

+ + + +
+ + +
+ +
+
+
AaBb
+
+
+ +
+
A​B​C​D​E​F​G​H​I​J​K​L​M​N​O​P​Q​R​S​T​U​V​W​X​Y​Z​a​b​c​d​e​f​g​h​i​j​k​l​m​n​o​p​q​r​s​t​u​v​w​x​y​z​1​2​3​4​5​6​7​8​9​0​&​.​,​?​!​@​(​)​#​$​%​*​+​-​=​:​;
+
+
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
10abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
11abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
12abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
13abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
14abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
16abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
18abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
20abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
24abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
30abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
36abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
48abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
60abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
72abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
90abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ
+ +
+ +
+ + +
+ + +
+
◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼◼body
+
body
+
body
+
body
+
+
+ bodyQuestrial Regular +
+
+ bodyArial +
+
+ bodyVerdana +
+
+ bodyGeorgia +
+ + +
+ + +
+ +
+

10.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

11.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

12.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

13.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+ +
+
+
+

14.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

16.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

18.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+ +
+ +
+ +
+
+

20.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+
+
+

24.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+
+ +
+ +
+ +
+
+

30.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+
+
+ +
+ + +
+
+

10.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

11.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

12.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

13.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+ +
+ +
+
+

14.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

16.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+

18.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+ +
+
+ +
+ +
+
+

20.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+
+
+

24.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+
+ +
+ +
+ +
+
+

30.Aenean lacinia bibendum nulla sed consectetur. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus. Nullam id dolor id nibh ultricies vehicula ut id elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Nulla vitae elit libero, a pharetra augue.

+
+
+ +
+ + +
+ +
+ +
+ +
+

Lorem Ipsum Dolor

+

Etiam porta sem malesuada magna mollis euismod

+ + +
+
+
+
+

Donec sed odio dui. Morbi leo risus, porta ac consectetur ac, vestibulum at eros. Fusce dapibus, tellus ac cursus commodo, tortor mauris condimentum nibh, ut fermentum massa justo sit amet risus.

+ + +

Pellentesque ornare sem

+ +

Maecenas sed diam eget risus varius blandit sit amet non magna. Maecenas faucibus mollis interdum. Donec ullamcorper nulla non metus auctor fringilla. Nullam id dolor id nibh ultricies vehicula ut id elit. Nullam id dolor id nibh ultricies vehicula ut id elit.

+ +

Aenean eu leo quam. Pellentesque ornare sem lacinia quam venenatis vestibulum. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus.

+ +

Nulla vitae elit libero, a pharetra augue. Praesent commodo cursus magna, vel scelerisque nisl consectetur et. Aenean lacinia bibendum nulla sed consectetur.

+ +

Nullam quis risus eget urna mollis ornare vel eu leo. Nullam quis risus eget urna mollis ornare vel eu leo. Maecenas sed diam eget risus varius blandit sit amet non magna. Donec ullamcorper nulla non metus auctor fringilla.

+ +

Cras mattis consectetur

+ +

Aenean eu leo quam. Pellentesque ornare sem lacinia quam venenatis vestibulum. Aenean lacinia bibendum nulla sed consectetur. Integer posuere erat a ante venenatis dapibus posuere velit aliquet. Cras mattis consectetur purus sit amet fermentum.

+ +

Nullam id dolor id nibh ultricies vehicula ut id elit. Nullam quis risus eget urna mollis ornare vel eu leo. Cras mattis consectetur purus sit amet fermentum.

+
+ + +
+ +
+ + + + +
+
+
+ +

Language Support

+

The subset of Questrial Regular in this kit supports the following languages:
+ + Albanian, Basque, Breton, Chamorro, Danish, Dutch, English, Faroese, Finnish, French, Frisian, Galician, German, Icelandic, Italian, Malagasy, Norwegian, Portuguese, Spanish, Alsatian, Aragonese, Arapaho, Arrernte, Asturian, Aymara, Bislama, Cebuano, Corsican, Fijian, French_creole, Genoese, Gilbertese, Greenlandic, Haitian_creole, Hiligaynon, Hmong, Hopi, Ibanag, Iloko_ilokano, Indonesian, Interglossa_glosa, Interlingua, Irish_gaelic, Jerriais, Lojban, Lombard, Luxembourgeois, Manx, Mohawk, Norfolk_pitcairnese, Occitan, Oromo, Pangasinan, Papiamento, Piedmontese, Potawatomi, Rhaeto-romance, Romansh, Rotokas, Sami_lule, Samoan, Sardinian, Scots_gaelic, Seychelles_creole, Shona, Sicilian, Somali, Southern_ndebele, Swahili, Swati_swazi, Tagalog_filipino_pilipino, Tetum, Tok_pisin, Uyghur_latinized, Volapuk, Walloon, Warlpiri, Xhosa, Yapese, Zulu, Latinbasic, Ubasic, Demo

+

Glyph Chart

+

The subset of Questrial Regular in this kit includes all the glyphs listed below. Unicode entities are included above each glyph to help you insert individual characters into your layout.

+
+ +

&#13;

+

&#32;

+

&#33;

!
+

&#34;

"
+

&#35;

#
+

&#36;

$
+

&#37;

%
+

&#38;

&
+

&#39;

'
+

&#40;

(
+

&#41;

)
+

&#42;

*
+

&#43;

+
+

&#44;

,
+

&#45;

-
+

&#46;

.
+

&#47;

/
+

&#48;

0
+

&#49;

1
+

&#50;

2
+

&#51;

3
+

&#52;

4
+

&#53;

5
+

&#54;

6
+

&#55;

7
+

&#56;

8
+

&#57;

9
+

&#58;

:
+

&#59;

;
+

&#60;

<
+

&#61;

=
+

&#62;

>
+

&#63;

?
+

&#64;

@
+

&#65;

A
+

&#66;

B
+

&#67;

C
+

&#68;

D
+

&#69;

E
+

&#70;

F
+

&#71;

G
+

&#72;

H
+

&#73;

I
+

&#74;

J
+

&#75;

K
+

&#76;

L
+

&#77;

M
+

&#78;

N
+

&#79;

O
+

&#80;

P
+

&#81;

Q
+

&#82;

R
+

&#83;

S
+

&#84;

T
+

&#85;

U
+

&#86;

V
+

&#87;

W
+

&#88;

X
+

&#89;

Y
+

&#90;

Z
+

&#91;

[
+

&#92;

\
+

&#93;

]
+

&#94;

^
+

&#95;

_
+

&#96;

`
+

&#97;

a
+

&#98;

b
+

&#99;

c
+

&#100;

d
+

&#101;

e
+

&#102;

f
+

&#103;

g
+

&#104;

h
+

&#105;

i
+

&#106;

j
+

&#107;

k
+

&#108;

l
+

&#109;

m
+

&#110;

n
+

&#111;

o
+

&#112;

p
+

&#113;

q
+

&#114;

r
+

&#115;

s
+

&#116;

t
+

&#117;

u
+

&#118;

v
+

&#119;

w
+

&#120;

x
+

&#121;

y
+

&#122;

z
+

&#123;

{
+

&#124;

|
+

&#125;

}
+

&#126;

~
+

&#160;

 
+

&#161;

¡
+

&#162;

¢
+

&#163;

£
+

&#164;

¤
+

&#165;

¥
+

&#166;

¦
+

&#167;

§
+

&#168;

¨
+

&#169;

©
+

&#170;

ª
+

&#171;

«
+

&#172;

¬
+

&#173;

­
+

&#174;

®
+

&#175;

¯
+

&#176;

°
+

&#177;

±
+

&#178;

²
+

&#179;

³
+

&#180;

´
+

&#181;

µ
+

&#182;

+

&#183;

·
+

&#184;

¸
+

&#185;

¹
+

&#186;

º
+

&#187;

»
+

&#188;

¼
+

&#189;

½
+

&#190;

¾
+

&#191;

¿
+

&#192;

À
+

&#193;

Á
+

&#194;

Â
+

&#195;

Ã
+

&#196;

Ä
+

&#197;

Å
+

&#198;

Æ
+

&#199;

Ç
+

&#200;

È
+

&#201;

É
+

&#202;

Ê
+

&#203;

Ë
+

&#204;

Ì
+

&#205;

Í
+

&#206;

Î
+

&#207;

Ï
+

&#208;

Ð
+

&#209;

Ñ
+

&#210;

Ò
+

&#211;

Ó
+

&#212;

Ô
+

&#213;

Õ
+

&#214;

Ö
+

&#215;

×
+

&#216;

Ø
+

&#217;

Ù
+

&#218;

Ú
+

&#219;

Û
+

&#220;

Ü
+

&#221;

Ý
+

&#222;

Þ
+

&#223;

ß
+

&#224;

à
+

&#225;

á
+

&#226;

â
+

&#227;

ã
+

&#228;

ä
+

&#229;

å
+

&#230;

æ
+

&#231;

ç
+

&#232;

è
+

&#233;

é
+

&#234;

ê
+

&#235;

ë
+

&#236;

ì
+

&#237;

í
+

&#238;

î
+

&#239;

ï
+

&#240;

ð
+

&#241;

ñ
+

&#242;

ò
+

&#243;

ó
+

&#244;

ô
+

&#245;

õ
+

&#246;

ö
+

&#247;

÷
+

&#248;

ø
+

&#249;

ù
+

&#250;

ú
+

&#251;

û
+

&#252;

ü
+

&#253;

ý
+

&#254;

þ
+

&#255;

ÿ
+

&#338;

Œ
+

&#339;

œ
+

&#376;

Ÿ
+

&#710;

ˆ
+

&#732;

˜
+

&#8192;

 
+

&#8193;

+

&#8194;

+

&#8195;

+

&#8196;

+

&#8197;

+

&#8198;

+

&#8199;

+

&#8200;

+

&#8201;

+

&#8202;

+

&#8208;

+

&#8209;

+

&#8210;

+

&#8211;

+

&#8212;

+

&#8216;

+

&#8217;

+

&#8218;

+

&#8220;

+

&#8221;

+

&#8222;

+

&#8226;

+

&#8230;

+

&#8239;

+

&#8249;

+

&#8250;

+

&#8287;

+

&#8364;

+

&#8482;

+

&#9724;

+

&#64257;

+

&#64258;

+

&#64259;

+

&#64260;

+
+
+ + +
+
+ + +
+ +
+ +
+
+
+

Installing Webfonts

+ +

Webfonts are supported by all major browser platforms but not all in the same way. There are currently four different font formats that must be included in order to target all browsers. This includes TTF, WOFF, EOT and SVG.

+ +

1. Upload your webfonts

+

You must upload your webfont kit to your website. They should be in or near the same directory as your CSS files.

+ +

2. Include the webfont stylesheet

+

A special CSS @font-face declaration helps the various browsers select the appropriate font it needs without causing you a bunch of headaches. Learn more about this syntax by reading the Fontspring blog post about it. The code for it is as follows:

+ + + + @font-face{ + font-family: 'MyWebFont'; + src: url('WebFont.eot'); + src: url('WebFont.eot?#iefix') format('embedded-opentype'), + url('WebFont.woff') format('woff'), + url('WebFont.ttf') format('truetype'), + url('WebFont.svg#webfont') format('svg'); + } + + +

We've already gone ahead and generated the code for you. All you have to do is link to the stylesheet in your HTML, like this:

+ <link rel="stylesheet" href="stylesheet.css" type="text/css" charset="utf-8" /> + +

3. Modify your own stylesheet

+

To take advantage of your new fonts, you must tell your stylesheet to use them. Look at the original @font-face declaration above and find the property called "font-family." The name linked there will be what you use to reference the font. Prepend that webfont name to the font stack in the "font-family" property, inside the selector you want to change. For example:

+ p { font-family: 'WebFont', Arial, sans-serif; } + +

4. Test

+

Getting webfonts to work cross-browser can be tricky. Use the information in the sidebar to help you if you find that fonts aren't loading in a particular browser.

+
+ + +
+ +
+ +
+ +
+ + \ No newline at end of file diff --git a/nationchains/www/adminapx/static/style/fonts/texte/questrial-regular-webfont.woff b/nationchains/www/adminapx/static/style/fonts/texte/questrial-regular-webfont.woff new file mode 100644 index 0000000000000000000000000000000000000000..50c93061d3ee480cfad59c294351a1dfe110cae4 GIT binary patch literal 26760 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 tx9CYZQHhO+qP}nwr$(?d5-OUzk8G2Y=5-V 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

Nd8W~e| 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 E54OpWH2?qr literal 0 HcmV?d00001 diff --git a/nationchains/www/adminapx/static/tpl/apxmodal_en.mustache b/nationchains/www/adminapx/static/tpl/apxmodal_en.mustache new file mode 100644 index 0000000..d75e97a --- /dev/null +++ b/nationchains/www/adminapx/static/tpl/apxmodal_en.mustache @@ -0,0 +1,29 @@ + +

+ + + + + diff --git a/nationchains/www/adminapx/static/tpl/articleeditor_en.mustache b/nationchains/www/adminapx/static/tpl/articleeditor_en.mustache new file mode 100644 index 0000000..66b94e0 --- /dev/null +++ b/nationchains/www/adminapx/static/tpl/articleeditor_en.mustache @@ -0,0 +1 @@ +

Set up editorjs here

\ No newline at end of file diff --git a/nationchains/www/adminapx/static/tpl/listofarticle_en.mustache b/nationchains/www/adminapx/static/tpl/articleslist_en.mustache similarity index 100% rename from nationchains/www/adminapx/static/tpl/listofarticle_en.mustache rename to nationchains/www/adminapx/static/tpl/articleslist_en.mustache diff --git a/nationchains/www/adminapx/static/tpl/emailregisterhtml_en.mustache b/nationchains/www/adminapx/static/tpl/emailregisterhtml_en.mustache new file mode 100644 index 0000000..f500d24 --- /dev/null +++ b/nationchains/www/adminapx/static/tpl/emailregisterhtml_en.mustache @@ -0,0 +1,17 @@ +

Your account is register

+

Please find your confidential information in a safe space

+

Your alias: {{alias}}

+

Your passphrase: {{passphrase}}

+

Your public key that you can share with anyone:

+ +

Your private key that you keep secret and use to proove you own the public key:

+ +{{#trustedtribe}} +

Thanks to trust us to keep your private key, + we'll be able to send back to this email address in case you need it

+{{/trustedtribe}} +{{^trustedtribe}} +

You decide to keep secret this private key, + Please save it in a safe place that noone else than you can access to proove you own it

+{{/trustedtribe}} +

Never share with someone else your privbatekey if someone can access, it will be possible to usurp your identity.

\ No newline at end of file diff --git a/nationchains/www/adminapx/static/tpl/emailregistertxt_en.mustache b/nationchains/www/adminapx/static/tpl/emailregistertxt_en.mustache new file mode 100644 index 0000000..6262e9f --- /dev/null +++ b/nationchains/www/adminapx/static/tpl/emailregistertxt_en.mustache @@ -0,0 +1,17 @@ +Your account is register \n\r +Please find your confidential information in a safe space\n\r +Your alias: {{alias}}\n\r +Your passphrase: {{passphrase}}\n\r +Your public key that you can share with anyone:\n\r + +Your private key that you keep secret and use to proove you own the public key:\n\r + +{{#trustedtribe}} +Thanks to trust us to keep your private key, +we'll be able to send back to this email address in case you need it\n\r +{{/trustedtribe}} +{{^trustedtribe}} +You decide to keep secret this private key, +Please save it in a safe place that noone else than you can access to proove you own it\n\r +{{/trustedtribe}} +Never share with someone else your privbatekey if someone can access, it will be possible to usurp your identity.\n\r \ No newline at end of file diff --git a/nationchains/www/adminapx/static/tpl/nginx.conf.mustache b/nationchains/www/adminapx/static/tpl/nginx.conf.mustache new file mode 100755 index 0000000..9f86395 --- /dev/null +++ b/nationchains/www/adminapx/static/tpl/nginx.conf.mustache @@ -0,0 +1,34 @@ +user {{{sudoerUser}}}; +worker_processes auto; +error_log {{{ }}}/var/log/nginx/error.log notice; +pid /var/run/nginx.pid; +#include /etc/nginx/modules-enabled/*.conf; + +events { + worker_connections 1024; +} +http { + include /etc/nginx/mime.types; + default_type application/octet-stream; + + log_format main '[$time_local]##"$http_x_forwarded_for"##"$request" ' + '"$http_user_agent"'; + + log_format mainold '$remote_addr - $remote_user [$time_local] "$request" ' + '$status $body_bytes_sent "$http_referer" ' + '"$http_user_agent" "$http_x_forwarded_for"'; + log_format trace '$remote_addr - $remote_user [$time_local] ' + '$host "$request" $status $body_bytes_sent ' + '"$http_referer" "$http_user_agent" ' + '"$http_x_forwarded_for" $request_id'; + access_log /var/log/nginx/access.log main; + sendfile on; + keepalive_timeout 65; + gzip on; + ## + # Virtual Host Configs + ## + {{#nginx.include}} + include {{{.}}}; + {{/nginx.include}} +} diff --git a/nationchains/www/adminapx/nginx/modelwebsite.conf.mustache b/nationchains/www/adminapx/static/tpl/nginxmodelwebsite.conf.mustache similarity index 100% rename from nationchains/www/adminapx/nginx/modelwebsite.conf.mustache rename to nationchains/www/adminapx/static/tpl/nginxmodelwebsite.conf.mustache diff --git a/nationchains/www/adminapx/static/tpl/nginxproxyparams.mustache b/nationchains/www/adminapx/static/tpl/nginxproxyparams.mustache new file mode 100644 index 0000000..1b73dde --- /dev/null +++ b/nationchains/www/adminapx/static/tpl/nginxproxyparams.mustache @@ -0,0 +1,11 @@ + proxy_set_header Host $http_host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header X-Forwarded-Proto $scheme; + client_max_body_size 10m; + client_body_buffer_size 128k; + proxy_connect_timeout 90; + proxy_send_timeout 90; + proxy_read_timeout 90; + proxy_buffers 32 4k; + proxy_set_header X-NginX-Proxy true; \ No newline at end of file diff --git a/nationchains/www/adminapx/static/tpl/pagancreate_en.mustache b/nationchains/www/adminapx/static/tpl/pagancreate_en.mustache index f2a59dd..8ff16aa 100644 --- a/nationchains/www/adminapx/static/tpl/pagancreate_en.mustache +++ b/nationchains/www/adminapx/static/tpl/pagancreate_en.mustache @@ -1,57 +1,130 @@
-
+

How it works

-

Alias is a unique string that humainly help to match a PublicKey to check existing alias

- - GET /api/odmdb/idx/pagans/pagans_alias_all.json with a correct headers - RESULT - data:{alias:publicKey} - -

- Run -

+

+ Mandatory: apixtrib header have to set with:
+ * {xalias,xhash,xdays,xtribe,xlang,xapp}
+ * xhash is a detached signature done with public and private key of message: 'alias_xdays' where xdays is a time + stamp + a xhash has an elapse of 24hours after it has to be recreate.
-

-
+ We need in local storage auth for this example {alias,passphrase,privatekey, publickey} to be able to create a + detached signature
+ On the server side we check that signature xhash of alias_timestamp is valid with the public key +

+

Alias is a unique string that humainly help to find a PublicKey that is the real identity. + To get the list of existing alias

+ +

+ + GET nationchains/pagans/idx/alias_all.json -> data:{alias:{alias:publicKey}} + +

To allow trustable Tribe to store the Private and Passphrase Key, you get from the townId_all.json key:tribes

+ + + GET /nationchains/towns/idx/townId_all.json -> data:{townId:{tribes:[list of tribeId inside a town]}} + + +
+
-

A decentralized Identity

-

apXtrib allow you to create keys to identify yourself with a universal alias

-
- - -
-
- - -
-
- - -
-
- -
-
- -
- - +

Am i authenticated to api?

+ + + GET 'api/pagans/isauth' -> status 200 : Well authenticated with alias, status 400: not authenticated + +
+

Logout

+ +
+

I proove that i own this alias

+
+ + +
+
+ + +
+ + +
+

Create a decentralized Identity

+

apXtrib allow you to create keys to identify yourself with a universal alias

+
+ + +
+
+ + +
+
+ + +
+ +
+
+
+
+ + +
+
+
+ +
+ + +
-
-

Download your keys at least PrivateKey this have to save in a secret place

- - -
-
- -
+
+
+

Download your keys at least PrivateKey this have to save in a secret place

+ + +
+
+ +
-
\ No newline at end of file +
\ No newline at end of file diff --git a/nationchains/www/adminapx/static/tpl/loginout_en.mustache b/nationchains/www/adminapx/static/tpl/paganloginout_en.mustache similarity index 100% rename from nationchains/www/adminapx/static/tpl/loginout_en.mustache rename to nationchains/www/adminapx/static/tpl/paganloginout_en.mustache diff --git a/nationchains/www/adminapx/static/tpl/townsetup_en.mustache b/nationchains/www/adminapx/static/tpl/townsetup_en.mustache new file mode 100644 index 0000000..b6bf9df --- /dev/null +++ b/nationchains/www/adminapx/static/tpl/townsetup_en.mustache @@ -0,0 +1,50 @@ +
+
+

Setup a new town

+

This form let you start joining a nation.

+

For dev you can stay like this and use http instead of https. To join a nation you need:

+
+
+
+
+
+ +
+ + +
+
+
+ +
+ +
+
+
+ +
+ +
+
+
+ +
+ +
+
+
+ +
+
+
+
\ No newline at end of file diff --git a/nationchains/www/adminapx/townconf.json b/nationchains/www/adminapx/static/tpldata/ASUPtownconf.json similarity index 89% rename from nationchains/www/adminapx/townconf.json rename to nationchains/www/adminapx/static/tpldata/ASUPtownconf.json index 16655c2..c021305 100644 --- a/nationchains/www/adminapx/townconf.json +++ b/nationchains/www/adminapx/static/tpldata/ASUPtownconf.json @@ -1,9 +1,9 @@ { "nationId": "ants", - "townId": "wall", - "IP":"213.32.65.213", + "townId": "usbfarm", + "IP":"192.168.1.1", "tribeId":"ndda", - "dns": ["wallant.ndda.fr"], + "dns": ["wallant.ndda.fr","adminapx"], "mayorId":"philc", "passphrase":"", "api": { diff --git a/nationchains/www/adminapx/static/tpldata/initconf.json b/nationchains/www/adminapx/static/tpldata/initconf.json new file mode 100644 index 0000000..055258d --- /dev/null +++ b/nationchains/www/adminapx/static/tpldata/initconf.json @@ -0,0 +1,31 @@ +{ + "dns": ["adminapx"], + "api": { + "port": 3020, + "languages": ["en", "fr"], + "exposedHeaders": ["xdays", "xhash", "xalias", "xlang", "xtribe", "xapp"], + "nationObjects": [ + "schema", + "blocks", + "nations", + "towns", + "tribes", + "pagans" + ], + "unittesting": ["middlewares", "models", "routes", "nationchains"], + "appset": { "trust proxy": true }, + "bodyparse": { + "urlencoded": { + "limit": "50mb", + "extended": true + }, + "json": { "limit": "500mb" } + } + }, + "nginx": { + "restart": "sudo systemctl restart nginx", + "worker_connections": 1024, + "include": ["/etc/nginx/conf.d/*.conf"], + "pageindex": "index_en.html" + } +} diff --git a/nationchains/www/adminapx/static/tpldata/loginout_en.json b/nationchains/www/adminapx/static/tpldata/paganloginout_en.json similarity index 100% rename from nationchains/www/adminapx/static/tpldata/loginout_en.json rename to nationchains/www/adminapx/static/tpldata/paganloginout_en.json diff --git a/nationchains/www/adminapx/static/tpldata/conf_en.json b/nationchains/www/adminapx/static/tpldata/setup_en.json similarity index 100% rename from nationchains/www/adminapx/static/tpldata/conf_en.json rename to nationchains/www/adminapx/static/tpldata/setup_en.json diff --git a/nationchains/www/adminapx/tribeconf.json b/nationchains/www/adminapx/static/tpldata/tribeconf.json similarity index 100% rename from nationchains/www/adminapx/tribeconf.json rename to nationchains/www/adminapx/static/tpldata/tribeconf.json diff --git a/nationchains/www/adminapx/static/js/axios.min.js b/nationchains/www/cdn/share/js/axios.min.js similarity index 100% rename from nationchains/www/adminapx/static/js/axios.min.js rename to nationchains/www/cdn/share/js/axios.min.js diff --git a/nationchains/www/adminapx/static/js/axios.min.map b/nationchains/www/cdn/share/js/axios.min.map similarity index 100% rename from nationchains/www/adminapx/static/js/axios.min.map rename to nationchains/www/cdn/share/js/axios.min.map diff --git a/nationchains/www/cdn/share/js/dayjs.min.js b/nationchains/www/cdn/share/js/dayjs.min.js new file mode 100644 index 0000000..ba16e65 --- /dev/null +++ b/nationchains/www/cdn/share/js/dayjs.min.js @@ -0,0 +1 @@ +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t="undefined"!=typeof globalThis?globalThis:t||self).dayjs=e()}(this,(function(){"use strict";var t=1e3,e=6e4,n=36e5,r="millisecond",i="second",s="minute",u="hour",a="day",o="week",f="month",h="quarter",c="year",d="date",l="Invalid Date",$=/^(\d{4})[-/]?(\d{1,2})?[-/]?(\d{0,2})[Tt\s]*(\d{1,2})?:?(\d{1,2})?:?(\d{1,2})?[.:]?(\d+)?$/,y=/\[([^\]]+)]|Y{1,4}|M{1,4}|D{1,2}|d{1,4}|H{1,2}|h{1,2}|a|A|m{1,2}|s{1,2}|Z{1,2}|SSS/g,M={name:"en",weekdays:"Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday".split("_"),months:"January_February_March_April_May_June_July_August_September_October_November_December".split("_"),ordinal:function(t){var e=["th","st","nd","rd"],n=t%100;return"["+t+(e[(n-20)%10]||e[n]||e[0])+"]"}},m=function(t,e,n){var r=String(t);return!r||r.length>=e?t:""+Array(e+1-r.length).join(n)+t},v={s:m,z:function(t){var e=-t.utcOffset(),n=Math.abs(e),r=Math.floor(n/60),i=n%60;return(e<=0?"+":"-")+m(r,2,"0")+":"+m(i,2,"0")},m:function t(e,n){if(e.date()1)return t(u[0])}else{var a=e.name;D[a]=e,i=a}return!r&&i&&(g=i),i||!r&&g},w=function(t,e){if(p(t))return t.clone();var n="object"==typeof e?e:{};return n.date=t,n.args=arguments,new _(n)},O=v;O.l=S,O.i=p,O.w=function(t,e){return w(t,{locale:e.$L,utc:e.$u,x:e.$x,$offset:e.$offset})};var _=function(){function M(t){this.$L=S(t.locale,null,!0),this.parse(t)}var m=M.prototype;return m.parse=function(t){this.$d=function(t){var e=t.date,n=t.utc;if(null===e)return new Date(NaN);if(O.u(e))return new Date;if(e instanceof Date)return new Date(e);if("string"==typeof e&&!/Z$/i.test(e)){var r=e.match($);if(r){var i=r[2]-1||0,s=(r[7]||"0").substring(0,3);return n?new Date(Date.UTC(r[1],i,r[3]||1,r[4]||0,r[5]||0,r[6]||0,s)):new Date(r[1],i,r[3]||1,r[4]||0,r[5]||0,r[6]||0,s)}}return new Date(e)}(t),this.$x=t.x||{},this.init()},m.init=function(){var t=this.$d;this.$y=t.getFullYear(),this.$M=t.getMonth(),this.$D=t.getDate(),this.$W=t.getDay(),this.$H=t.getHours(),this.$m=t.getMinutes(),this.$s=t.getSeconds(),this.$ms=t.getMilliseconds()},m.$utils=function(){return O},m.isValid=function(){return!(this.$d.toString()===l)},m.isSame=function(t,e){var n=w(t);return this.startOf(e)<=n&&n<=this.endOf(e)},m.isAfter=function(t,e){return w(t) { + this[doneWritingResolve] = resolve; + this[doneWritingReject] = reject; + }); + this[doneWritingPromise].catch(() => {}); + } + } + + ArrayStream.prototype.getReader = function() { + if (this[readingIndex] === undefined) { + this[readingIndex] = 0; + } + return { + read: async () => { + await this[doneWritingPromise]; + if (this[readingIndex] === this.length) { + return { value: undefined, done: true }; + } + return { value: this[this[readingIndex]++], done: false }; + } + }; + }; + + ArrayStream.prototype.readToEnd = async function(join) { + await this[doneWritingPromise]; + const result = join(this.slice(this[readingIndex])); + this.length = 0; + return result; + }; + + ArrayStream.prototype.clone = function() { + const clone = new ArrayStream(); + clone[doneWritingPromise] = this[doneWritingPromise].then(() => { + clone.push(...this); + }); + return clone; + }; + + /** + * Check whether data is an ArrayStream + * @param {Any} input data to check + * @returns {boolean} + */ + function isArrayStream(input) { + return input && input.getReader && Array.isArray(input); + } + + /** + * A wrapper class over the native WritableStreamDefaultWriter. + * It also lets you "write data to" array streams instead of streams. + * @class + */ + function Writer(input) { + if (!isArrayStream(input)) { + const writer = input.getWriter(); + const releaseLock = writer.releaseLock; + writer.releaseLock = () => { + writer.closed.catch(function() {}); + releaseLock.call(writer); + }; + return writer; + } + this.stream = input; + } + + /** + * Write a chunk of data. + * @returns {Promise} + * @async + */ + Writer.prototype.write = async function(chunk) { + this.stream.push(chunk); + }; + + /** + * Close the stream. + * @returns {Promise} + * @async + */ + Writer.prototype.close = async function() { + this.stream[doneWritingResolve](); + }; + + /** + * Error the stream. + * @returns {Promise} + * @async + */ + Writer.prototype.abort = async function(reason) { + this.stream[doneWritingReject](reason); + return reason; + }; + + /** + * Release the writer's lock. + * @returns {undefined} + * @async + */ + Writer.prototype.releaseLock = function() {}; + + const isNode = typeof globalThis.process === 'object' && + typeof globalThis.process.versions === 'object'; + + const NodeReadableStream = isNode && void('stream').Readable; + + /** + * Check whether data is a Stream, and if so of which type + * @param {Any} input data to check + * @returns {'web'|'ponyfill'|'node'|'array'|'web-like'|false} + */ + function isStream(input) { + if (isArrayStream(input)) { + return 'array'; + } + if (globalThis.ReadableStream && globalThis.ReadableStream.prototype.isPrototypeOf(input)) { + return 'web'; + } + if (ReadableStream && ReadableStream.prototype.isPrototypeOf(input)) { + return 'ponyfill'; + } + if (NodeReadableStream && NodeReadableStream.prototype.isPrototypeOf(input)) { + return 'node'; + } + if (input && input.getReader) { + return 'web-like'; + } + return false; + } + + /** + * Check whether data is a Uint8Array + * @param {Any} input data to check + * @returns {Boolean} + */ + function isUint8Array(input) { + return Uint8Array.prototype.isPrototypeOf(input); + } + + /** + * Concat Uint8Arrays + * @param {Array} Array of Uint8Arrays to concatenate + * @returns {Uint8array} Concatenated array + */ + function concatUint8Array(arrays) { + if (arrays.length === 1) return arrays[0]; + + let totalLength = 0; + for (let i = 0; i < arrays.length; i++) { + if (!isUint8Array(arrays[i])) { + throw new Error('concatUint8Array: Data must be in the form of a Uint8Array'); + } + + totalLength += arrays[i].length; + } + + const result = new Uint8Array(totalLength); + let pos = 0; + arrays.forEach(function (element) { + result.set(element, pos); + pos += element.length; + }); + + return result; + } + + const NodeBuffer = isNode && void('buffer').Buffer; + const NodeReadableStream$1 = isNode && void('stream').Readable; + + /** + * Web / node stream conversion functions + * From https://github.com/gwicke/node-web-streams + */ + + let nodeToWeb; + let webToNode; + + if (NodeReadableStream$1) { + + /** + * Convert a Node Readable Stream to a Web ReadableStream + * @param {Readable} nodeStream + * @returns {ReadableStream} + */ + nodeToWeb = function(nodeStream) { + let canceled = false; + return new ReadableStream({ + start(controller) { + nodeStream.pause(); + nodeStream.on('data', chunk => { + if (canceled) { + return; + } + if (NodeBuffer.isBuffer(chunk)) { + chunk = new Uint8Array(chunk.buffer, chunk.byteOffset, chunk.byteLength); + } + controller.enqueue(chunk); + nodeStream.pause(); + }); + nodeStream.on('end', () => { + if (canceled) { + return; + } + controller.close(); + }); + nodeStream.on('error', e => controller.error(e)); + }, + pull() { + nodeStream.resume(); + }, + cancel(reason) { + canceled = true; + nodeStream.destroy(reason); + } + }); + }; + + + class NodeReadable extends NodeReadableStream$1 { + constructor(webStream, options) { + super(options); + this._reader = getReader(webStream); + } + + async _read(size) { + try { + while (true) { + const { done, value } = await this._reader.read(); + if (done) { + this.push(null); + break; + } + if (!this.push(value) || this._cancelling) { + this._reading = false; + break; + } + } + } catch(e) { + this.emit('error', e); + } + } + + _destroy(reason) { + this._reader.cancel(reason); + } + } + + /** + * Convert a Web ReadableStream to a Node Readable Stream + * @param {ReadableStream} webStream + * @param {Object} options + * @returns {Readable} + */ + webToNode = function(webStream, options) { + return new NodeReadable(webStream, options); + }; + + } + + const doneReadingSet = new WeakSet(); + const externalBuffer = Symbol('externalBuffer'); + + /** + * A wrapper class over the native ReadableStreamDefaultReader. + * This additionally implements pushing back data on the stream, which + * lets us implement peeking and a host of convenience functions. + * It also lets you read data other than streams, such as a Uint8Array. + * @class + */ + function Reader(input) { + this.stream = input; + if (input[externalBuffer]) { + this[externalBuffer] = input[externalBuffer].slice(); + } + if (isArrayStream(input)) { + const reader = input.getReader(); + this._read = reader.read.bind(reader); + this._releaseLock = () => {}; + this._cancel = () => {}; + return; + } + let streamType = isStream(input); + if (streamType === 'node') { + input = nodeToWeb(input); + } + if (streamType) { + const reader = input.getReader(); + this._read = reader.read.bind(reader); + this._releaseLock = () => { + reader.closed.catch(function() {}); + reader.releaseLock(); + }; + this._cancel = reader.cancel.bind(reader); + return; + } + let doneReading = false; + this._read = async () => { + if (doneReading || doneReadingSet.has(input)) { + return { value: undefined, done: true }; + } + doneReading = true; + return { value: input, done: false }; + }; + this._releaseLock = () => { + if (doneReading) { + try { + doneReadingSet.add(input); + } catch(e) {} + } + }; + } + + /** + * Read a chunk of data. + * @returns {Promise} Either { done: false, value: Uint8Array | String } or { done: true, value: undefined } + * @async + */ + Reader.prototype.read = async function() { + if (this[externalBuffer] && this[externalBuffer].length) { + const value = this[externalBuffer].shift(); + return { done: false, value }; + } + return this._read(); + }; + + /** + * Allow others to read the stream. + */ + Reader.prototype.releaseLock = function() { + if (this[externalBuffer]) { + this.stream[externalBuffer] = this[externalBuffer]; + } + this._releaseLock(); + }; + + /** + * Cancel the stream. + */ + Reader.prototype.cancel = function(reason) { + return this._cancel(reason); + }; + + /** + * Read up to and including the first \n character. + * @returns {Promise} + * @async + */ + Reader.prototype.readLine = async function() { + let buffer = []; + let returnVal; + while (!returnVal) { + let { done, value } = await this.read(); + value += ''; + if (done) { + if (buffer.length) return concat(buffer); + return; + } + const lineEndIndex = value.indexOf('\n') + 1; + if (lineEndIndex) { + returnVal = concat(buffer.concat(value.substr(0, lineEndIndex))); + buffer = []; + } + if (lineEndIndex !== value.length) { + buffer.push(value.substr(lineEndIndex)); + } + } + this.unshift(...buffer); + return returnVal; + }; + + /** + * Read a single byte/character. + * @returns {Promise} + * @async + */ + Reader.prototype.readByte = async function() { + const { done, value } = await this.read(); + if (done) return; + const byte = value[0]; + this.unshift(slice(value, 1)); + return byte; + }; + + /** + * Read a specific amount of bytes/characters, unless the stream ends before that amount. + * @returns {Promise} + * @async + */ + Reader.prototype.readBytes = async function(length) { + const buffer = []; + let bufferLength = 0; + while (true) { + const { done, value } = await this.read(); + if (done) { + if (buffer.length) return concat(buffer); + return; + } + buffer.push(value); + bufferLength += value.length; + if (bufferLength >= length) { + const bufferConcat = concat(buffer); + this.unshift(slice(bufferConcat, length)); + return slice(bufferConcat, 0, length); + } + } + }; + + /** + * Peek (look ahead) a specific amount of bytes/characters, unless the stream ends before that amount. + * @returns {Promise} + * @async + */ + Reader.prototype.peekBytes = async function(length) { + const bytes = await this.readBytes(length); + this.unshift(bytes); + return bytes; + }; + + /** + * Push data to the front of the stream. + * Data must have been read in the last call to read*. + * @param {...(Uint8Array|String|Undefined)} values + */ + Reader.prototype.unshift = function(...values) { + if (!this[externalBuffer]) { + this[externalBuffer] = []; + } + if ( + values.length === 1 && isUint8Array(values[0]) && + this[externalBuffer].length && values[0].length && + this[externalBuffer][0].byteOffset >= values[0].length + ) { + this[externalBuffer][0] = new Uint8Array( + this[externalBuffer][0].buffer, + this[externalBuffer][0].byteOffset - values[0].length, + this[externalBuffer][0].byteLength + values[0].length + ); + return; + } + this[externalBuffer].unshift(...values.filter(value => value && value.length)); + }; + + /** + * Read the stream to the end and return its contents, concatenated by the join function (defaults to streams.concat). + * @param {Function} join + * @returns {Promise} the return value of join() + * @async + */ + Reader.prototype.readToEnd = async function(join=concat) { + const result = []; + while (true) { + const { done, value } = await this.read(); + if (done) break; + result.push(value); + } + return join(result); + }; + + let { ReadableStream, WritableStream, TransformStream } = globalThis; + + let toPonyfillReadable, toNativeReadable; + + async function loadStreamsPonyfill() { + if (TransformStream) { + return; + } + + const [ponyfill, adapter] = await Promise.all([ + Promise.resolve().then(function () { return ponyfill_es6; }), + Promise.resolve().then(function () { return webStreamsAdapter; }) + ]); + + ({ ReadableStream, WritableStream, TransformStream } = ponyfill); + + const { createReadableStreamWrapper } = adapter; + + if (globalThis.ReadableStream && ReadableStream !== globalThis.ReadableStream) { + toPonyfillReadable = createReadableStreamWrapper(ReadableStream); + toNativeReadable = createReadableStreamWrapper(globalThis.ReadableStream); + } + } + + const NodeBuffer$1 = isNode && void('buffer').Buffer; + + /** + * Convert data to Stream + * @param {ReadableStream|Uint8array|String} input data to convert + * @returns {ReadableStream} Converted data + */ + function toStream(input) { + let streamType = isStream(input); + if (streamType === 'node') { + return nodeToWeb(input); + } + if (streamType === 'web' && toPonyfillReadable) { + return toPonyfillReadable(input); + } + if (streamType) { + return input; + } + return new ReadableStream({ + start(controller) { + controller.enqueue(input); + controller.close(); + } + }); + } + + /** + * Convert data to ArrayStream + * @param {Object} input data to convert + * @returns {ArrayStream} Converted data + */ + function toArrayStream(input) { + if (isStream(input)) { + return input; + } + const stream = new ArrayStream(); + (async () => { + const writer = getWriter(stream); + await writer.write(input); + await writer.close(); + })(); + return stream; + } + + /** + * Concat a list of Uint8Arrays, Strings or Streams + * The caller should not mix Uint8Arrays with Strings, but may mix Streams with non-Streams. + * @param {Array} Array of Uint8Arrays/Strings/Streams to concatenate + * @returns {Uint8array|String|ReadableStream} Concatenated array + */ + function concat(list) { + if (list.some(stream => isStream(stream) && !isArrayStream(stream))) { + return concatStream(list); + } + if (list.some(stream => isArrayStream(stream))) { + return concatArrayStream(list); + } + if (typeof list[0] === 'string') { + return list.join(''); + } + if (NodeBuffer$1 && NodeBuffer$1.isBuffer(list[0])) { + return NodeBuffer$1.concat(list); + } + return concatUint8Array(list); + } + + /** + * Concat a list of Streams + * @param {Array} list Array of Uint8Arrays/Strings/Streams to concatenate + * @returns {ReadableStream} Concatenated list + */ + function concatStream(list) { + list = list.map(toStream); + const transform = transformWithCancel(async function(reason) { + await Promise.all(transforms.map(stream => cancel(stream, reason))); + }); + let prev = Promise.resolve(); + const transforms = list.map((stream, i) => transformPair(stream, (readable, writable) => { + prev = prev.then(() => pipe(readable, transform.writable, { + preventClose: i !== list.length - 1 + })); + return prev; + })); + return transform.readable; + } + + /** + * Concat a list of ArrayStreams + * @param {Array} list Array of Uint8Arrays/Strings/ArrayStreams to concatenate + * @returns {ArrayStream} Concatenated streams + */ + function concatArrayStream(list) { + const result = new ArrayStream(); + let prev = Promise.resolve(); + list.forEach((stream, i) => { + prev = prev.then(() => pipe(stream, result, { + preventClose: i !== list.length - 1 + })); + return prev; + }); + return result; + } + + /** + * Get a Reader + * @param {ReadableStream|Uint8array|String} input + * @returns {Reader} + */ + function getReader(input) { + return new Reader(input); + } + + /** + * Get a Writer + * @param {WritableStream} input + * @returns {Writer} + */ + function getWriter(input) { + return new Writer(input); + } + + /** + * Pipe a readable stream to a writable stream. Don't throw on input stream errors, but forward them to the output stream. + * @param {ReadableStream|Uint8array|String} input + * @param {WritableStream} target + * @param {Object} (optional) options + * @returns {Promise} Promise indicating when piping has finished (input stream closed or errored) + * @async + */ + async function pipe(input, target, { + preventClose = false, + preventAbort = false, + preventCancel = false + } = {}) { + if (isStream(input) && !isArrayStream(input)) { + input = toStream(input); + try { + if (input[externalBuffer]) { + const writer = getWriter(target); + for (let i = 0; i < input[externalBuffer].length; i++) { + await writer.ready; + await writer.write(input[externalBuffer][i]); + } + writer.releaseLock(); + } + await input.pipeTo(target, { + preventClose, + preventAbort, + preventCancel + }); + } catch(e) {} + return; + } + input = toArrayStream(input); + const reader = getReader(input); + const writer = getWriter(target); + try { + while (true) { + await writer.ready; + const { done, value } = await reader.read(); + if (done) { + if (!preventClose) await writer.close(); + break; + } + await writer.write(value); + } + } catch (e) { + if (!preventAbort) await writer.abort(e); + } finally { + reader.releaseLock(); + writer.releaseLock(); + } + } + + /** + * Pipe a readable stream through a transform stream. + * @param {ReadableStream|Uint8array|String} input + * @param {Object} (optional) options + * @returns {ReadableStream} transformed stream + */ + function transformRaw(input, options) { + const transformStream = new TransformStream(options); + pipe(input, transformStream.writable); + return transformStream.readable; + } + + /** + * Create a cancelable TransformStream. + * @param {Function} cancel + * @returns {TransformStream} + */ + function transformWithCancel(cancel) { + let pulled = false; + let backpressureChangePromiseResolve; + let outputController; + return { + readable: new ReadableStream({ + start(controller) { + outputController = controller; + }, + pull() { + if (backpressureChangePromiseResolve) { + backpressureChangePromiseResolve(); + } else { + pulled = true; + } + }, + cancel + }, {highWaterMark: 0}), + writable: new WritableStream({ + write: async function(chunk) { + outputController.enqueue(chunk); + if (!pulled) { + await new Promise(resolve => { + backpressureChangePromiseResolve = resolve; + }); + backpressureChangePromiseResolve = null; + } else { + pulled = false; + } + }, + close: outputController.close.bind(outputController), + abort: outputController.error.bind(outputController) + }) + }; + } + + /** + * Transform a stream using helper functions which are called on each chunk, and on stream close, respectively. + * @param {ReadableStream|Uint8array|String} input + * @param {Function} process + * @param {Function} finish + * @returns {ReadableStream|Uint8array|String} + */ + function transform(input, process = () => undefined, finish = () => undefined) { + if (isArrayStream(input)) { + const output = new ArrayStream(); + (async () => { + const writer = getWriter(output); + try { + const data = await readToEnd(input); + const result1 = process(data); + const result2 = finish(); + let result; + if (result1 !== undefined && result2 !== undefined) result = concat([result1, result2]); + else result = result1 !== undefined ? result1 : result2; + await writer.write(result); + await writer.close(); + } catch (e) { + await writer.abort(e); + } + })(); + return output; + } + if (isStream(input)) { + return transformRaw(input, { + async transform(value, controller) { + try { + const result = await process(value); + if (result !== undefined) controller.enqueue(result); + } catch(e) { + controller.error(e); + } + }, + async flush(controller) { + try { + const result = await finish(); + if (result !== undefined) controller.enqueue(result); + } catch(e) { + controller.error(e); + } + } + }); + } + const result1 = process(input); + const result2 = finish(); + if (result1 !== undefined && result2 !== undefined) return concat([result1, result2]); + return result1 !== undefined ? result1 : result2; + } + + /** + * Transform a stream using a helper function which is passed a readable and a writable stream. + * This function also maintains the possibility to cancel the input stream, + * and does so on cancelation of the output stream, despite cancelation + * normally being impossible when the input stream is being read from. + * @param {ReadableStream|Uint8array|String} input + * @param {Function} fn + * @returns {ReadableStream} + */ + function transformPair(input, fn) { + if (isStream(input) && !isArrayStream(input)) { + let incomingTransformController; + const incoming = new TransformStream({ + start(controller) { + incomingTransformController = controller; + } + }); + + const pipeDonePromise = pipe(input, incoming.writable); + + const outgoing = transformWithCancel(async function(reason) { + incomingTransformController.error(reason); + await pipeDonePromise; + await new Promise(setTimeout); + }); + fn(incoming.readable, outgoing.writable); + return outgoing.readable; + } + input = toArrayStream(input); + const output = new ArrayStream(); + fn(input, output); + return output; + } + + /** + * Parse a stream using a helper function which is passed a Reader. + * The reader additionally has a remainder() method which returns a + * stream pointing to the remainder of input, and is linked to input + * for cancelation. + * @param {ReadableStream|Uint8array|String} input + * @param {Function} fn + * @returns {Any} the return value of fn() + */ + function parse(input, fn) { + let returnValue; + const transformed = transformPair(input, (readable, writable) => { + const reader = getReader(readable); + reader.remainder = () => { + reader.releaseLock(); + pipe(readable, writable); + return transformed; + }; + returnValue = fn(reader); + }); + return returnValue; + } + + /** + * Tee a Stream for reading it twice. The input stream can no longer be read after tee()ing. + * Reading either of the two returned streams will pull from the input stream. + * The input stream will only be canceled if both of the returned streams are canceled. + * @param {ReadableStream|Uint8array|String} input + * @returns {Array} array containing two copies of input + */ + function tee(input) { + if (isArrayStream(input)) { + throw new Error('ArrayStream cannot be tee()d, use clone() instead'); + } + if (isStream(input)) { + const teed = toStream(input).tee(); + teed[0][externalBuffer] = teed[1][externalBuffer] = input[externalBuffer]; + return teed; + } + return [slice(input), slice(input)]; + } + + /** + * Clone a Stream for reading it twice. The input stream can still be read after clone()ing. + * Reading from the clone will pull from the input stream. + * The input stream will only be canceled if both the clone and the input stream are canceled. + * @param {ReadableStream|Uint8array|String} input + * @returns {ReadableStream|Uint8array|String} cloned input + */ + function clone(input) { + if (isArrayStream(input)) { + return input.clone(); + } + if (isStream(input)) { + const teed = tee(input); + overwrite(input, teed[0]); + return teed[1]; + } + return slice(input); + } + + /** + * Clone a Stream for reading it twice. Data will arrive at the same rate as the input stream is being read. + * Reading from the clone will NOT pull from the input stream. Data only arrives when reading the input stream. + * The input stream will NOT be canceled if the clone is canceled, only if the input stream are canceled. + * If the input stream is canceled, the clone will be errored. + * @param {ReadableStream|Uint8array|String} input + * @returns {ReadableStream|Uint8array|String} cloned input + */ + function passiveClone(input) { + if (isArrayStream(input)) { + return clone(input); + } + if (isStream(input)) { + return new ReadableStream({ + start(controller) { + const transformed = transformPair(input, async (readable, writable) => { + const reader = getReader(readable); + const writer = getWriter(writable); + try { + while (true) { + await writer.ready; + const { done, value } = await reader.read(); + if (done) { + try { controller.close(); } catch(e) {} + await writer.close(); + return; + } + try { controller.enqueue(value); } catch(e) {} + await writer.write(value); + } + } catch(e) { + controller.error(e); + await writer.abort(e); + } + }); + overwrite(input, transformed); + } + }); + } + return slice(input); + } + + /** + * Modify a stream object to point to a different stream object. + * This is used internally by clone() and passiveClone() to provide an abstraction over tee(). + * @param {ReadableStream} input + * @param {ReadableStream} clone + */ + function overwrite(input, clone) { + // Overwrite input.getReader, input.locked, etc to point to clone + Object.entries(Object.getOwnPropertyDescriptors(input.constructor.prototype)).forEach(([name, descriptor]) => { + if (name === 'constructor') { + return; + } + if (descriptor.value) { + descriptor.value = descriptor.value.bind(clone); + } else { + descriptor.get = descriptor.get.bind(clone); + } + Object.defineProperty(input, name, descriptor); + }); + } + + /** + * Return a stream pointing to a part of the input stream. + * @param {ReadableStream|Uint8array|String} input + * @returns {ReadableStream|Uint8array|String} clone + */ + function slice(input, begin=0, end=Infinity) { + if (isArrayStream(input)) { + throw new Error('Not implemented'); + } + if (isStream(input)) { + if (begin >= 0 && end >= 0) { + let bytesRead = 0; + return transformRaw(input, { + transform(value, controller) { + if (bytesRead < end) { + if (bytesRead + value.length >= begin) { + controller.enqueue(slice(value, Math.max(begin - bytesRead, 0), end - bytesRead)); + } + bytesRead += value.length; + } else { + controller.terminate(); + } + } + }); + } + if (begin < 0 && (end < 0 || end === Infinity)) { + let lastBytes = []; + return transform(input, value => { + if (value.length >= -begin) lastBytes = [value]; + else lastBytes.push(value); + }, () => slice(concat(lastBytes), begin, end)); + } + if (begin === 0 && end < 0) { + let lastBytes; + return transform(input, value => { + const returnValue = lastBytes ? concat([lastBytes, value]) : value; + if (returnValue.length >= -end) { + lastBytes = slice(returnValue, end); + return slice(returnValue, begin, end); + } else { + lastBytes = returnValue; + } + }); + } + console.warn(`stream.slice(input, ${begin}, ${end}) not implemented efficiently.`); + return fromAsync(async () => slice(await readToEnd(input), begin, end)); + } + if (input[externalBuffer]) { + input = concat(input[externalBuffer].concat([input])); + } + if (isUint8Array(input) && !(NodeBuffer$1 && NodeBuffer$1.isBuffer(input))) { + if (end === Infinity) end = input.length; + return input.subarray(begin, end); + } + return input.slice(begin, end); + } + + /** + * Read a stream to the end and return its contents, concatenated by the join function (defaults to concat). + * @param {ReadableStream|Uint8array|String} input + * @param {Function} join + * @returns {Promise} the return value of join() + * @async + */ + async function readToEnd(input, join=concat) { + if (isArrayStream(input)) { + return input.readToEnd(join); + } + if (isStream(input)) { + return getReader(input).readToEnd(join); + } + return input; + } + + /** + * Cancel a stream. + * @param {ReadableStream|Uint8array|String} input + * @param {Any} reason + * @returns {Promise} indicates when the stream has been canceled + * @async + */ + async function cancel(input, reason) { + if (isStream(input)) { + if (input.cancel) { + return input.cancel(reason); + } + if (input.destroy) { + input.destroy(reason); + await new Promise(setTimeout); + return reason; + } + } + } + + /** + * Convert an async function to an ArrayStream. When the function returns, its return value is written to the stream. + * @param {Function} fn + * @returns {ArrayStream} + */ + function fromAsync(fn) { + const arrayStream = new ArrayStream(); + (async () => { + const writer = getWriter(arrayStream); + try { + await writer.write(await fn()); + await writer.close(); + } catch (e) { + await writer.abort(e); + } + })(); + return arrayStream; + } + + /* eslint-disable new-cap */ + + /** + * @fileoverview + * BigInteger implementation of basic operations + * that wraps the native BigInt library. + * Operations are not constant time, + * but we try and limit timing leakage where we can + * @module biginteger/native + * @private + */ + + /** + * @private + */ + class BigInteger { + /** + * Get a BigInteger (input must be big endian for strings and arrays) + * @param {Number|String|Uint8Array} n - Value to convert + * @throws {Error} on null or undefined input + */ + constructor(n) { + if (n === undefined) { + throw new Error('Invalid BigInteger input'); + } + + if (n instanceof Uint8Array) { + const bytes = n; + const hex = new Array(bytes.length); + for (let i = 0; i < bytes.length; i++) { + const hexByte = bytes[i].toString(16); + hex[i] = (bytes[i] <= 0xF) ? ('0' + hexByte) : hexByte; + } + this.value = BigInt('0x0' + hex.join('')); + } else { + this.value = BigInt(n); + } + } + + clone() { + return new BigInteger(this.value); + } + + /** + * BigInteger increment in place + */ + iinc() { + this.value++; + return this; + } + + /** + * BigInteger increment + * @returns {BigInteger} this + 1. + */ + inc() { + return this.clone().iinc(); + } + + /** + * BigInteger decrement in place + */ + idec() { + this.value--; + return this; + } + + /** + * BigInteger decrement + * @returns {BigInteger} this - 1. + */ + dec() { + return this.clone().idec(); + } + + /** + * BigInteger addition in place + * @param {BigInteger} x - Value to add + */ + iadd(x) { + this.value += x.value; + return this; + } + + /** + * BigInteger addition + * @param {BigInteger} x - Value to add + * @returns {BigInteger} this + x. + */ + add(x) { + return this.clone().iadd(x); + } + + /** + * BigInteger subtraction in place + * @param {BigInteger} x - Value to subtract + */ + isub(x) { + this.value -= x.value; + return this; + } + + /** + * BigInteger subtraction + * @param {BigInteger} x - Value to subtract + * @returns {BigInteger} this - x. + */ + sub(x) { + return this.clone().isub(x); + } + + /** + * BigInteger multiplication in place + * @param {BigInteger} x - Value to multiply + */ + imul(x) { + this.value *= x.value; + return this; + } + + /** + * BigInteger multiplication + * @param {BigInteger} x - Value to multiply + * @returns {BigInteger} this * x. + */ + mul(x) { + return this.clone().imul(x); + } + + /** + * Compute value modulo m, in place + * @param {BigInteger} m - Modulo + */ + imod(m) { + this.value %= m.value; + if (this.isNegative()) { + this.iadd(m); + } + return this; + } + + /** + * Compute value modulo m + * @param {BigInteger} m - Modulo + * @returns {BigInteger} this mod m. + */ + mod(m) { + return this.clone().imod(m); + } + + /** + * Compute modular exponentiation using square and multiply + * @param {BigInteger} e - Exponent + * @param {BigInteger} n - Modulo + * @returns {BigInteger} this ** e mod n. + */ + modExp(e, n) { + if (n.isZero()) throw Error('Modulo cannot be zero'); + if (n.isOne()) return new BigInteger(0); + if (e.isNegative()) throw Error('Unsopported negative exponent'); + + let exp = e.value; + let x = this.value; + + x %= n.value; + let r = BigInt(1); + while (exp > BigInt(0)) { + const lsb = exp & BigInt(1); + exp >>= BigInt(1); // e / 2 + // Always compute multiplication step, to reduce timing leakage + const rx = (r * x) % n.value; + // Update r only if lsb is 1 (odd exponent) + r = lsb ? rx : r; + x = (x * x) % n.value; // Square + } + return new BigInteger(r); + } + + + /** + * Compute the inverse of this value modulo n + * Note: this and and n must be relatively prime + * @param {BigInteger} n - Modulo + * @returns {BigInteger} x such that this*x = 1 mod n + * @throws {Error} if the inverse does not exist + */ + modInv(n) { + const { gcd, x } = this._egcd(n); + if (!gcd.isOne()) { + throw new Error('Inverse does not exist'); + } + return x.add(n).mod(n); + } + + /** + * Extended Eucleadian algorithm (http://anh.cs.luc.edu/331/notes/xgcd.pdf) + * Given a = this and b, compute (x, y) such that ax + by = gdc(a, b) + * @param {BigInteger} b - Second operand + * @returns {{ gcd, x, y: BigInteger }} + */ + _egcd(b) { + let x = BigInt(0); + let y = BigInt(1); + let xPrev = BigInt(1); + let yPrev = BigInt(0); + + let a = this.value; + b = b.value; + + while (b !== BigInt(0)) { + const q = a / b; + let tmp = x; + x = xPrev - q * x; + xPrev = tmp; + + tmp = y; + y = yPrev - q * y; + yPrev = tmp; + + tmp = b; + b = a % b; + a = tmp; + } + + return { + x: new BigInteger(xPrev), + y: new BigInteger(yPrev), + gcd: new BigInteger(a) + }; + } + + /** + * Compute greatest common divisor between this and n + * @param {BigInteger} b - Operand + * @returns {BigInteger} gcd + */ + gcd(b) { + let a = this.value; + b = b.value; + while (b !== BigInt(0)) { + const tmp = b; + b = a % b; + a = tmp; + } + return new BigInteger(a); + } + + /** + * Shift this to the left by x, in place + * @param {BigInteger} x - Shift value + */ + ileftShift(x) { + this.value <<= x.value; + return this; + } + + /** + * Shift this to the left by x + * @param {BigInteger} x - Shift value + * @returns {BigInteger} this << x. + */ + leftShift(x) { + return this.clone().ileftShift(x); + } + + /** + * Shift this to the right by x, in place + * @param {BigInteger} x - Shift value + */ + irightShift(x) { + this.value >>= x.value; + return this; + } + + /** + * Shift this to the right by x + * @param {BigInteger} x - Shift value + * @returns {BigInteger} this >> x. + */ + rightShift(x) { + return this.clone().irightShift(x); + } + + /** + * Whether this value is equal to x + * @param {BigInteger} x + * @returns {Boolean} + */ + equal(x) { + return this.value === x.value; + } + + /** + * Whether this value is less than x + * @param {BigInteger} x + * @returns {Boolean} + */ + lt(x) { + return this.value < x.value; + } + + /** + * Whether this value is less than or equal to x + * @param {BigInteger} x + * @returns {Boolean} + */ + lte(x) { + return this.value <= x.value; + } + + /** + * Whether this value is greater than x + * @param {BigInteger} x + * @returns {Boolean} + */ + gt(x) { + return this.value > x.value; + } + + /** + * Whether this value is greater than or equal to x + * @param {BigInteger} x + * @returns {Boolean} + */ + gte(x) { + return this.value >= x.value; + } + + isZero() { + return this.value === BigInt(0); + } + + isOne() { + return this.value === BigInt(1); + } + + isNegative() { + return this.value < BigInt(0); + } + + isEven() { + return !(this.value & BigInt(1)); + } + + abs() { + const res = this.clone(); + if (this.isNegative()) { + res.value = -res.value; + } + return res; + } + + /** + * Get this value as a string + * @returns {String} this value. + */ + toString() { + return this.value.toString(); + } + + /** + * Get this value as an exact Number (max 53 bits) + * Fails if this value is too large + * @returns {Number} + */ + toNumber() { + const number = Number(this.value); + if (number > Number.MAX_SAFE_INTEGER) { + // We throw and error to conform with the bn.js implementation + throw new Error('Number can only safely store up to 53 bits'); + } + return number; + } + + /** + * Get value of i-th bit + * @param {Number} i - Bit index + * @returns {Number} Bit value. + */ + getBit(i) { + const bit = (this.value >> BigInt(i)) & BigInt(1); + return (bit === BigInt(0)) ? 0 : 1; + } + + /** + * Compute bit length + * @returns {Number} Bit length. + */ + bitLength() { + const zero = new BigInteger(0); + const one = new BigInteger(1); + const negOne = new BigInteger(-1); + + // -1n >> -1n is -1n + // 1n >> 1n is 0n + const target = this.isNegative() ? negOne : zero; + let bitlen = 1; + const tmp = this.clone(); + while (!tmp.irightShift(one).equal(target)) { + bitlen++; + } + return bitlen; + } + + /** + * Compute byte length + * @returns {Number} Byte length. + */ + byteLength() { + const zero = new BigInteger(0); + const negOne = new BigInteger(-1); + + const target = this.isNegative() ? negOne : zero; + const eight = new BigInteger(8); + let len = 1; + const tmp = this.clone(); + while (!tmp.irightShift(eight).equal(target)) { + len++; + } + return len; + } + + /** + * Get Uint8Array representation of this number + * @param {String} endian - Endianess of output array (defaults to 'be') + * @param {Number} length - Of output array + * @returns {Uint8Array} + */ + toUint8Array(endian = 'be', length) { + // we get and parse the hex string (https://coolaj86.com/articles/convert-js-bigints-to-typedarrays/) + // this is faster than shift+mod iterations + let hex = this.value.toString(16); + if (hex.length % 2 === 1) { + hex = '0' + hex; + } + + const rawLength = hex.length / 2; + const bytes = new Uint8Array(length || rawLength); + // parse hex + const offset = length ? (length - rawLength) : 0; + let i = 0; + while (i < rawLength) { + bytes[i + offset] = parseInt(hex.slice(2 * i, 2 * i + 2), 16); + i++; + } + + if (endian !== 'be') { + bytes.reverse(); + } + + return bytes; + } + } + + const detectBigInt = () => typeof BigInt !== 'undefined'; + + async function getBigInteger() { + if (detectBigInt()) { + return BigInteger; + } else { + const { default: BigInteger } = await Promise.resolve().then(function () { return bn_interface; }); + return BigInteger; + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + const debugMode = (() => { + try { + return process.env.NODE_ENV === 'development'; // eslint-disable-line no-process-env + } catch (e) {} + return false; + })(); + + const util = { + isString: function(data) { + return typeof data === 'string' || data instanceof String; + }, + + isArray: function(data) { + return data instanceof Array; + }, + + isUint8Array: isUint8Array, + + isStream: isStream, + + readNumber: function (bytes) { + let n = 0; + for (let i = 0; i < bytes.length; i++) { + n += (256 ** i) * bytes[bytes.length - 1 - i]; + } + return n; + }, + + writeNumber: function (n, bytes) { + const b = new Uint8Array(bytes); + for (let i = 0; i < bytes; i++) { + b[i] = (n >> (8 * (bytes - i - 1))) & 0xFF; + } + + return b; + }, + + readDate: function (bytes) { + const n = util.readNumber(bytes); + const d = new Date(n * 1000); + return d; + }, + + writeDate: function (time) { + const numeric = Math.floor(time.getTime() / 1000); + + return util.writeNumber(numeric, 4); + }, + + normalizeDate: function (time = Date.now()) { + return time === null || time === Infinity ? time : new Date(Math.floor(+time / 1000) * 1000); + }, + + /** + * Read one MPI from bytes in input + * @param {Uint8Array} bytes - Input data to parse + * @returns {Uint8Array} Parsed MPI. + */ + readMPI: function (bytes) { + const bits = (bytes[0] << 8) | bytes[1]; + const bytelen = (bits + 7) >>> 3; + return bytes.subarray(2, 2 + bytelen); + }, + + /** + * Left-pad Uint8Array to length by adding 0x0 bytes + * @param {Uint8Array} bytes - Data to pad + * @param {Number} length - Padded length + * @returns {Uint8Array} Padded bytes. + */ + leftPad(bytes, length) { + const padded = new Uint8Array(length); + const offset = length - bytes.length; + padded.set(bytes, offset); + return padded; + }, + + /** + * Convert a Uint8Array to an MPI-formatted Uint8Array. + * @param {Uint8Array} bin - An array of 8-bit integers to convert + * @returns {Uint8Array} MPI-formatted Uint8Array. + */ + uint8ArrayToMPI: function (bin) { + const bitSize = util.uint8ArrayBitLength(bin); + if (bitSize === 0) { + throw new Error('Zero MPI'); + } + const stripped = bin.subarray(bin.length - Math.ceil(bitSize / 8)); + const prefix = new Uint8Array([(bitSize & 0xFF00) >> 8, bitSize & 0xFF]); + return util.concatUint8Array([prefix, stripped]); + }, + + /** + * Return bit length of the input data + * @param {Uint8Array} bin input data (big endian) + * @returns bit length + */ + uint8ArrayBitLength: function (bin) { + let i; // index of leading non-zero byte + for (i = 0; i < bin.length; i++) if (bin[i] !== 0) break; + if (i === bin.length) { + return 0; + } + const stripped = bin.subarray(i); + return (stripped.length - 1) * 8 + util.nbits(stripped[0]); + }, + + /** + * Convert a hex string to an array of 8-bit integers + * @param {String} hex - A hex string to convert + * @returns {Uint8Array} An array of 8-bit integers. + */ + hexToUint8Array: function (hex) { + const result = new Uint8Array(hex.length >> 1); + for (let k = 0; k < hex.length >> 1; k++) { + result[k] = parseInt(hex.substr(k << 1, 2), 16); + } + return result; + }, + + /** + * Convert an array of 8-bit integers to a hex string + * @param {Uint8Array} bytes - Array of 8-bit integers to convert + * @returns {String} Hexadecimal representation of the array. + */ + uint8ArrayToHex: function (bytes) { + const r = []; + const e = bytes.length; + let c = 0; + let h; + while (c < e) { + h = bytes[c++].toString(16); + while (h.length < 2) { + h = '0' + h; + } + r.push('' + h); + } + return r.join(''); + }, + + /** + * Convert a string to an array of 8-bit integers + * @param {String} str - String to convert + * @returns {Uint8Array} An array of 8-bit integers. + */ + stringToUint8Array: function (str) { + return transform(str, str => { + if (!util.isString(str)) { + throw new Error('stringToUint8Array: Data must be in the form of a string'); + } + + const result = new Uint8Array(str.length); + for (let i = 0; i < str.length; i++) { + result[i] = str.charCodeAt(i); + } + return result; + }); + }, + + /** + * Convert an array of 8-bit integers to a string + * @param {Uint8Array} bytes - An array of 8-bit integers to convert + * @returns {String} String representation of the array. + */ + uint8ArrayToString: function (bytes) { + bytes = new Uint8Array(bytes); + const result = []; + const bs = 1 << 14; + const j = bytes.length; + + for (let i = 0; i < j; i += bs) { + result.push(String.fromCharCode.apply(String, bytes.subarray(i, i + bs < j ? i + bs : j))); + } + return result.join(''); + }, + + /** + * Convert a native javascript string to a Uint8Array of utf8 bytes + * @param {String|ReadableStream} str - The string to convert + * @returns {Uint8Array|ReadableStream} A valid squence of utf8 bytes. + */ + encodeUTF8: function (str) { + const encoder = new TextEncoder('utf-8'); + // eslint-disable-next-line no-inner-declarations + function process(value, lastChunk = false) { + return encoder.encode(value, { stream: !lastChunk }); + } + return transform(str, process, () => process('', true)); + }, + + /** + * Convert a Uint8Array of utf8 bytes to a native javascript string + * @param {Uint8Array|ReadableStream} utf8 - A valid squence of utf8 bytes + * @returns {String|ReadableStream} A native javascript string. + */ + decodeUTF8: function (utf8) { + const decoder = new TextDecoder('utf-8'); + // eslint-disable-next-line no-inner-declarations + function process(value, lastChunk = false) { + return decoder.decode(value, { stream: !lastChunk }); + } + return transform(utf8, process, () => process(new Uint8Array(), true)); + }, + + /** + * Concat a list of Uint8Arrays, Strings or Streams + * The caller must not mix Uint8Arrays with Strings, but may mix Streams with non-Streams. + * @param {Array} Array - Of Uint8Arrays/Strings/Streams to concatenate + * @returns {Uint8Array|String|ReadableStream} Concatenated array. + */ + concat: concat, + + /** + * Concat Uint8Arrays + * @param {Array} Array - Of Uint8Arrays to concatenate + * @returns {Uint8Array} Concatenated array. + */ + concatUint8Array: concatUint8Array, + + /** + * Check Uint8Array equality + * @param {Uint8Array} array1 - First array + * @param {Uint8Array} array2 - Second array + * @returns {Boolean} Equality. + */ + equalsUint8Array: function (array1, array2) { + if (!util.isUint8Array(array1) || !util.isUint8Array(array2)) { + throw new Error('Data must be in the form of a Uint8Array'); + } + + if (array1.length !== array2.length) { + return false; + } + + for (let i = 0; i < array1.length; i++) { + if (array1[i] !== array2[i]) { + return false; + } + } + return true; + }, + + /** + * Calculates a 16bit sum of a Uint8Array by adding each character + * codes modulus 65535 + * @param {Uint8Array} Uint8Array - To create a sum of + * @returns {Uint8Array} 2 bytes containing the sum of all charcodes % 65535. + */ + writeChecksum: function (text) { + let s = 0; + for (let i = 0; i < text.length; i++) { + s = (s + text[i]) & 0xFFFF; + } + return util.writeNumber(s, 2); + }, + + /** + * Helper function to print a debug message. Debug + * messages are only printed if + * @param {String} str - String of the debug message + */ + printDebug: function (str) { + if (debugMode) { + console.log('[OpenPGP.js debug]', str); + } + }, + + /** + * Helper function to print a debug error. Debug + * messages are only printed if + * @param {String} str - String of the debug message + */ + printDebugError: function (error) { + if (debugMode) { + console.error('[OpenPGP.js debug]', error); + } + }, + + // returns bit length of the integer x + nbits: function (x) { + let r = 1; + let t = x >>> 16; + if (t !== 0) { + x = t; + r += 16; + } + t = x >> 8; + if (t !== 0) { + x = t; + r += 8; + } + t = x >> 4; + if (t !== 0) { + x = t; + r += 4; + } + t = x >> 2; + if (t !== 0) { + x = t; + r += 2; + } + t = x >> 1; + if (t !== 0) { + x = t; + r += 1; + } + return r; + }, + + /** + * If S[1] == 0, then double(S) == (S[2..128] || 0); + * otherwise, double(S) == (S[2..128] || 0) xor + * (zeros(120) || 10000111). + * + * Both OCB and EAX (through CMAC) require this function to be constant-time. + * + * @param {Uint8Array} data + */ + double: function(data) { + const doubleVar = new Uint8Array(data.length); + const last = data.length - 1; + for (let i = 0; i < last; i++) { + doubleVar[i] = (data[i] << 1) ^ (data[i + 1] >> 7); + } + doubleVar[last] = (data[last] << 1) ^ ((data[0] >> 7) * 0x87); + return doubleVar; + }, + + /** + * Shift a Uint8Array to the right by n bits + * @param {Uint8Array} array - The array to shift + * @param {Integer} bits - Amount of bits to shift (MUST be smaller + * than 8) + * @returns {String} Resulting array. + */ + shiftRight: function (array, bits) { + if (bits) { + for (let i = array.length - 1; i >= 0; i--) { + array[i] >>= bits; + if (i > 0) { + array[i] |= (array[i - 1] << (8 - bits)); + } + } + } + return array; + }, + + /** + * Get native Web Cryptography api, only the current version of the spec. + * @returns {Object} The SubtleCrypto api or 'undefined'. + */ + getWebCrypto: function() { + return typeof globalThis !== 'undefined' && globalThis.crypto && globalThis.crypto.subtle; + }, + + /** + * Get BigInteger class + * It wraps the native BigInt type if it's available + * Otherwise it relies on bn.js + * @returns {BigInteger} + * @async + */ + getBigInteger, + + /** + * Get native Node.js crypto api. + * @returns {Object} The crypto module or 'undefined'. + */ + getNodeCrypto: function() { + return void('crypto'); + }, + + getNodeZlib: function() { + return void('zlib'); + }, + + /** + * Get native Node.js Buffer constructor. This should be used since + * Buffer is not available under browserify. + * @returns {Function} The Buffer constructor or 'undefined'. + */ + getNodeBuffer: function() { + return ({}).Buffer; + }, + + getHardwareConcurrency: function() { + if (typeof navigator !== 'undefined') { + return navigator.hardwareConcurrency || 1; + } + + const os = void('os'); // Assume we're on Node.js. + return os.cpus().length; + }, + + isEmailAddress: function(data) { + if (!util.isString(data)) { + return false; + } + const re = /^(([^<>()[\]\\.,;:\s@"]+(\.[^<>()[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+([a-zA-Z]{2,}|xn--[a-zA-Z\-0-9]+)))$/; + return re.test(data); + }, + + /** + * Normalize line endings to + * Support any encoding where CR=0x0D, LF=0x0A + */ + canonicalizeEOL: function(data) { + const CR = 13; + const LF = 10; + let carryOverCR = false; + + return transform(data, bytes => { + if (carryOverCR) { + bytes = util.concatUint8Array([new Uint8Array([CR]), bytes]); + } + + if (bytes[bytes.length - 1] === CR) { + carryOverCR = true; + bytes = bytes.subarray(0, -1); + } else { + carryOverCR = false; + } + + let index; + const indices = []; + for (let i = 0; ; i = index) { + index = bytes.indexOf(LF, i) + 1; + if (index) { + if (bytes[index - 2] !== CR) indices.push(index); + } else { + break; + } + } + if (!indices.length) { + return bytes; + } + + const normalized = new Uint8Array(bytes.length + indices.length); + let j = 0; + for (let i = 0; i < indices.length; i++) { + const sub = bytes.subarray(indices[i - 1] || 0, indices[i]); + normalized.set(sub, j); + j += sub.length; + normalized[j - 1] = CR; + normalized[j] = LF; + j++; + } + normalized.set(bytes.subarray(indices[indices.length - 1] || 0), j); + return normalized; + }, () => (carryOverCR ? new Uint8Array([CR]) : undefined)); + }, + + /** + * Convert line endings from canonicalized to native + * Support any encoding where CR=0x0D, LF=0x0A + */ + nativeEOL: function(data) { + const CR = 13; + const LF = 10; + let carryOverCR = false; + + return transform(data, bytes => { + if (carryOverCR && bytes[0] !== LF) { + bytes = util.concatUint8Array([new Uint8Array([CR]), bytes]); + } else { + bytes = new Uint8Array(bytes); // Don't mutate passed bytes + } + + if (bytes[bytes.length - 1] === CR) { + carryOverCR = true; + bytes = bytes.subarray(0, -1); + } else { + carryOverCR = false; + } + + let index; + let j = 0; + for (let i = 0; i !== bytes.length; i = index) { + index = bytes.indexOf(CR, i) + 1; + if (!index) index = bytes.length; + const last = index - (bytes[index] === LF ? 1 : 0); + if (i) bytes.copyWithin(j, i, last); + j += last - i; + } + return bytes.subarray(0, j); + }, () => (carryOverCR ? new Uint8Array([CR]) : undefined)); + }, + + /** + * Remove trailing spaces, carriage returns and tabs from each line + */ + removeTrailingSpaces: function(text) { + return text.split('\n').map(line => { + let i = line.length - 1; + for (; i >= 0 && (line[i] === ' ' || line[i] === '\t' || line[i] === '\r'); i--); + return line.substr(0, i + 1); + }).join('\n'); + }, + + wrapError: function(message, error) { + if (!error) { + return new Error(message); + } + + // update error message + try { + error.message = message + ': ' + error.message; + } catch (e) {} + + return error; + }, + + /** + * Map allowed packet tags to corresponding classes + * Meant to be used to format `allowedPacket` for Packetlist.read + * @param {Array} allowedClasses + * @returns {Object} map from enum.packet to corresponding *Packet class + */ + constructAllowedPackets: function(allowedClasses) { + const map = {}; + allowedClasses.forEach(PacketClass => { + if (!PacketClass.tag) { + throw new Error('Invalid input: expected a packet class'); + } + map[PacketClass.tag] = PacketClass; + }); + return map; + }, + + /** + * Return a Promise that will resolve as soon as one of the promises in input resolves + * or will reject if all input promises all rejected + * (similar to Promise.any, but with slightly different error handling) + * @param {Array} promises + * @return {Promise} Promise resolving to the result of the fastest fulfilled promise + * or rejected with the Error of the last resolved Promise (if all promises are rejected) + */ + anyPromise: function(promises) { + // eslint-disable-next-line no-async-promise-executor + return new Promise(async (resolve, reject) => { + let exception; + await Promise.all(promises.map(async promise => { + try { + resolve(await promise); + } catch (e) { + exception = e; + } + })); + reject(exception); + }); + }, + + /** + * Return either `a` or `b` based on `cond`, in algorithmic constant time. + * @param {Boolean} cond + * @param {Uint8Array} a + * @param {Uint8Array} b + * @returns `a` if `cond` is true, `b` otherwise + */ + selectUint8Array: function(cond, a, b) { + const length = Math.max(a.length, b.length); + const result = new Uint8Array(length); + let end = 0; + for (let i = 0; i < result.length; i++) { + result[i] = (a[i] & (256 - cond)) | (b[i] & (255 + cond)); + end += (cond & i < a.length) | ((1 - cond) & i < b.length); + } + return result.subarray(0, end); + }, + /** + * Return either `a` or `b` based on `cond`, in algorithmic constant time. + * NB: it only supports `a, b` with values between 0-255. + * @param {Boolean} cond + * @param {Uint8} a + * @param {Uint8} b + * @returns `a` if `cond` is true, `b` otherwise + */ + selectUint8: function(cond, a, b) { + return (a & (256 - cond)) | (b & (255 + cond)); + } + }; + + /* OpenPGP radix-64/base64 string encoding/decoding + * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de + * version 1.0, check www.haneWIN.de for the latest version + * + * This software is provided as-is, without express or implied warranty. + * Permission to use, copy, modify, distribute or sell this software, with or + * without fee, for any purpose and by any individual or organization, is hereby + * granted, provided that the above copyright notice and this paragraph appear + * in all copies. Distribution as a part of an application or binary must + * include the above copyright notice in the documentation and/or other materials + * provided with the application or distribution. + */ + + const Buffer = util.getNodeBuffer(); + + let encodeChunk; + let decodeChunk; + if (Buffer) { + encodeChunk = buf => Buffer.from(buf).toString('base64'); + decodeChunk = str => { + const b = Buffer.from(str, 'base64'); + return new Uint8Array(b.buffer, b.byteOffset, b.byteLength); + }; + } else { + encodeChunk = buf => btoa(util.uint8ArrayToString(buf)); + decodeChunk = str => util.stringToUint8Array(atob(str)); + } + + /** + * Convert binary array to radix-64 + * @param {Uint8Array | ReadableStream} data - Uint8Array to convert + * @returns {String | ReadableStream} Radix-64 version of input string. + * @static + */ + function encode(data) { + let buf = new Uint8Array(); + return transform(data, value => { + buf = util.concatUint8Array([buf, value]); + const r = []; + const bytesPerLine = 45; // 60 chars per line * (3 bytes / 4 chars of base64). + const lines = Math.floor(buf.length / bytesPerLine); + const bytes = lines * bytesPerLine; + const encoded = encodeChunk(buf.subarray(0, bytes)); + for (let i = 0; i < lines; i++) { + r.push(encoded.substr(i * 60, 60)); + r.push('\n'); + } + buf = buf.subarray(bytes); + return r.join(''); + }, () => (buf.length ? encodeChunk(buf) + '\n' : '')); + } + + /** + * Convert radix-64 to binary array + * @param {String | ReadableStream} data - Radix-64 string to convert + * @returns {Uint8Array | ReadableStream} Binary array version of input string. + * @static + */ + function decode(data) { + let buf = ''; + return transform(data, value => { + buf += value; + + // Count how many whitespace characters there are in buf + let spaces = 0; + const spacechars = [' ', '\t', '\r', '\n']; + for (let i = 0; i < spacechars.length; i++) { + const spacechar = spacechars[i]; + for (let pos = buf.indexOf(spacechar); pos !== -1; pos = buf.indexOf(spacechar, pos + 1)) { + spaces++; + } + } + + // Backtrack until we have 4n non-whitespace characters + // that we can safely base64-decode + let length = buf.length; + for (; length > 0 && (length - spaces) % 4 !== 0; length--) { + if (spacechars.includes(buf[length])) spaces--; + } + + const decoded = decodeChunk(buf.substr(0, length)); + buf = buf.substr(length); + return decoded; + }, () => decodeChunk(buf)); + } + + /** + * Convert a Base-64 encoded string an array of 8-bit integer + * + * Note: accepts both Radix-64 and URL-safe strings + * @param {String} base64 - Base-64 encoded string to convert + * @returns {Uint8Array} An array of 8-bit integers. + */ + function b64ToUint8Array(base64) { + return decode(base64.replace(/-/g, '+').replace(/_/g, '/')); + } + + /** + * Convert an array of 8-bit integer to a Base-64 encoded string + * @param {Uint8Array} bytes - An array of 8-bit integers to convert + * @param {bool} url - If true, output is URL-safe + * @returns {String} Base-64 encoded string. + */ + function uint8ArrayToB64(bytes, url) { + let encoded = encode(bytes).replace(/[\r\n]/g, ''); + if (url) { + encoded = encoded.replace(/[+]/g, '-').replace(/[/]/g, '_').replace(/[=]/g, ''); + } + return encoded; + } + + /** + * @module enums + */ + + const byValue = Symbol('byValue'); + + var enums = { + + /** Maps curve names under various standards to one + * @see {@link https://wiki.gnupg.org/ECC|ECC - GnuPG wiki} + * @enum {String} + * @readonly + */ + curve: { + /** NIST P-256 Curve */ + 'p256': 'p256', + 'P-256': 'p256', + 'secp256r1': 'p256', + 'prime256v1': 'p256', + '1.2.840.10045.3.1.7': 'p256', + '2a8648ce3d030107': 'p256', + '2A8648CE3D030107': 'p256', + + /** NIST P-384 Curve */ + 'p384': 'p384', + 'P-384': 'p384', + 'secp384r1': 'p384', + '1.3.132.0.34': 'p384', + '2b81040022': 'p384', + '2B81040022': 'p384', + + /** NIST P-521 Curve */ + 'p521': 'p521', + 'P-521': 'p521', + 'secp521r1': 'p521', + '1.3.132.0.35': 'p521', + '2b81040023': 'p521', + '2B81040023': 'p521', + + /** SECG SECP256k1 Curve */ + 'secp256k1': 'secp256k1', + '1.3.132.0.10': 'secp256k1', + '2b8104000a': 'secp256k1', + '2B8104000A': 'secp256k1', + + /** Ed25519 */ + 'ED25519': 'ed25519', + 'ed25519': 'ed25519', + 'Ed25519': 'ed25519', + '1.3.6.1.4.1.11591.15.1': 'ed25519', + '2b06010401da470f01': 'ed25519', + '2B06010401DA470F01': 'ed25519', + + /** Curve25519 */ + 'X25519': 'curve25519', + 'cv25519': 'curve25519', + 'curve25519': 'curve25519', + 'Curve25519': 'curve25519', + '1.3.6.1.4.1.3029.1.5.1': 'curve25519', + '2b060104019755010501': 'curve25519', + '2B060104019755010501': 'curve25519', + + /** BrainpoolP256r1 Curve */ + 'brainpoolP256r1': 'brainpoolP256r1', + '1.3.36.3.3.2.8.1.1.7': 'brainpoolP256r1', + '2b2403030208010107': 'brainpoolP256r1', + '2B2403030208010107': 'brainpoolP256r1', + + /** BrainpoolP384r1 Curve */ + 'brainpoolP384r1': 'brainpoolP384r1', + '1.3.36.3.3.2.8.1.1.11': 'brainpoolP384r1', + '2b240303020801010b': 'brainpoolP384r1', + '2B240303020801010B': 'brainpoolP384r1', + + /** BrainpoolP512r1 Curve */ + 'brainpoolP512r1': 'brainpoolP512r1', + '1.3.36.3.3.2.8.1.1.13': 'brainpoolP512r1', + '2b240303020801010d': 'brainpoolP512r1', + '2B240303020801010D': 'brainpoolP512r1' + }, + + /** A string to key specifier type + * @enum {Integer} + * @readonly + */ + s2k: { + simple: 0, + salted: 1, + iterated: 3, + gnu: 101 + }, + + /** {@link https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-04#section-9.1|RFC4880bis-04, section 9.1} + * @enum {Integer} + * @readonly + */ + publicKey: { + /** RSA (Encrypt or Sign) [HAC] */ + rsaEncryptSign: 1, + /** RSA (Encrypt only) [HAC] */ + rsaEncrypt: 2, + /** RSA (Sign only) [HAC] */ + rsaSign: 3, + /** Elgamal (Encrypt only) [ELGAMAL] [HAC] */ + elgamal: 16, + /** DSA (Sign only) [FIPS186] [HAC] */ + dsa: 17, + /** ECDH (Encrypt only) [RFC6637] */ + ecdh: 18, + /** ECDSA (Sign only) [RFC6637] */ + ecdsa: 19, + /** EdDSA (Sign only) + * [{@link https://tools.ietf.org/html/draft-koch-eddsa-for-openpgp-04|Draft RFC}] */ + eddsa: 22, + /** Reserved for AEDH */ + aedh: 23, + /** Reserved for AEDSA */ + aedsa: 24 + }, + + /** {@link https://tools.ietf.org/html/rfc4880#section-9.2|RFC4880, section 9.2} + * @enum {Integer} + * @readonly + */ + symmetric: { + plaintext: 0, + /** Not implemented! */ + idea: 1, + tripledes: 2, + cast5: 3, + blowfish: 4, + aes128: 7, + aes192: 8, + aes256: 9, + twofish: 10 + }, + + /** {@link https://tools.ietf.org/html/rfc4880#section-9.3|RFC4880, section 9.3} + * @enum {Integer} + * @readonly + */ + compression: { + uncompressed: 0, + /** RFC1951 */ + zip: 1, + /** RFC1950 */ + zlib: 2, + bzip2: 3 + }, + + /** {@link https://tools.ietf.org/html/rfc4880#section-9.4|RFC4880, section 9.4} + * @enum {Integer} + * @readonly + */ + hash: { + md5: 1, + sha1: 2, + ripemd: 3, + sha256: 8, + sha384: 9, + sha512: 10, + sha224: 11 + }, + + /** A list of hash names as accepted by webCrypto functions. + * {@link https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto/digest|Parameters, algo} + * @enum {String} + */ + webHash: { + 'SHA-1': 2, + 'SHA-256': 8, + 'SHA-384': 9, + 'SHA-512': 10 + }, + + /** {@link https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-04#section-9.6|RFC4880bis-04, section 9.6} + * @enum {Integer} + * @readonly + */ + aead: { + eax: 1, + ocb: 2, + experimentalGCM: 100 // Private algorithm + }, + + /** A list of packet types and numeric tags associated with them. + * @enum {Integer} + * @readonly + */ + packet: { + publicKeyEncryptedSessionKey: 1, + signature: 2, + symEncryptedSessionKey: 3, + onePassSignature: 4, + secretKey: 5, + publicKey: 6, + secretSubkey: 7, + compressedData: 8, + symmetricallyEncryptedData: 9, + marker: 10, + literalData: 11, + trust: 12, + userID: 13, + publicSubkey: 14, + userAttribute: 17, + symEncryptedIntegrityProtectedData: 18, + modificationDetectionCode: 19, + aeadEncryptedData: 20 // see IETF draft: https://tools.ietf.org/html/draft-ford-openpgp-format-00#section-2.1 + }, + + /** Data types in the literal packet + * @enum {Integer} + * @readonly + */ + literal: { + /** Binary data 'b' */ + binary: 'b'.charCodeAt(), + /** Text data 't' */ + text: 't'.charCodeAt(), + /** Utf8 data 'u' */ + utf8: 'u'.charCodeAt(), + /** MIME message body part 'm' */ + mime: 'm'.charCodeAt() + }, + + + /** One pass signature packet type + * @enum {Integer} + * @readonly + */ + signature: { + /** 0x00: Signature of a binary document. */ + binary: 0, + /** 0x01: Signature of a canonical text document. + * + * Canonicalyzing the document by converting line endings. */ + text: 1, + /** 0x02: Standalone signature. + * + * This signature is a signature of only its own subpacket contents. + * It is calculated identically to a signature over a zero-lengh + * binary document. Note that it doesn't make sense to have a V3 + * standalone signature. */ + standalone: 2, + /** 0x10: Generic certification of a User ID and Public-Key packet. + * + * The issuer of this certification does not make any particular + * assertion as to how well the certifier has checked that the owner + * of the key is in fact the person described by the User ID. */ + certGeneric: 16, + /** 0x11: Persona certification of a User ID and Public-Key packet. + * + * The issuer of this certification has not done any verification of + * the claim that the owner of this key is the User ID specified. */ + certPersona: 17, + /** 0x12: Casual certification of a User ID and Public-Key packet. + * + * The issuer of this certification has done some casual + * verification of the claim of identity. */ + certCasual: 18, + /** 0x13: Positive certification of a User ID and Public-Key packet. + * + * The issuer of this certification has done substantial + * verification of the claim of identity. + * + * Most OpenPGP implementations make their "key signatures" as 0x10 + * certifications. Some implementations can issue 0x11-0x13 + * certifications, but few differentiate between the types. */ + certPositive: 19, + /** 0x30: Certification revocation signature + * + * This signature revokes an earlier User ID certification signature + * (signature class 0x10 through 0x13) or direct-key signature + * (0x1F). It should be issued by the same key that issued the + * revoked signature or an authorized revocation key. The signature + * is computed over the same data as the certificate that it + * revokes, and should have a later creation date than that + * certificate. */ + certRevocation: 48, + /** 0x18: Subkey Binding Signature + * + * This signature is a statement by the top-level signing key that + * indicates that it owns the subkey. This signature is calculated + * directly on the primary key and subkey, and not on any User ID or + * other packets. A signature that binds a signing subkey MUST have + * an Embedded Signature subpacket in this binding signature that + * contains a 0x19 signature made by the signing subkey on the + * primary key and subkey. */ + subkeyBinding: 24, + /** 0x19: Primary Key Binding Signature + * + * This signature is a statement by a signing subkey, indicating + * that it is owned by the primary key and subkey. This signature + * is calculated the same way as a 0x18 signature: directly on the + * primary key and subkey, and not on any User ID or other packets. + * + * When a signature is made over a key, the hash data starts with the + * octet 0x99, followed by a two-octet length of the key, and then body + * of the key packet. (Note that this is an old-style packet header for + * a key packet with two-octet length.) A subkey binding signature + * (type 0x18) or primary key binding signature (type 0x19) then hashes + * the subkey using the same format as the main key (also using 0x99 as + * the first octet). */ + keyBinding: 25, + /** 0x1F: Signature directly on a key + * + * This signature is calculated directly on a key. It binds the + * information in the Signature subpackets to the key, and is + * appropriate to be used for subpackets that provide information + * about the key, such as the Revocation Key subpacket. It is also + * appropriate for statements that non-self certifiers want to make + * about the key itself, rather than the binding between a key and a + * name. */ + key: 31, + /** 0x20: Key revocation signature + * + * The signature is calculated directly on the key being revoked. A + * revoked key is not to be used. Only revocation signatures by the + * key being revoked, or by an authorized revocation key, should be + * considered valid revocation signatures.a */ + keyRevocation: 32, + /** 0x28: Subkey revocation signature + * + * The signature is calculated directly on the subkey being revoked. + * A revoked subkey is not to be used. Only revocation signatures + * by the top-level signature key that is bound to this subkey, or + * by an authorized revocation key, should be considered valid + * revocation signatures. + * + * Key revocation signatures (types 0x20 and 0x28) + * hash only the key being revoked. */ + subkeyRevocation: 40, + /** 0x40: Timestamp signature. + * This signature is only meaningful for the timestamp contained in + * it. */ + timestamp: 64, + /** 0x50: Third-Party Confirmation signature. + * + * This signature is a signature over some other OpenPGP Signature + * packet(s). It is analogous to a notary seal on the signed data. + * A third-party signature SHOULD include Signature Target + * subpacket(s) to give easy identification. Note that we really do + * mean SHOULD. There are plausible uses for this (such as a blind + * party that only sees the signature, not the key or source + * document) that cannot include a target subpacket. */ + thirdParty: 80 + }, + + /** Signature subpacket type + * @enum {Integer} + * @readonly + */ + signatureSubpacket: { + signatureCreationTime: 2, + signatureExpirationTime: 3, + exportableCertification: 4, + trustSignature: 5, + regularExpression: 6, + revocable: 7, + keyExpirationTime: 9, + placeholderBackwardsCompatibility: 10, + preferredSymmetricAlgorithms: 11, + revocationKey: 12, + issuer: 16, + notationData: 20, + preferredHashAlgorithms: 21, + preferredCompressionAlgorithms: 22, + keyServerPreferences: 23, + preferredKeyServer: 24, + primaryUserID: 25, + policyURI: 26, + keyFlags: 27, + signersUserID: 28, + reasonForRevocation: 29, + features: 30, + signatureTarget: 31, + embeddedSignature: 32, + issuerFingerprint: 33, + preferredAEADAlgorithms: 34 + }, + + /** Key flags + * @enum {Integer} + * @readonly + */ + keyFlags: { + /** 0x01 - This key may be used to certify other keys. */ + certifyKeys: 1, + /** 0x02 - This key may be used to sign data. */ + signData: 2, + /** 0x04 - This key may be used to encrypt communications. */ + encryptCommunication: 4, + /** 0x08 - This key may be used to encrypt storage. */ + encryptStorage: 8, + /** 0x10 - The private component of this key may have been split + * by a secret-sharing mechanism. */ + splitPrivateKey: 16, + /** 0x20 - This key may be used for authentication. */ + authentication: 32, + /** 0x80 - The private component of this key may be in the + * possession of more than one person. */ + sharedPrivateKey: 128 + }, + + /** Armor type + * @enum {Integer} + * @readonly + */ + armor: { + multipartSection: 0, + multipartLast: 1, + signed: 2, + message: 3, + publicKey: 4, + privateKey: 5, + signature: 6 + }, + + /** {@link https://tools.ietf.org/html/rfc4880#section-5.2.3.23|RFC4880, section 5.2.3.23} + * @enum {Integer} + * @readonly + */ + reasonForRevocation: { + /** No reason specified (key revocations or cert revocations) */ + noReason: 0, + /** Key is superseded (key revocations) */ + keySuperseded: 1, + /** Key material has been compromised (key revocations) */ + keyCompromised: 2, + /** Key is retired and no longer used (key revocations) */ + keyRetired: 3, + /** User ID information is no longer valid (cert revocations) */ + userIDInvalid: 32 + }, + + /** {@link https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-04#section-5.2.3.25|RFC4880bis-04, section 5.2.3.25} + * @enum {Integer} + * @readonly + */ + features: { + /** 0x01 - Modification Detection (packets 18 and 19) */ + modificationDetection: 1, + /** 0x02 - AEAD Encrypted Data Packet (packet 20) and version 5 + * Symmetric-Key Encrypted Session Key Packets (packet 3) */ + aead: 2, + /** 0x04 - Version 5 Public-Key Packet format and corresponding new + * fingerprint format */ + v5Keys: 4 + }, + + /** + * Asserts validity of given value and converts from string/integer to integer. + * @param {Object} type target enum type + * @param {String|Integer} e value to check and/or convert + * @returns {Integer} enum value if it exists + * @throws {Error} if the value is invalid + */ + write: function(type, e) { + if (typeof e === 'number') { + e = this.read(type, e); + } + + if (type[e] !== undefined) { + return type[e]; + } + + throw new Error('Invalid enum value.'); + }, + + /** + * Converts enum integer value to the corresponding string, if it exists. + * @param {Object} type target enum type + * @param {Integer} e value to convert + * @returns {String} name of enum value if it exists + * @throws {Error} if the value is invalid + */ + read: function(type, e) { + if (!type[byValue]) { + type[byValue] = []; + Object.entries(type).forEach(([key, value]) => { + type[byValue][value] = key; + }); + } + + if (type[byValue][e] !== undefined) { + return type[byValue][e]; + } + + throw new Error('Invalid enum value.'); + } + }; + + // GPG4Browsers - An OpenPGP implementation in javascript + + var config = { + /** + * @memberof module:config + * @property {Integer} preferredHashAlgorithm Default hash algorithm {@link module:enums.hash} + */ + preferredHashAlgorithm: enums.hash.sha256, + /** + * @memberof module:config + * @property {Integer} preferredSymmetricAlgorithm Default encryption cipher {@link module:enums.symmetric} + */ + preferredSymmetricAlgorithm: enums.symmetric.aes256, + /** + * @memberof module:config + * @property {Integer} compression Default compression algorithm {@link module:enums.compression} + */ + preferredCompressionAlgorithm: enums.compression.uncompressed, + /** + * @memberof module:config + * @property {Integer} deflateLevel Default zip/zlib compression level, between 1 and 9 + */ + deflateLevel: 6, + + /** + * Use Authenticated Encryption with Additional Data (AEAD) protection for symmetric encryption. + * Note: not all OpenPGP implementations are compatible with this option. + * **FUTURE OPENPGP.JS VERSIONS MAY BREAK COMPATIBILITY WHEN USING THIS OPTION** + * @see {@link https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-07|RFC4880bis-07} + * @memberof module:config + * @property {Boolean} aeadProtect + */ + aeadProtect: false, + /** + * Default Authenticated Encryption with Additional Data (AEAD) encryption mode + * Only has an effect when aeadProtect is set to true. + * @memberof module:config + * @property {Integer} preferredAEADAlgorithm Default AEAD mode {@link module:enums.aead} + */ + preferredAEADAlgorithm: enums.aead.eax, + /** + * Chunk Size Byte for Authenticated Encryption with Additional Data (AEAD) mode + * Only has an effect when aeadProtect is set to true. + * Must be an integer value from 0 to 56. + * @memberof module:config + * @property {Integer} aeadChunkSizeByte + */ + aeadChunkSizeByte: 12, + /** + * Use V5 keys. + * Note: not all OpenPGP implementations are compatible with this option. + * **FUTURE OPENPGP.JS VERSIONS MAY BREAK COMPATIBILITY WHEN USING THIS OPTION** + * @memberof module:config + * @property {Boolean} v5Keys + */ + v5Keys: false, + /** + * {@link https://tools.ietf.org/html/rfc4880#section-3.7.1.3|RFC4880 3.7.1.3}: + * Iteration Count Byte for S2K (String to Key) + * @memberof module:config + * @property {Integer} s2kIterationCountByte + */ + s2kIterationCountByte: 224, + /** + * Allow decryption of messages without integrity protection. + * This is an **insecure** setting: + * - message modifications cannot be detected, thus processing the decrypted data is potentially unsafe. + * - it enables downgrade attacks against integrity-protected messages. + * @memberof module:config + * @property {Boolean} allowUnauthenticatedMessages + */ + allowUnauthenticatedMessages: false, + /** + * Allow streaming unauthenticated data before its integrity has been checked. This would allow the application to + * process large streams while limiting memory usage by releasing the decrypted chunks as soon as possible + * and deferring checking their integrity until the decrypted stream has been read in full. + * + * This setting is **insecure** if the partially decrypted message is processed further or displayed to the user. + * @memberof module:config + * @property {Boolean} allowUnauthenticatedStream + */ + allowUnauthenticatedStream: false, + /** + * @memberof module:config + * @property {Boolean} checksumRequired Do not throw error when armor is missing a checksum + */ + checksumRequired: false, + /** + * Minimum RSA key size allowed for key generation and message signing, verification and encryption. + * The default is 2047 since due to a bug, previous versions of OpenPGP.js could generate 2047-bit keys instead of 2048-bit ones. + * @memberof module:config + * @property {Number} minRSABits + */ + minRSABits: 2047, + /** + * Work-around for rare GPG decryption bug when encrypting with multiple passwords. + * **Slower and slightly less secure** + * @memberof module:config + * @property {Boolean} passwordCollisionCheck + */ + passwordCollisionCheck: false, + /** + * @memberof module:config + * @property {Boolean} revocationsExpire If true, expired revocation signatures are ignored + */ + revocationsExpire: false, + /** + * Allow decryption using RSA keys without `encrypt` flag. + * This setting is potentially insecure, but it is needed to get around an old openpgpjs bug + * where key flags were ignored when selecting a key for encryption. + * @memberof module:config + * @property {Boolean} allowInsecureDecryptionWithSigningKeys + */ + allowInsecureDecryptionWithSigningKeys: false, + /** + * Allow verification of message signatures with keys whose validity at the time of signing cannot be determined. + * Instead, a verification key will also be consider valid as long as it is valid at the current time. + * This setting is potentially insecure, but it is needed to verify messages signed with keys that were later reformatted, + * and have self-signature's creation date that does not match the primary key creation date. + * @memberof module:config + * @property {Boolean} allowInsecureDecryptionWithSigningKeys + */ + allowInsecureVerificationWithReformattedKeys: false, + + /** + * Enable constant-time decryption of RSA- and ElGamal-encrypted session keys, to hinder Bleichenbacher-like attacks (https://link.springer.com/chapter/10.1007/BFb0055716). + * This setting has measurable performance impact and it is only helpful in application scenarios where both of the following conditions apply: + * - new/incoming messages are automatically decrypted (without user interaction); + * - an attacker can determine how long it takes to decrypt each message (e.g. due to decryption errors being logged remotely). + * See also `constantTimePKCS1DecryptionSupportedSymmetricAlgorithms`. + * @memberof module:config + * @property {Boolean} constantTimePKCS1Decryption + */ + constantTimePKCS1Decryption: false, + /** + * This setting is only meaningful if `constantTimePKCS1Decryption` is enabled. + * Decryption of RSA- and ElGamal-encrypted session keys of symmetric algorithms different from the ones specified here will fail. + * However, the more algorithms are added, the slower the decryption procedure becomes. + * @memberof module:config + * @property {Set} constantTimePKCS1DecryptionSupportedSymmetricAlgorithms {@link module:enums.symmetric} + */ + constantTimePKCS1DecryptionSupportedSymmetricAlgorithms: new Set([enums.symmetric.aes128, enums.symmetric.aes192, enums.symmetric.aes256]), + + /** + * @memberof module:config + * @property {Integer} minBytesForWebCrypto The minimum amount of bytes for which to use native WebCrypto APIs when available + */ + minBytesForWebCrypto: 1000, + /** + * @memberof module:config + * @property {Boolean} ignoreUnsupportedPackets Ignore unsupported/unrecognizable packets on parsing instead of throwing an error + */ + ignoreUnsupportedPackets: true, + /** + * @memberof module:config + * @property {Boolean} ignoreMalformedPackets Ignore malformed packets on parsing instead of throwing an error + */ + ignoreMalformedPackets: false, + /** + * Parsing of packets is normally restricted to a predefined set of packets. For example a Sym. Encrypted Integrity Protected Data Packet can only + * contain a certain set of packets including LiteralDataPacket. With this setting we can allow additional packets, which is probably not advisable + * as a global config setting, but can be used for specific function calls (e.g. decrypt method of Message). + * @memberof module:config + * @property {Array} additionalAllowedPackets Allow additional packets on parsing. Defined as array of packet classes, e.g. [PublicKeyPacket] + */ + additionalAllowedPackets: [], + /** + * @memberof module:config + * @property {Boolean} showVersion Whether to include {@link module:config/config.versionString} in armored messages + */ + showVersion: false, + /** + * @memberof module:config + * @property {Boolean} showComment Whether to include {@link module:config/config.commentString} in armored messages + */ + showComment: false, + /** + * @memberof module:config + * @property {String} versionString A version string to be included in armored messages + */ + versionString: 'OpenPGP.js 5.8.0', + /** + * @memberof module:config + * @property {String} commentString A comment string to be included in armored messages + */ + commentString: 'https://openpgpjs.org', + + /** + * Max userID string length (used for parsing) + * @memberof module:config + * @property {Integer} maxUserIDLength + */ + maxUserIDLength: 1024 * 5, + /** + * Contains notatations that are considered "known". Known notations do not trigger + * validation error when the notation is marked as critical. + * @memberof module:config + * @property {Array} knownNotations + */ + knownNotations: [], + /** + * Whether to use the indutny/elliptic library for curves (other than Curve25519) that are not supported by the available native crypto API. + * When false, certain standard curves will not be supported (depending on the platform). + * Note: the indutny/elliptic curve library is not designed to be constant time. + * @memberof module:config + * @property {Boolean} useIndutnyElliptic + */ + useIndutnyElliptic: true, + /** + * Reject insecure hash algorithms + * @memberof module:config + * @property {Set} rejectHashAlgorithms {@link module:enums.hash} + */ + rejectHashAlgorithms: new Set([enums.hash.md5, enums.hash.ripemd]), + /** + * Reject insecure message hash algorithms + * @memberof module:config + * @property {Set} rejectMessageHashAlgorithms {@link module:enums.hash} + */ + rejectMessageHashAlgorithms: new Set([enums.hash.md5, enums.hash.ripemd, enums.hash.sha1]), + /** + * Reject insecure public key algorithms for key generation and message encryption, signing or verification + * @memberof module:config + * @property {Set} rejectPublicKeyAlgorithms {@link module:enums.publicKey} + */ + rejectPublicKeyAlgorithms: new Set([enums.publicKey.elgamal, enums.publicKey.dsa]), + /** + * Reject non-standard curves for key generation, message encryption, signing or verification + * @memberof module:config + * @property {Set} rejectCurves {@link module:enums.curve} + */ + rejectCurves: new Set([enums.curve.secp256k1]) + }; + + /** + * @fileoverview This object contains global configuration values. + * @see module:config/config + * @module config + */ + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Finds out which Ascii Armoring type is used. Throws error if unknown type. + * @param {String} text - ascii armored text + * @returns {Integer} 0 = MESSAGE PART n of m. + * 1 = MESSAGE PART n + * 2 = SIGNED MESSAGE + * 3 = PGP MESSAGE + * 4 = PUBLIC KEY BLOCK + * 5 = PRIVATE KEY BLOCK + * 6 = SIGNATURE + * @private + */ + function getType(text) { + const reHeader = /^-----BEGIN PGP (MESSAGE, PART \d+\/\d+|MESSAGE, PART \d+|SIGNED MESSAGE|MESSAGE|PUBLIC KEY BLOCK|PRIVATE KEY BLOCK|SIGNATURE)-----$/m; + + const header = text.match(reHeader); + + if (!header) { + throw new Error('Unknown ASCII armor type'); + } + + // BEGIN PGP MESSAGE, PART X/Y + // Used for multi-part messages, where the armor is split amongst Y + // parts, and this is the Xth part out of Y. + if (/MESSAGE, PART \d+\/\d+/.test(header[1])) { + return enums.armor.multipartSection; + } else + // BEGIN PGP MESSAGE, PART X + // Used for multi-part messages, where this is the Xth part of an + // unspecified number of parts. Requires the MESSAGE-ID Armor + // Header to be used. + if (/MESSAGE, PART \d+/.test(header[1])) { + return enums.armor.multipartLast; + } else + // BEGIN PGP SIGNED MESSAGE + if (/SIGNED MESSAGE/.test(header[1])) { + return enums.armor.signed; + } else + // BEGIN PGP MESSAGE + // Used for signed, encrypted, or compressed files. + if (/MESSAGE/.test(header[1])) { + return enums.armor.message; + } else + // BEGIN PGP PUBLIC KEY BLOCK + // Used for armoring public keys. + if (/PUBLIC KEY BLOCK/.test(header[1])) { + return enums.armor.publicKey; + } else + // BEGIN PGP PRIVATE KEY BLOCK + // Used for armoring private keys. + if (/PRIVATE KEY BLOCK/.test(header[1])) { + return enums.armor.privateKey; + } else + // BEGIN PGP SIGNATURE + // Used for detached signatures, OpenPGP/MIME signatures, and + // cleartext signatures. Note that PGP 2.x uses BEGIN PGP MESSAGE + // for detached signatures. + if (/SIGNATURE/.test(header[1])) { + return enums.armor.signature; + } + } + + /** + * Add additional information to the armor version of an OpenPGP binary + * packet block. + * @author Alex + * @version 2011-12-16 + * @param {String} [customComment] - Additional comment to add to the armored string + * @returns {String} The header information. + * @private + */ + function addheader(customComment, config) { + let result = ''; + if (config.showVersion) { + result += 'Version: ' + config.versionString + '\n'; + } + if (config.showComment) { + result += 'Comment: ' + config.commentString + '\n'; + } + if (customComment) { + result += 'Comment: ' + customComment + '\n'; + } + result += '\n'; + return result; + } + + + /** + * Calculates a checksum over the given data and returns it base64 encoded + * @param {String | ReadableStream} data - Data to create a CRC-24 checksum for + * @returns {String | ReadableStream} Base64 encoded checksum. + * @private + */ + function getCheckSum(data) { + const crc = createcrc24(data); + return encode(crc); + } + + // https://create.stephan-brumme.com/crc32/#slicing-by-8-overview + + const crc_table = [ + new Array(0xFF), + new Array(0xFF), + new Array(0xFF), + new Array(0xFF) + ]; + + for (let i = 0; i <= 0xFF; i++) { + let crc = i << 16; + for (let j = 0; j < 8; j++) { + crc = (crc << 1) ^ ((crc & 0x800000) !== 0 ? 0x864CFB : 0); + } + crc_table[0][i] = + ((crc & 0xFF0000) >> 16) | + (crc & 0x00FF00) | + ((crc & 0x0000FF) << 16); + } + for (let i = 0; i <= 0xFF; i++) { + crc_table[1][i] = (crc_table[0][i] >> 8) ^ crc_table[0][crc_table[0][i] & 0xFF]; + } + for (let i = 0; i <= 0xFF; i++) { + crc_table[2][i] = (crc_table[1][i] >> 8) ^ crc_table[0][crc_table[1][i] & 0xFF]; + } + for (let i = 0; i <= 0xFF; i++) { + crc_table[3][i] = (crc_table[2][i] >> 8) ^ crc_table[0][crc_table[2][i] & 0xFF]; + } + + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DataView#Endianness + const isLittleEndian = (function() { + const buffer = new ArrayBuffer(2); + new DataView(buffer).setInt16(0, 0xFF, true /* littleEndian */); + // Int16Array uses the platform's endianness. + return new Int16Array(buffer)[0] === 0xFF; + }()); + + /** + * Internal function to calculate a CRC-24 checksum over a given string (data) + * @param {String | ReadableStream} input - Data to create a CRC-24 checksum for + * @returns {Uint8Array | ReadableStream} The CRC-24 checksum. + * @private + */ + function createcrc24(input) { + let crc = 0xCE04B7; + return transform(input, value => { + const len32 = isLittleEndian ? Math.floor(value.length / 4) : 0; + const arr32 = new Uint32Array(value.buffer, value.byteOffset, len32); + for (let i = 0; i < len32; i++) { + crc ^= arr32[i]; + crc = + crc_table[0][(crc >> 24) & 0xFF] ^ + crc_table[1][(crc >> 16) & 0xFF] ^ + crc_table[2][(crc >> 8) & 0xFF] ^ + crc_table[3][(crc >> 0) & 0xFF]; + } + for (let i = len32 * 4; i < value.length; i++) { + crc = (crc >> 8) ^ crc_table[0][(crc & 0xFF) ^ value[i]]; + } + }, () => new Uint8Array([crc, crc >> 8, crc >> 16])); + } + + /** + * Verify armored headers. crypto-refresh-06, section 6.2: + * "An OpenPGP implementation may consider improperly formatted Armor + * Headers to be corruption of the ASCII Armor, but SHOULD make an + * effort to recover." + * @private + * @param {Array} headers - Armor headers + */ + function verifyHeaders(headers) { + for (let i = 0; i < headers.length; i++) { + if (!/^([^\s:]|[^\s:][^:]*[^\s:]): .+$/.test(headers[i])) { + util.printDebugError(new Error('Improperly formatted armor header: ' + headers[i])); + } + if (!/^(Version|Comment|MessageID|Hash|Charset): .+$/.test(headers[i])) { + util.printDebugError(new Error('Unknown header: ' + headers[i])); + } + } + } + + /** + * Splits a message into two parts, the body and the checksum. This is an internal function + * @param {String} text - OpenPGP armored message part + * @returns {Object} An object with attribute "body" containing the body. + * and an attribute "checksum" containing the checksum. + * @private + */ + function splitChecksum(text) { + let body = text; + let checksum = ''; + + const lastEquals = text.lastIndexOf('='); + + if (lastEquals >= 0 && lastEquals !== text.length - 1) { // '=' as the last char means no checksum + body = text.slice(0, lastEquals); + checksum = text.slice(lastEquals + 1).substr(0, 4); + } + + return { body: body, checksum: checksum }; + } + + /** + * Dearmor an OpenPGP armored message; verify the checksum and return + * the encoded bytes + * @param {String} input - OpenPGP armored message + * @returns {Promise} An object with attribute "text" containing the message text, + * an attribute "data" containing a stream of bytes and "type" for the ASCII armor type + * @async + * @static + */ + function unarmor(input, config$1 = config) { + // eslint-disable-next-line no-async-promise-executor + return new Promise(async (resolve, reject) => { + try { + const reSplit = /^-----[^-]+-----$/m; + const reEmptyLine = /^[ \f\r\t\u00a0\u2000-\u200a\u202f\u205f\u3000]*$/; + + let type; + const headers = []; + let lastHeaders = headers; + let headersDone; + let text = []; + let textDone; + let checksum; + let data = decode(transformPair(input, async (readable, writable) => { + const reader = getReader(readable); + try { + while (true) { + let line = await reader.readLine(); + if (line === undefined) { + throw new Error('Misformed armored text'); + } + // remove trailing whitespace at end of lines + line = util.removeTrailingSpaces(line.replace(/[\r\n]/g, '')); + if (!type) { + if (reSplit.test(line)) { + type = getType(line); + } + } else if (!headersDone) { + if (reSplit.test(line)) { + reject(new Error('Mandatory blank line missing between armor headers and armor data')); + } + if (!reEmptyLine.test(line)) { + lastHeaders.push(line); + } else { + verifyHeaders(lastHeaders); + headersDone = true; + if (textDone || type !== 2) { + resolve({ text, data, headers, type }); + break; + } + } + } else if (!textDone && type === 2) { + if (!reSplit.test(line)) { + // Reverse dash-escaping for msg + text.push(line.replace(/^- /, '')); + } else { + text = text.join('\r\n'); + textDone = true; + verifyHeaders(lastHeaders); + lastHeaders = []; + headersDone = false; + } + } + } + } catch (e) { + reject(e); + return; + } + const writer = getWriter(writable); + try { + while (true) { + await writer.ready; + const { done, value } = await reader.read(); + if (done) { + throw new Error('Misformed armored text'); + } + const line = value + ''; + if (line.indexOf('=') === -1 && line.indexOf('-') === -1) { + await writer.write(line); + } else { + let remainder = await reader.readToEnd(); + if (!remainder.length) remainder = ''; + remainder = line + remainder; + remainder = util.removeTrailingSpaces(remainder.replace(/\r/g, '')); + const parts = remainder.split(reSplit); + if (parts.length === 1) { + throw new Error('Misformed armored text'); + } + const split = splitChecksum(parts[0].slice(0, -1)); + checksum = split.checksum; + await writer.write(split.body); + break; + } + } + await writer.ready; + await writer.close(); + } catch (e) { + await writer.abort(e); + } + })); + data = transformPair(data, async (readable, writable) => { + const checksumVerified = readToEnd(getCheckSum(passiveClone(readable))); + checksumVerified.catch(() => {}); + await pipe(readable, writable, { + preventClose: true + }); + const writer = getWriter(writable); + try { + const checksumVerifiedString = (await checksumVerified).replace('\n', ''); + if (checksum !== checksumVerifiedString && (checksum || config$1.checksumRequired)) { + throw new Error('Ascii armor integrity check failed'); + } + await writer.ready; + await writer.close(); + } catch (e) { + await writer.abort(e); + } + }); + } catch (e) { + reject(e); + } + }).then(async result => { + if (isArrayStream(result.data)) { + result.data = await readToEnd(result.data); + } + return result; + }); + } + + + /** + * Armor an OpenPGP binary packet block + * @param {module:enums.armor} messageType - Type of the message + * @param {Uint8Array | ReadableStream} body - The message body to armor + * @param {Integer} [partIndex] + * @param {Integer} [partTotal] + * @param {String} [customComment] - Additional comment to add to the armored string + * @returns {String | ReadableStream} Armored text. + * @static + */ + function armor(messageType, body, partIndex, partTotal, customComment, config$1 = config) { + let text; + let hash; + if (messageType === enums.armor.signed) { + text = body.text; + hash = body.hash; + body = body.data; + } + const bodyClone = passiveClone(body); + const result = []; + switch (messageType) { + case enums.armor.multipartSection: + result.push('-----BEGIN PGP MESSAGE, PART ' + partIndex + '/' + partTotal + '-----\n'); + result.push(addheader(customComment, config$1)); + result.push(encode(body)); + result.push('=', getCheckSum(bodyClone)); + result.push('-----END PGP MESSAGE, PART ' + partIndex + '/' + partTotal + '-----\n'); + break; + case enums.armor.multipartLast: + result.push('-----BEGIN PGP MESSAGE, PART ' + partIndex + '-----\n'); + result.push(addheader(customComment, config$1)); + result.push(encode(body)); + result.push('=', getCheckSum(bodyClone)); + result.push('-----END PGP MESSAGE, PART ' + partIndex + '-----\n'); + break; + case enums.armor.signed: + result.push('-----BEGIN PGP SIGNED MESSAGE-----\n'); + result.push('Hash: ' + hash + '\n\n'); + result.push(text.replace(/^-/mg, '- -')); + result.push('\n-----BEGIN PGP SIGNATURE-----\n'); + result.push(addheader(customComment, config$1)); + result.push(encode(body)); + result.push('=', getCheckSum(bodyClone)); + result.push('-----END PGP SIGNATURE-----\n'); + break; + case enums.armor.message: + result.push('-----BEGIN PGP MESSAGE-----\n'); + result.push(addheader(customComment, config$1)); + result.push(encode(body)); + result.push('=', getCheckSum(bodyClone)); + result.push('-----END PGP MESSAGE-----\n'); + break; + case enums.armor.publicKey: + result.push('-----BEGIN PGP PUBLIC KEY BLOCK-----\n'); + result.push(addheader(customComment, config$1)); + result.push(encode(body)); + result.push('=', getCheckSum(bodyClone)); + result.push('-----END PGP PUBLIC KEY BLOCK-----\n'); + break; + case enums.armor.privateKey: + result.push('-----BEGIN PGP PRIVATE KEY BLOCK-----\n'); + result.push(addheader(customComment, config$1)); + result.push(encode(body)); + result.push('=', getCheckSum(bodyClone)); + result.push('-----END PGP PRIVATE KEY BLOCK-----\n'); + break; + case enums.armor.signature: + result.push('-----BEGIN PGP SIGNATURE-----\n'); + result.push(addheader(customComment, config$1)); + result.push(encode(body)); + result.push('=', getCheckSum(bodyClone)); + result.push('-----END PGP SIGNATURE-----\n'); + break; + } + + return util.concat(result); + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Implementation of type key id + * + * {@link https://tools.ietf.org/html/rfc4880#section-3.3|RFC4880 3.3}: + * A Key ID is an eight-octet scalar that identifies a key. + * Implementations SHOULD NOT assume that Key IDs are unique. The + * section "Enhanced Key Formats" below describes how Key IDs are + * formed. + */ + class KeyID { + constructor() { + this.bytes = ''; + } + + /** + * Parsing method for a key id + * @param {Uint8Array} bytes - Input to read the key id from + */ + read(bytes) { + this.bytes = util.uint8ArrayToString(bytes.subarray(0, 8)); + } + + /** + * Serializes the Key ID + * @returns {Uint8Array} Key ID as a Uint8Array. + */ + write() { + return util.stringToUint8Array(this.bytes); + } + + /** + * Returns the Key ID represented as a hexadecimal string + * @returns {String} Key ID as a hexadecimal string. + */ + toHex() { + return util.uint8ArrayToHex(util.stringToUint8Array(this.bytes)); + } + + /** + * Checks equality of Key ID's + * @param {KeyID} keyID + * @param {Boolean} matchWildcard - Indicates whether to check if either keyID is a wildcard + */ + equals(keyID, matchWildcard = false) { + return (matchWildcard && (keyID.isWildcard() || this.isWildcard())) || this.bytes === keyID.bytes; + } + + /** + * Checks to see if the Key ID is unset + * @returns {Boolean} True if the Key ID is null. + */ + isNull() { + return this.bytes === ''; + } + + /** + * Checks to see if the Key ID is a "wildcard" Key ID (all zeros) + * @returns {Boolean} True if this is a wildcard Key ID. + */ + isWildcard() { + return /^0+$/.test(this.toHex()); + } + + static mapToHex(keyID) { + return keyID.toHex(); + } + + static fromID(hex) { + const keyID = new KeyID(); + keyID.read(util.hexToUint8Array(hex)); + return keyID; + } + + static wildcard() { + const keyID = new KeyID(); + keyID.read(new Uint8Array(8)); + return keyID; + } + } + + /** + * @file {@link http://asmjs.org Asm.js} implementation of the {@link https://en.wikipedia.org/wiki/Advanced_Encryption_Standard Advanced Encryption Standard}. + * @author Artem S Vybornov + * @license MIT + */ + var AES_asm = function () { + + /** + * Galois Field stuff init flag + */ + var ginit_done = false; + + /** + * Galois Field exponentiation and logarithm tables for 3 (the generator) + */ + var gexp3, glog3; + + /** + * Init Galois Field tables + */ + function ginit() { + gexp3 = [], + glog3 = []; + + var a = 1, c, d; + for (c = 0; c < 255; c++) { + gexp3[c] = a; + + // Multiply by three + d = a & 0x80, a <<= 1, a &= 255; + if (d === 0x80) a ^= 0x1b; + a ^= gexp3[c]; + + // Set the log table value + glog3[gexp3[c]] = c; + } + gexp3[255] = gexp3[0]; + glog3[0] = 0; + + ginit_done = true; + } + + /** + * Galois Field multiplication + * @param {number} a + * @param {number} b + * @return {number} + */ + function gmul(a, b) { + var c = gexp3[(glog3[a] + glog3[b]) % 255]; + if (a === 0 || b === 0) c = 0; + return c; + } + + /** + * Galois Field reciprocal + * @param {number} a + * @return {number} + */ + function ginv(a) { + var i = gexp3[255 - glog3[a]]; + if (a === 0) i = 0; + return i; + } + + /** + * AES stuff init flag + */ + var aes_init_done = false; + + /** + * Encryption, Decryption, S-Box and KeyTransform tables + * + * @type {number[]} + */ + var aes_sbox; + + /** + * @type {number[]} + */ + var aes_sinv; + + /** + * @type {number[][]} + */ + var aes_enc; + + /** + * @type {number[][]} + */ + var aes_dec; + + /** + * Init AES tables + */ + function aes_init() { + if (!ginit_done) ginit(); + + // Calculates AES S-Box value + function _s(a) { + var c, s, x; + s = x = ginv(a); + for (c = 0; c < 4; c++) { + s = ((s << 1) | (s >>> 7)) & 255; + x ^= s; + } + x ^= 99; + return x; + } + + // Tables + aes_sbox = [], + aes_sinv = [], + aes_enc = [[], [], [], []], + aes_dec = [[], [], [], []]; + + for (var i = 0; i < 256; i++) { + var s = _s(i); + + // S-Box and its inverse + aes_sbox[i] = s; + aes_sinv[s] = i; + + // Ecryption and Decryption tables + aes_enc[0][i] = (gmul(2, s) << 24) | (s << 16) | (s << 8) | gmul(3, s); + aes_dec[0][s] = (gmul(14, i) << 24) | (gmul(9, i) << 16) | (gmul(13, i) << 8) | gmul(11, i); + // Rotate tables + for (var t = 1; t < 4; t++) { + aes_enc[t][i] = (aes_enc[t - 1][i] >>> 8) | (aes_enc[t - 1][i] << 24); + aes_dec[t][s] = (aes_dec[t - 1][s] >>> 8) | (aes_dec[t - 1][s] << 24); + } + } + + aes_init_done = true; + } + + /** + * Asm.js module constructor. + * + *

+ * Heap buffer layout by offset: + *

+     * 0x0000   encryption key schedule
+     * 0x0400   decryption key schedule
+     * 0x0800   sbox
+     * 0x0c00   inv sbox
+     * 0x1000   encryption tables
+     * 0x2000   decryption tables
+     * 0x3000   reserved (future GCM multiplication lookup table)
+     * 0x4000   data
+     * 
+ * Don't touch anything before 0x400. + *

+ * + * @alias AES_asm + * @class + * @param foreign - ignored + * @param buffer - heap buffer to link with + */ + var wrapper = function (foreign, buffer) { + // Init AES stuff for the first time + if (!aes_init_done) aes_init(); + + // Fill up AES tables + var heap = new Uint32Array(buffer); + heap.set(aes_sbox, 0x0800 >> 2); + heap.set(aes_sinv, 0x0c00 >> 2); + for (var i = 0; i < 4; i++) { + heap.set(aes_enc[i], (0x1000 + 0x400 * i) >> 2); + heap.set(aes_dec[i], (0x2000 + 0x400 * i) >> 2); + } + + /** + * Calculate AES key schedules. + * @instance + * @memberof AES_asm + * @param {number} ks - key size, 4/6/8 (for 128/192/256-bit key correspondingly) + * @param {number} k0 - key vector components + * @param {number} k1 - key vector components + * @param {number} k2 - key vector components + * @param {number} k3 - key vector components + * @param {number} k4 - key vector components + * @param {number} k5 - key vector components + * @param {number} k6 - key vector components + * @param {number} k7 - key vector components + */ + function set_key(ks, k0, k1, k2, k3, k4, k5, k6, k7) { + var ekeys = heap.subarray(0x000, 60), + dkeys = heap.subarray(0x100, 0x100 + 60); + + // Encryption key schedule + ekeys.set([k0, k1, k2, k3, k4, k5, k6, k7]); + for (var i = ks, rcon = 1; i < 4 * ks + 28; i++) { + var k = ekeys[i - 1]; + if ((i % ks === 0) || (ks === 8 && i % ks === 4)) { + k = aes_sbox[k >>> 24] << 24 ^ aes_sbox[k >>> 16 & 255] << 16 ^ aes_sbox[k >>> 8 & 255] << 8 ^ aes_sbox[k & 255]; + } + if (i % ks === 0) { + k = (k << 8) ^ (k >>> 24) ^ (rcon << 24); + rcon = (rcon << 1) ^ ((rcon & 0x80) ? 0x1b : 0); + } + ekeys[i] = ekeys[i - ks] ^ k; + } + + // Decryption key schedule + for (var j = 0; j < i; j += 4) { + for (var jj = 0; jj < 4; jj++) { + var k = ekeys[i - (4 + j) + (4 - jj) % 4]; + if (j < 4 || j >= i - 4) { + dkeys[j + jj] = k; + } else { + dkeys[j + jj] = aes_dec[0][aes_sbox[k >>> 24]] + ^ aes_dec[1][aes_sbox[k >>> 16 & 255]] + ^ aes_dec[2][aes_sbox[k >>> 8 & 255]] + ^ aes_dec[3][aes_sbox[k & 255]]; + } + } + } + + // Set rounds number + asm.set_rounds(ks + 5); + } + + // create library object with necessary properties + var stdlib = {Uint8Array: Uint8Array, Uint32Array: Uint32Array}; + + var asm = function (stdlib, foreign, buffer) { + "use asm"; + + var S0 = 0, S1 = 0, S2 = 0, S3 = 0, + I0 = 0, I1 = 0, I2 = 0, I3 = 0, + N0 = 0, N1 = 0, N2 = 0, N3 = 0, + M0 = 0, M1 = 0, M2 = 0, M3 = 0, + H0 = 0, H1 = 0, H2 = 0, H3 = 0, + R = 0; + + var HEAP = new stdlib.Uint32Array(buffer), + DATA = new stdlib.Uint8Array(buffer); + + /** + * AES core + * @param {number} k - precomputed key schedule offset + * @param {number} s - precomputed sbox table offset + * @param {number} t - precomputed round table offset + * @param {number} r - number of inner rounds to perform + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _core(k, s, t, r, x0, x1, x2, x3) { + k = k | 0; + s = s | 0; + t = t | 0; + r = r | 0; + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + var t1 = 0, t2 = 0, t3 = 0, + y0 = 0, y1 = 0, y2 = 0, y3 = 0, + i = 0; + + t1 = t | 0x400, t2 = t | 0x800, t3 = t | 0xc00; + + // round 0 + x0 = x0 ^ HEAP[(k | 0) >> 2], + x1 = x1 ^ HEAP[(k | 4) >> 2], + x2 = x2 ^ HEAP[(k | 8) >> 2], + x3 = x3 ^ HEAP[(k | 12) >> 2]; + + // round 1..r + for (i = 16; (i | 0) <= (r << 4); i = (i + 16) | 0) { + y0 = HEAP[(t | x0 >> 22 & 1020) >> 2] ^ HEAP[(t1 | x1 >> 14 & 1020) >> 2] ^ HEAP[(t2 | x2 >> 6 & 1020) >> 2] ^ HEAP[(t3 | x3 << 2 & 1020) >> 2] ^ HEAP[(k | i | 0) >> 2], + y1 = HEAP[(t | x1 >> 22 & 1020) >> 2] ^ HEAP[(t1 | x2 >> 14 & 1020) >> 2] ^ HEAP[(t2 | x3 >> 6 & 1020) >> 2] ^ HEAP[(t3 | x0 << 2 & 1020) >> 2] ^ HEAP[(k | i | 4) >> 2], + y2 = HEAP[(t | x2 >> 22 & 1020) >> 2] ^ HEAP[(t1 | x3 >> 14 & 1020) >> 2] ^ HEAP[(t2 | x0 >> 6 & 1020) >> 2] ^ HEAP[(t3 | x1 << 2 & 1020) >> 2] ^ HEAP[(k | i | 8) >> 2], + y3 = HEAP[(t | x3 >> 22 & 1020) >> 2] ^ HEAP[(t1 | x0 >> 14 & 1020) >> 2] ^ HEAP[(t2 | x1 >> 6 & 1020) >> 2] ^ HEAP[(t3 | x2 << 2 & 1020) >> 2] ^ HEAP[(k | i | 12) >> 2]; + x0 = y0, x1 = y1, x2 = y2, x3 = y3; + } + + // final round + S0 = HEAP[(s | x0 >> 22 & 1020) >> 2] << 24 ^ HEAP[(s | x1 >> 14 & 1020) >> 2] << 16 ^ HEAP[(s | x2 >> 6 & 1020) >> 2] << 8 ^ HEAP[(s | x3 << 2 & 1020) >> 2] ^ HEAP[(k | i | 0) >> 2], + S1 = HEAP[(s | x1 >> 22 & 1020) >> 2] << 24 ^ HEAP[(s | x2 >> 14 & 1020) >> 2] << 16 ^ HEAP[(s | x3 >> 6 & 1020) >> 2] << 8 ^ HEAP[(s | x0 << 2 & 1020) >> 2] ^ HEAP[(k | i | 4) >> 2], + S2 = HEAP[(s | x2 >> 22 & 1020) >> 2] << 24 ^ HEAP[(s | x3 >> 14 & 1020) >> 2] << 16 ^ HEAP[(s | x0 >> 6 & 1020) >> 2] << 8 ^ HEAP[(s | x1 << 2 & 1020) >> 2] ^ HEAP[(k | i | 8) >> 2], + S3 = HEAP[(s | x3 >> 22 & 1020) >> 2] << 24 ^ HEAP[(s | x0 >> 14 & 1020) >> 2] << 16 ^ HEAP[(s | x1 >> 6 & 1020) >> 2] << 8 ^ HEAP[(s | x2 << 2 & 1020) >> 2] ^ HEAP[(k | i | 12) >> 2]; + } + + /** + * ECB mode encryption + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _ecb_enc(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + _core( + 0x0000, 0x0800, 0x1000, + R, + x0, + x1, + x2, + x3 + ); + } + + /** + * ECB mode decryption + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _ecb_dec(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + var t = 0; + + _core( + 0x0400, 0x0c00, 0x2000, + R, + x0, + x3, + x2, + x1 + ); + + t = S1, S1 = S3, S3 = t; + } + + + /** + * CBC mode encryption + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _cbc_enc(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + _core( + 0x0000, 0x0800, 0x1000, + R, + I0 ^ x0, + I1 ^ x1, + I2 ^ x2, + I3 ^ x3 + ); + + I0 = S0, + I1 = S1, + I2 = S2, + I3 = S3; + } + + /** + * CBC mode decryption + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _cbc_dec(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + var t = 0; + + _core( + 0x0400, 0x0c00, 0x2000, + R, + x0, + x3, + x2, + x1 + ); + + t = S1, S1 = S3, S3 = t; + + S0 = S0 ^ I0, + S1 = S1 ^ I1, + S2 = S2 ^ I2, + S3 = S3 ^ I3; + + I0 = x0, + I1 = x1, + I2 = x2, + I3 = x3; + } + + /** + * CFB mode encryption + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _cfb_enc(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + _core( + 0x0000, 0x0800, 0x1000, + R, + I0, + I1, + I2, + I3 + ); + + I0 = S0 = S0 ^ x0, + I1 = S1 = S1 ^ x1, + I2 = S2 = S2 ^ x2, + I3 = S3 = S3 ^ x3; + } + + + /** + * CFB mode decryption + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _cfb_dec(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + _core( + 0x0000, 0x0800, 0x1000, + R, + I0, + I1, + I2, + I3 + ); + + S0 = S0 ^ x0, + S1 = S1 ^ x1, + S2 = S2 ^ x2, + S3 = S3 ^ x3; + + I0 = x0, + I1 = x1, + I2 = x2, + I3 = x3; + } + + /** + * OFB mode encryption / decryption + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _ofb(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + _core( + 0x0000, 0x0800, 0x1000, + R, + I0, + I1, + I2, + I3 + ); + + I0 = S0, + I1 = S1, + I2 = S2, + I3 = S3; + + S0 = S0 ^ x0, + S1 = S1 ^ x1, + S2 = S2 ^ x2, + S3 = S3 ^ x3; + } + + /** + * CTR mode encryption / decryption + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _ctr(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + _core( + 0x0000, 0x0800, 0x1000, + R, + N0, + N1, + N2, + N3 + ); + + N3 = (~M3 & N3) | M3 & (N3 + 1); + N2 = (~M2 & N2) | M2 & (N2 + ((N3 | 0) == 0)); + N1 = (~M1 & N1) | M1 & (N1 + ((N2 | 0) == 0)); + N0 = (~M0 & N0) | M0 & (N0 + ((N1 | 0) == 0)); + + S0 = S0 ^ x0; + S1 = S1 ^ x1; + S2 = S2 ^ x2; + S3 = S3 ^ x3; + } + + /** + * GCM mode MAC calculation + * @param {number} x0 - 128-bit input block vector + * @param {number} x1 - 128-bit input block vector + * @param {number} x2 - 128-bit input block vector + * @param {number} x3 - 128-bit input block vector + */ + function _gcm_mac(x0, x1, x2, x3) { + x0 = x0 | 0; + x1 = x1 | 0; + x2 = x2 | 0; + x3 = x3 | 0; + + var y0 = 0, y1 = 0, y2 = 0, y3 = 0, + z0 = 0, z1 = 0, z2 = 0, z3 = 0, + i = 0, c = 0; + + x0 = x0 ^ I0, + x1 = x1 ^ I1, + x2 = x2 ^ I2, + x3 = x3 ^ I3; + + y0 = H0 | 0, + y1 = H1 | 0, + y2 = H2 | 0, + y3 = H3 | 0; + + for (; (i | 0) < 128; i = (i + 1) | 0) { + if (y0 >>> 31) { + z0 = z0 ^ x0, + z1 = z1 ^ x1, + z2 = z2 ^ x2, + z3 = z3 ^ x3; + } + + y0 = (y0 << 1) | (y1 >>> 31), + y1 = (y1 << 1) | (y2 >>> 31), + y2 = (y2 << 1) | (y3 >>> 31), + y3 = (y3 << 1); + + c = x3 & 1; + + x3 = (x3 >>> 1) | (x2 << 31), + x2 = (x2 >>> 1) | (x1 << 31), + x1 = (x1 >>> 1) | (x0 << 31), + x0 = (x0 >>> 1); + + if (c) x0 = x0 ^ 0xe1000000; + } + + I0 = z0, + I1 = z1, + I2 = z2, + I3 = z3; + } + + /** + * Set the internal rounds number. + * @instance + * @memberof AES_asm + * @param {number} r - number if inner AES rounds + */ + function set_rounds(r) { + r = r | 0; + R = r; + } + + /** + * Populate the internal state of the module. + * @instance + * @memberof AES_asm + * @param {number} s0 - state vector + * @param {number} s1 - state vector + * @param {number} s2 - state vector + * @param {number} s3 - state vector + */ + function set_state(s0, s1, s2, s3) { + s0 = s0 | 0; + s1 = s1 | 0; + s2 = s2 | 0; + s3 = s3 | 0; + + S0 = s0, + S1 = s1, + S2 = s2, + S3 = s3; + } + + /** + * Populate the internal iv of the module. + * @instance + * @memberof AES_asm + * @param {number} i0 - iv vector + * @param {number} i1 - iv vector + * @param {number} i2 - iv vector + * @param {number} i3 - iv vector + */ + function set_iv(i0, i1, i2, i3) { + i0 = i0 | 0; + i1 = i1 | 0; + i2 = i2 | 0; + i3 = i3 | 0; + + I0 = i0, + I1 = i1, + I2 = i2, + I3 = i3; + } + + /** + * Set nonce for CTR-family modes. + * @instance + * @memberof AES_asm + * @param {number} n0 - nonce vector + * @param {number} n1 - nonce vector + * @param {number} n2 - nonce vector + * @param {number} n3 - nonce vector + */ + function set_nonce(n0, n1, n2, n3) { + n0 = n0 | 0; + n1 = n1 | 0; + n2 = n2 | 0; + n3 = n3 | 0; + + N0 = n0, + N1 = n1, + N2 = n2, + N3 = n3; + } + + /** + * Set counter mask for CTR-family modes. + * @instance + * @memberof AES_asm + * @param {number} m0 - counter mask vector + * @param {number} m1 - counter mask vector + * @param {number} m2 - counter mask vector + * @param {number} m3 - counter mask vector + */ + function set_mask(m0, m1, m2, m3) { + m0 = m0 | 0; + m1 = m1 | 0; + m2 = m2 | 0; + m3 = m3 | 0; + + M0 = m0, + M1 = m1, + M2 = m2, + M3 = m3; + } + + /** + * Set counter for CTR-family modes. + * @instance + * @memberof AES_asm + * @param {number} c0 - counter vector + * @param {number} c1 - counter vector + * @param {number} c2 - counter vector + * @param {number} c3 - counter vector + */ + function set_counter(c0, c1, c2, c3) { + c0 = c0 | 0; + c1 = c1 | 0; + c2 = c2 | 0; + c3 = c3 | 0; + + N3 = (~M3 & N3) | M3 & c3, + N2 = (~M2 & N2) | M2 & c2, + N1 = (~M1 & N1) | M1 & c1, + N0 = (~M0 & N0) | M0 & c0; + } + + /** + * Store the internal state vector into the heap. + * @instance + * @memberof AES_asm + * @param {number} pos - offset where to put the data + * @return {number} The number of bytes have been written into the heap, always 16. + */ + function get_state(pos) { + pos = pos | 0; + + if (pos & 15) return -1; + + DATA[pos | 0] = S0 >>> 24, + DATA[pos | 1] = S0 >>> 16 & 255, + DATA[pos | 2] = S0 >>> 8 & 255, + DATA[pos | 3] = S0 & 255, + DATA[pos | 4] = S1 >>> 24, + DATA[pos | 5] = S1 >>> 16 & 255, + DATA[pos | 6] = S1 >>> 8 & 255, + DATA[pos | 7] = S1 & 255, + DATA[pos | 8] = S2 >>> 24, + DATA[pos | 9] = S2 >>> 16 & 255, + DATA[pos | 10] = S2 >>> 8 & 255, + DATA[pos | 11] = S2 & 255, + DATA[pos | 12] = S3 >>> 24, + DATA[pos | 13] = S3 >>> 16 & 255, + DATA[pos | 14] = S3 >>> 8 & 255, + DATA[pos | 15] = S3 & 255; + + return 16; + } + + /** + * Store the internal iv vector into the heap. + * @instance + * @memberof AES_asm + * @param {number} pos - offset where to put the data + * @return {number} The number of bytes have been written into the heap, always 16. + */ + function get_iv(pos) { + pos = pos | 0; + + if (pos & 15) return -1; + + DATA[pos | 0] = I0 >>> 24, + DATA[pos | 1] = I0 >>> 16 & 255, + DATA[pos | 2] = I0 >>> 8 & 255, + DATA[pos | 3] = I0 & 255, + DATA[pos | 4] = I1 >>> 24, + DATA[pos | 5] = I1 >>> 16 & 255, + DATA[pos | 6] = I1 >>> 8 & 255, + DATA[pos | 7] = I1 & 255, + DATA[pos | 8] = I2 >>> 24, + DATA[pos | 9] = I2 >>> 16 & 255, + DATA[pos | 10] = I2 >>> 8 & 255, + DATA[pos | 11] = I2 & 255, + DATA[pos | 12] = I3 >>> 24, + DATA[pos | 13] = I3 >>> 16 & 255, + DATA[pos | 14] = I3 >>> 8 & 255, + DATA[pos | 15] = I3 & 255; + + return 16; + } + + /** + * GCM initialization. + * @instance + * @memberof AES_asm + */ + function gcm_init() { + _ecb_enc(0, 0, 0, 0); + H0 = S0, + H1 = S1, + H2 = S2, + H3 = S3; + } + + /** + * Perform ciphering operation on the supplied data. + * @instance + * @memberof AES_asm + * @param {number} mode - block cipher mode (see {@link AES_asm} mode constants) + * @param {number} pos - offset of the data being processed + * @param {number} len - length of the data being processed + * @return {number} Actual amount of data have been processed. + */ + function cipher(mode, pos, len) { + mode = mode | 0; + pos = pos | 0; + len = len | 0; + + var ret = 0; + + if (pos & 15) return -1; + + while ((len | 0) >= 16) { + _cipher_modes[mode & 7]( + DATA[pos | 0] << 24 | DATA[pos | 1] << 16 | DATA[pos | 2] << 8 | DATA[pos | 3], + DATA[pos | 4] << 24 | DATA[pos | 5] << 16 | DATA[pos | 6] << 8 | DATA[pos | 7], + DATA[pos | 8] << 24 | DATA[pos | 9] << 16 | DATA[pos | 10] << 8 | DATA[pos | 11], + DATA[pos | 12] << 24 | DATA[pos | 13] << 16 | DATA[pos | 14] << 8 | DATA[pos | 15] + ); + + DATA[pos | 0] = S0 >>> 24, + DATA[pos | 1] = S0 >>> 16 & 255, + DATA[pos | 2] = S0 >>> 8 & 255, + DATA[pos | 3] = S0 & 255, + DATA[pos | 4] = S1 >>> 24, + DATA[pos | 5] = S1 >>> 16 & 255, + DATA[pos | 6] = S1 >>> 8 & 255, + DATA[pos | 7] = S1 & 255, + DATA[pos | 8] = S2 >>> 24, + DATA[pos | 9] = S2 >>> 16 & 255, + DATA[pos | 10] = S2 >>> 8 & 255, + DATA[pos | 11] = S2 & 255, + DATA[pos | 12] = S3 >>> 24, + DATA[pos | 13] = S3 >>> 16 & 255, + DATA[pos | 14] = S3 >>> 8 & 255, + DATA[pos | 15] = S3 & 255; + + ret = (ret + 16) | 0, + pos = (pos + 16) | 0, + len = (len - 16) | 0; + } + + return ret | 0; + } + + /** + * Calculates MAC of the supplied data. + * @instance + * @memberof AES_asm + * @param {number} mode - block cipher mode (see {@link AES_asm} mode constants) + * @param {number} pos - offset of the data being processed + * @param {number} len - length of the data being processed + * @return {number} Actual amount of data have been processed. + */ + function mac(mode, pos, len) { + mode = mode | 0; + pos = pos | 0; + len = len | 0; + + var ret = 0; + + if (pos & 15) return -1; + + while ((len | 0) >= 16) { + _mac_modes[mode & 1]( + DATA[pos | 0] << 24 | DATA[pos | 1] << 16 | DATA[pos | 2] << 8 | DATA[pos | 3], + DATA[pos | 4] << 24 | DATA[pos | 5] << 16 | DATA[pos | 6] << 8 | DATA[pos | 7], + DATA[pos | 8] << 24 | DATA[pos | 9] << 16 | DATA[pos | 10] << 8 | DATA[pos | 11], + DATA[pos | 12] << 24 | DATA[pos | 13] << 16 | DATA[pos | 14] << 8 | DATA[pos | 15] + ); + + ret = (ret + 16) | 0, + pos = (pos + 16) | 0, + len = (len - 16) | 0; + } + + return ret | 0; + } + + /** + * AES cipher modes table (virual methods) + */ + var _cipher_modes = [_ecb_enc, _ecb_dec, _cbc_enc, _cbc_dec, _cfb_enc, _cfb_dec, _ofb, _ctr]; + + /** + * AES MAC modes table (virual methods) + */ + var _mac_modes = [_cbc_enc, _gcm_mac]; + + /** + * Asm.js module exports + */ + return { + set_rounds: set_rounds, + set_state: set_state, + set_iv: set_iv, + set_nonce: set_nonce, + set_mask: set_mask, + set_counter: set_counter, + get_state: get_state, + get_iv: get_iv, + gcm_init: gcm_init, + cipher: cipher, + mac: mac, + }; + }(stdlib, foreign, buffer); + + asm.set_key = set_key; + + return asm; + }; + + /** + * AES enciphering mode constants + * @enum {number} + * @const + */ + wrapper.ENC = { + ECB: 0, + CBC: 2, + CFB: 4, + OFB: 6, + CTR: 7, + }, + + /** + * AES deciphering mode constants + * @enum {number} + * @const + */ + wrapper.DEC = { + ECB: 1, + CBC: 3, + CFB: 5, + OFB: 6, + CTR: 7, + }, + + /** + * AES MAC mode constants + * @enum {number} + * @const + */ + wrapper.MAC = { + CBC: 0, + GCM: 1, + }; + + /** + * Heap data offset + * @type {number} + * @const + */ + wrapper.HEAP_DATA = 0x4000; + + return wrapper; + }(); + + function is_bytes(a) { + return a instanceof Uint8Array; + } + function _heap_init(heap, heapSize) { + const size = heap ? heap.byteLength : heapSize || 65536; + if (size & 0xfff || size <= 0) + throw new Error('heap size must be a positive integer and a multiple of 4096'); + heap = heap || new Uint8Array(new ArrayBuffer(size)); + return heap; + } + function _heap_write(heap, hpos, data, dpos, dlen) { + const hlen = heap.length - hpos; + const wlen = hlen < dlen ? hlen : dlen; + heap.set(data.subarray(dpos, dpos + wlen), hpos); + return wlen; + } + function joinBytes(...arg) { + const totalLenght = arg.reduce((sum, curr) => sum + curr.length, 0); + const ret = new Uint8Array(totalLenght); + let cursor = 0; + for (let i = 0; i < arg.length; i++) { + ret.set(arg[i], cursor); + cursor += arg[i].length; + } + return ret; + } + + class IllegalStateError extends Error { + constructor(...args) { + super(...args); + } + } + class IllegalArgumentError extends Error { + constructor(...args) { + super(...args); + } + } + class SecurityError extends Error { + constructor(...args) { + super(...args); + } + } + + const heap_pool = []; + const asm_pool = []; + class AES { + constructor(key, iv, padding = true, mode, heap, asm) { + this.pos = 0; + this.len = 0; + this.mode = mode; + // The AES object state + this.pos = 0; + this.len = 0; + this.key = key; + this.iv = iv; + this.padding = padding; + // The AES "worker" + this.acquire_asm(heap, asm); + } + acquire_asm(heap, asm) { + if (this.heap === undefined || this.asm === undefined) { + this.heap = heap || heap_pool.pop() || _heap_init().subarray(AES_asm.HEAP_DATA); + this.asm = asm || asm_pool.pop() || new AES_asm(null, this.heap.buffer); + this.reset(this.key, this.iv); + } + return { heap: this.heap, asm: this.asm }; + } + release_asm() { + if (this.heap !== undefined && this.asm !== undefined) { + heap_pool.push(this.heap); + asm_pool.push(this.asm); + } + this.heap = undefined; + this.asm = undefined; + } + reset(key, iv) { + const { asm } = this.acquire_asm(); + // Key + const keylen = key.length; + if (keylen !== 16 && keylen !== 24 && keylen !== 32) + throw new IllegalArgumentError('illegal key size'); + const keyview = new DataView(key.buffer, key.byteOffset, key.byteLength); + asm.set_key(keylen >> 2, keyview.getUint32(0), keyview.getUint32(4), keyview.getUint32(8), keyview.getUint32(12), keylen > 16 ? keyview.getUint32(16) : 0, keylen > 16 ? keyview.getUint32(20) : 0, keylen > 24 ? keyview.getUint32(24) : 0, keylen > 24 ? keyview.getUint32(28) : 0); + // IV + if (iv !== undefined) { + if (iv.length !== 16) + throw new IllegalArgumentError('illegal iv size'); + let ivview = new DataView(iv.buffer, iv.byteOffset, iv.byteLength); + asm.set_iv(ivview.getUint32(0), ivview.getUint32(4), ivview.getUint32(8), ivview.getUint32(12)); + } + else { + asm.set_iv(0, 0, 0, 0); + } + } + AES_Encrypt_process(data) { + if (!is_bytes(data)) + throw new TypeError("data isn't of expected type"); + let { heap, asm } = this.acquire_asm(); + let amode = AES_asm.ENC[this.mode]; + let hpos = AES_asm.HEAP_DATA; + let pos = this.pos; + let len = this.len; + let dpos = 0; + let dlen = data.length || 0; + let rpos = 0; + let rlen = (len + dlen) & -16; + let wlen = 0; + let result = new Uint8Array(rlen); + while (dlen > 0) { + wlen = _heap_write(heap, pos + len, data, dpos, dlen); + len += wlen; + dpos += wlen; + dlen -= wlen; + wlen = asm.cipher(amode, hpos + pos, len); + if (wlen) + result.set(heap.subarray(pos, pos + wlen), rpos); + rpos += wlen; + if (wlen < len) { + pos += wlen; + len -= wlen; + } + else { + pos = 0; + len = 0; + } + } + this.pos = pos; + this.len = len; + return result; + } + AES_Encrypt_finish() { + let { heap, asm } = this.acquire_asm(); + let amode = AES_asm.ENC[this.mode]; + let hpos = AES_asm.HEAP_DATA; + let pos = this.pos; + let len = this.len; + let plen = 16 - (len % 16); + let rlen = len; + if (this.hasOwnProperty('padding')) { + if (this.padding) { + for (let p = 0; p < plen; ++p) { + heap[pos + len + p] = plen; + } + len += plen; + rlen = len; + } + else if (len % 16) { + throw new IllegalArgumentError('data length must be a multiple of the block size'); + } + } + else { + len += plen; + } + const result = new Uint8Array(rlen); + if (len) + asm.cipher(amode, hpos + pos, len); + if (rlen) + result.set(heap.subarray(pos, pos + rlen)); + this.pos = 0; + this.len = 0; + this.release_asm(); + return result; + } + AES_Decrypt_process(data) { + if (!is_bytes(data)) + throw new TypeError("data isn't of expected type"); + let { heap, asm } = this.acquire_asm(); + let amode = AES_asm.DEC[this.mode]; + let hpos = AES_asm.HEAP_DATA; + let pos = this.pos; + let len = this.len; + let dpos = 0; + let dlen = data.length || 0; + let rpos = 0; + let rlen = (len + dlen) & -16; + let plen = 0; + let wlen = 0; + if (this.padding) { + plen = len + dlen - rlen || 16; + rlen -= plen; + } + const result = new Uint8Array(rlen); + while (dlen > 0) { + wlen = _heap_write(heap, pos + len, data, dpos, dlen); + len += wlen; + dpos += wlen; + dlen -= wlen; + wlen = asm.cipher(amode, hpos + pos, len - (!dlen ? plen : 0)); + if (wlen) + result.set(heap.subarray(pos, pos + wlen), rpos); + rpos += wlen; + if (wlen < len) { + pos += wlen; + len -= wlen; + } + else { + pos = 0; + len = 0; + } + } + this.pos = pos; + this.len = len; + return result; + } + AES_Decrypt_finish() { + let { heap, asm } = this.acquire_asm(); + let amode = AES_asm.DEC[this.mode]; + let hpos = AES_asm.HEAP_DATA; + let pos = this.pos; + let len = this.len; + let rlen = len; + if (len > 0) { + if (len % 16) { + if (this.hasOwnProperty('padding')) { + throw new IllegalArgumentError('data length must be a multiple of the block size'); + } + else { + len += 16 - (len % 16); + } + } + asm.cipher(amode, hpos + pos, len); + if (this.hasOwnProperty('padding') && this.padding) { + let pad = heap[pos + rlen - 1]; + if (pad < 1 || pad > 16 || pad > rlen) + throw new SecurityError('bad padding'); + let pcheck = 0; + for (let i = pad; i > 1; i--) + pcheck |= pad ^ heap[pos + rlen - i]; + if (pcheck) + throw new SecurityError('bad padding'); + rlen -= pad; + } + } + const result = new Uint8Array(rlen); + if (rlen > 0) { + result.set(heap.subarray(pos, pos + rlen)); + } + this.pos = 0; + this.len = 0; + this.release_asm(); + return result; + } + } + + class AES_ECB { + static encrypt(data, key, padding = false) { + return new AES_ECB(key, padding).encrypt(data); + } + static decrypt(data, key, padding = false) { + return new AES_ECB(key, padding).decrypt(data); + } + constructor(key, padding = false, aes) { + this.aes = aes ? aes : new AES(key, undefined, padding, 'ECB'); + } + encrypt(data) { + const r1 = this.aes.AES_Encrypt_process(data); + const r2 = this.aes.AES_Encrypt_finish(); + return joinBytes(r1, r2); + } + decrypt(data) { + const r1 = this.aes.AES_Decrypt_process(data); + const r2 = this.aes.AES_Decrypt_finish(); + return joinBytes(r1, r2); + } + } + + /** + * Javascript AES implementation. + * This is used as fallback if the native Crypto APIs are not available. + */ + function aes(length) { + const C = function(key) { + const aesECB = new AES_ECB(key); + + this.encrypt = function(block) { + return aesECB.encrypt(block); + }; + + this.decrypt = function(block) { + return aesECB.decrypt(block); + }; + }; + + C.blockSize = C.prototype.blockSize = 16; + C.keySize = C.prototype.keySize = length / 8; + + return C; + } + + //Paul Tero, July 2001 + //http://www.tero.co.uk/des/ + // + //Optimised for performance with large blocks by Michael Hayworth, November 2001 + //http://www.netdealing.com + // + // Modified by Recurity Labs GmbH + + //THIS SOFTWARE IS PROVIDED "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 AUTHOR 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. + + //des + //this takes the key, the message, and whether to encrypt or decrypt + + function des(keys, message, encrypt, mode, iv, padding) { + //declaring this locally speeds things up a bit + const spfunction1 = [ + 0x1010400, 0, 0x10000, 0x1010404, 0x1010004, 0x10404, 0x4, 0x10000, 0x400, 0x1010400, + 0x1010404, 0x400, 0x1000404, 0x1010004, 0x1000000, 0x4, 0x404, 0x1000400, 0x1000400, 0x10400, 0x10400, 0x1010000, + 0x1010000, 0x1000404, 0x10004, 0x1000004, 0x1000004, 0x10004, 0, 0x404, 0x10404, 0x1000000, 0x10000, 0x1010404, 0x4, + 0x1010000, 0x1010400, 0x1000000, 0x1000000, 0x400, 0x1010004, 0x10000, 0x10400, 0x1000004, 0x400, 0x4, 0x1000404, + 0x10404, 0x1010404, 0x10004, 0x1010000, 0x1000404, 0x1000004, 0x404, 0x10404, 0x1010400, 0x404, 0x1000400, + 0x1000400, 0, 0x10004, 0x10400, 0, 0x1010004 + ]; + const spfunction2 = [ + -0x7fef7fe0, -0x7fff8000, 0x8000, 0x108020, 0x100000, 0x20, -0x7fefffe0, -0x7fff7fe0, + -0x7fffffe0, -0x7fef7fe0, -0x7fef8000, -0x80000000, -0x7fff8000, 0x100000, 0x20, -0x7fefffe0, 0x108000, 0x100020, + -0x7fff7fe0, 0, -0x80000000, 0x8000, 0x108020, -0x7ff00000, 0x100020, -0x7fffffe0, 0, 0x108000, 0x8020, -0x7fef8000, + -0x7ff00000, 0x8020, 0, 0x108020, -0x7fefffe0, 0x100000, -0x7fff7fe0, -0x7ff00000, -0x7fef8000, 0x8000, -0x7ff00000, + -0x7fff8000, 0x20, -0x7fef7fe0, 0x108020, 0x20, 0x8000, -0x80000000, 0x8020, -0x7fef8000, 0x100000, -0x7fffffe0, + 0x100020, -0x7fff7fe0, -0x7fffffe0, 0x100020, 0x108000, 0, -0x7fff8000, 0x8020, -0x80000000, -0x7fefffe0, + -0x7fef7fe0, 0x108000 + ]; + const spfunction3 = [ + 0x208, 0x8020200, 0, 0x8020008, 0x8000200, 0, 0x20208, 0x8000200, 0x20008, 0x8000008, + 0x8000008, 0x20000, 0x8020208, 0x20008, 0x8020000, 0x208, 0x8000000, 0x8, 0x8020200, 0x200, 0x20200, 0x8020000, + 0x8020008, 0x20208, 0x8000208, 0x20200, 0x20000, 0x8000208, 0x8, 0x8020208, 0x200, 0x8000000, 0x8020200, 0x8000000, + 0x20008, 0x208, 0x20000, 0x8020200, 0x8000200, 0, 0x200, 0x20008, 0x8020208, 0x8000200, 0x8000008, 0x200, 0, + 0x8020008, 0x8000208, 0x20000, 0x8000000, 0x8020208, 0x8, 0x20208, 0x20200, 0x8000008, 0x8020000, 0x8000208, 0x208, + 0x8020000, 0x20208, 0x8, 0x8020008, 0x20200 + ]; + const spfunction4 = [ + 0x802001, 0x2081, 0x2081, 0x80, 0x802080, 0x800081, 0x800001, 0x2001, 0, 0x802000, + 0x802000, 0x802081, 0x81, 0, 0x800080, 0x800001, 0x1, 0x2000, 0x800000, 0x802001, 0x80, 0x800000, 0x2001, 0x2080, + 0x800081, 0x1, 0x2080, 0x800080, 0x2000, 0x802080, 0x802081, 0x81, 0x800080, 0x800001, 0x802000, 0x802081, 0x81, 0, + 0, 0x802000, 0x2080, 0x800080, 0x800081, 0x1, 0x802001, 0x2081, 0x2081, 0x80, 0x802081, 0x81, 0x1, 0x2000, 0x800001, + 0x2001, 0x802080, 0x800081, 0x2001, 0x2080, 0x800000, 0x802001, 0x80, 0x800000, 0x2000, 0x802080 + ]; + const spfunction5 = [ + 0x100, 0x2080100, 0x2080000, 0x42000100, 0x80000, 0x100, 0x40000000, 0x2080000, + 0x40080100, 0x80000, 0x2000100, 0x40080100, 0x42000100, 0x42080000, 0x80100, 0x40000000, 0x2000000, 0x40080000, + 0x40080000, 0, 0x40000100, 0x42080100, 0x42080100, 0x2000100, 0x42080000, 0x40000100, 0, 0x42000000, 0x2080100, + 0x2000000, 0x42000000, 0x80100, 0x80000, 0x42000100, 0x100, 0x2000000, 0x40000000, 0x2080000, 0x42000100, + 0x40080100, 0x2000100, 0x40000000, 0x42080000, 0x2080100, 0x40080100, 0x100, 0x2000000, 0x42080000, 0x42080100, + 0x80100, 0x42000000, 0x42080100, 0x2080000, 0, 0x40080000, 0x42000000, 0x80100, 0x2000100, 0x40000100, 0x80000, 0, + 0x40080000, 0x2080100, 0x40000100 + ]; + const spfunction6 = [ + 0x20000010, 0x20400000, 0x4000, 0x20404010, 0x20400000, 0x10, 0x20404010, 0x400000, + 0x20004000, 0x404010, 0x400000, 0x20000010, 0x400010, 0x20004000, 0x20000000, 0x4010, 0, 0x400010, 0x20004010, + 0x4000, 0x404000, 0x20004010, 0x10, 0x20400010, 0x20400010, 0, 0x404010, 0x20404000, 0x4010, 0x404000, 0x20404000, + 0x20000000, 0x20004000, 0x10, 0x20400010, 0x404000, 0x20404010, 0x400000, 0x4010, 0x20000010, 0x400000, 0x20004000, + 0x20000000, 0x4010, 0x20000010, 0x20404010, 0x404000, 0x20400000, 0x404010, 0x20404000, 0, 0x20400010, 0x10, 0x4000, + 0x20400000, 0x404010, 0x4000, 0x400010, 0x20004010, 0, 0x20404000, 0x20000000, 0x400010, 0x20004010 + ]; + const spfunction7 = [ + 0x200000, 0x4200002, 0x4000802, 0, 0x800, 0x4000802, 0x200802, 0x4200800, 0x4200802, + 0x200000, 0, 0x4000002, 0x2, 0x4000000, 0x4200002, 0x802, 0x4000800, 0x200802, 0x200002, 0x4000800, 0x4000002, + 0x4200000, 0x4200800, 0x200002, 0x4200000, 0x800, 0x802, 0x4200802, 0x200800, 0x2, 0x4000000, 0x200800, 0x4000000, + 0x200800, 0x200000, 0x4000802, 0x4000802, 0x4200002, 0x4200002, 0x2, 0x200002, 0x4000000, 0x4000800, 0x200000, + 0x4200800, 0x802, 0x200802, 0x4200800, 0x802, 0x4000002, 0x4200802, 0x4200000, 0x200800, 0, 0x2, 0x4200802, 0, + 0x200802, 0x4200000, 0x800, 0x4000002, 0x4000800, 0x800, 0x200002 + ]; + const spfunction8 = [ + 0x10001040, 0x1000, 0x40000, 0x10041040, 0x10000000, 0x10001040, 0x40, 0x10000000, + 0x40040, 0x10040000, 0x10041040, 0x41000, 0x10041000, 0x41040, 0x1000, 0x40, 0x10040000, 0x10000040, 0x10001000, + 0x1040, 0x41000, 0x40040, 0x10040040, 0x10041000, 0x1040, 0, 0, 0x10040040, 0x10000040, 0x10001000, 0x41040, + 0x40000, 0x41040, 0x40000, 0x10041000, 0x1000, 0x40, 0x10040040, 0x1000, 0x41040, 0x10001000, 0x40, 0x10000040, + 0x10040000, 0x10040040, 0x10000000, 0x40000, 0x10001040, 0, 0x10041040, 0x40040, 0x10000040, 0x10040000, 0x10001000, + 0x10001040, 0, 0x10041040, 0x41000, 0x41000, 0x1040, 0x1040, 0x40040, 0x10000000, 0x10041000 + ]; + + //create the 16 or 48 subkeys we will need + let m = 0; + let i; + let j; + let temp; + let right1; + let right2; + let left; + let right; + let looping; + let cbcleft; + let cbcleft2; + let cbcright; + let cbcright2; + let endloop; + let loopinc; + let len = message.length; + + //set up the loops for single and triple des + const iterations = keys.length === 32 ? 3 : 9; //single or triple des + if (iterations === 3) { + looping = encrypt ? [0, 32, 2] : [30, -2, -2]; + } else { + looping = encrypt ? [0, 32, 2, 62, 30, -2, 64, 96, 2] : [94, 62, -2, 32, 64, 2, 30, -2, -2]; + } + + //pad the message depending on the padding parameter + //only add padding if encrypting - note that you need to use the same padding option for both encrypt and decrypt + if (encrypt) { + message = desAddPadding(message, padding); + len = message.length; + } + + //store the result here + let result = new Uint8Array(len); + let k = 0; + + if (mode === 1) { //CBC mode + cbcleft = (iv[m++] << 24) | (iv[m++] << 16) | (iv[m++] << 8) | iv[m++]; + cbcright = (iv[m++] << 24) | (iv[m++] << 16) | (iv[m++] << 8) | iv[m++]; + m = 0; + } + + //loop through each 64 bit chunk of the message + while (m < len) { + left = (message[m++] << 24) | (message[m++] << 16) | (message[m++] << 8) | message[m++]; + right = (message[m++] << 24) | (message[m++] << 16) | (message[m++] << 8) | message[m++]; + + //for Cipher Block Chaining mode, xor the message with the previous result + if (mode === 1) { + if (encrypt) { + left ^= cbcleft; + right ^= cbcright; + } else { + cbcleft2 = cbcleft; + cbcright2 = cbcright; + cbcleft = left; + cbcright = right; + } + } + + //first each 64 but chunk of the message must be permuted according to IP + temp = ((left >>> 4) ^ right) & 0x0f0f0f0f; + right ^= temp; + left ^= (temp << 4); + temp = ((left >>> 16) ^ right) & 0x0000ffff; + right ^= temp; + left ^= (temp << 16); + temp = ((right >>> 2) ^ left) & 0x33333333; + left ^= temp; + right ^= (temp << 2); + temp = ((right >>> 8) ^ left) & 0x00ff00ff; + left ^= temp; + right ^= (temp << 8); + temp = ((left >>> 1) ^ right) & 0x55555555; + right ^= temp; + left ^= (temp << 1); + + left = ((left << 1) | (left >>> 31)); + right = ((right << 1) | (right >>> 31)); + + //do this either 1 or 3 times for each chunk of the message + for (j = 0; j < iterations; j += 3) { + endloop = looping[j + 1]; + loopinc = looping[j + 2]; + //now go through and perform the encryption or decryption + for (i = looping[j]; i !== endloop; i += loopinc) { //for efficiency + right1 = right ^ keys[i]; + right2 = ((right >>> 4) | (right << 28)) ^ keys[i + 1]; + //the result is attained by passing these bytes through the S selection functions + temp = left; + left = right; + right = temp ^ (spfunction2[(right1 >>> 24) & 0x3f] | spfunction4[(right1 >>> 16) & 0x3f] | spfunction6[(right1 >>> + 8) & 0x3f] | spfunction8[right1 & 0x3f] | spfunction1[(right2 >>> 24) & 0x3f] | spfunction3[(right2 >>> 16) & + 0x3f] | spfunction5[(right2 >>> 8) & 0x3f] | spfunction7[right2 & 0x3f]); + } + temp = left; + left = right; + right = temp; //unreverse left and right + } //for either 1 or 3 iterations + + //move then each one bit to the right + left = ((left >>> 1) | (left << 31)); + right = ((right >>> 1) | (right << 31)); + + //now perform IP-1, which is IP in the opposite direction + temp = ((left >>> 1) ^ right) & 0x55555555; + right ^= temp; + left ^= (temp << 1); + temp = ((right >>> 8) ^ left) & 0x00ff00ff; + left ^= temp; + right ^= (temp << 8); + temp = ((right >>> 2) ^ left) & 0x33333333; + left ^= temp; + right ^= (temp << 2); + temp = ((left >>> 16) ^ right) & 0x0000ffff; + right ^= temp; + left ^= (temp << 16); + temp = ((left >>> 4) ^ right) & 0x0f0f0f0f; + right ^= temp; + left ^= (temp << 4); + + //for Cipher Block Chaining mode, xor the message with the previous result + if (mode === 1) { + if (encrypt) { + cbcleft = left; + cbcright = right; + } else { + left ^= cbcleft2; + right ^= cbcright2; + } + } + + result[k++] = (left >>> 24); + result[k++] = ((left >>> 16) & 0xff); + result[k++] = ((left >>> 8) & 0xff); + result[k++] = (left & 0xff); + result[k++] = (right >>> 24); + result[k++] = ((right >>> 16) & 0xff); + result[k++] = ((right >>> 8) & 0xff); + result[k++] = (right & 0xff); + } //for every 8 characters, or 64 bits in the message + + //only remove padding if decrypting - note that you need to use the same padding option for both encrypt and decrypt + if (!encrypt) { + result = desRemovePadding(result, padding); + } + + return result; + } //end of des + + + //desCreateKeys + //this takes as input a 64 bit key (even though only 56 bits are used) + //as an array of 2 integers, and returns 16 48 bit keys + + function desCreateKeys(key) { + //declaring this locally speeds things up a bit + const pc2bytes0 = [ + 0, 0x4, 0x20000000, 0x20000004, 0x10000, 0x10004, 0x20010000, 0x20010004, 0x200, 0x204, + 0x20000200, 0x20000204, 0x10200, 0x10204, 0x20010200, 0x20010204 + ]; + const pc2bytes1 = [ + 0, 0x1, 0x100000, 0x100001, 0x4000000, 0x4000001, 0x4100000, 0x4100001, 0x100, 0x101, 0x100100, + 0x100101, 0x4000100, 0x4000101, 0x4100100, 0x4100101 + ]; + const pc2bytes2 = [ + 0, 0x8, 0x800, 0x808, 0x1000000, 0x1000008, 0x1000800, 0x1000808, 0, 0x8, 0x800, 0x808, + 0x1000000, 0x1000008, 0x1000800, 0x1000808 + ]; + const pc2bytes3 = [ + 0, 0x200000, 0x8000000, 0x8200000, 0x2000, 0x202000, 0x8002000, 0x8202000, 0x20000, 0x220000, + 0x8020000, 0x8220000, 0x22000, 0x222000, 0x8022000, 0x8222000 + ]; + const pc2bytes4 = [ + 0, 0x40000, 0x10, 0x40010, 0, 0x40000, 0x10, 0x40010, 0x1000, 0x41000, 0x1010, 0x41010, 0x1000, + 0x41000, 0x1010, 0x41010 + ]; + const pc2bytes5 = [ + 0, 0x400, 0x20, 0x420, 0, 0x400, 0x20, 0x420, 0x2000000, 0x2000400, 0x2000020, 0x2000420, + 0x2000000, 0x2000400, 0x2000020, 0x2000420 + ]; + const pc2bytes6 = [ + 0, 0x10000000, 0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002, 0, 0x10000000, + 0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002 + ]; + const pc2bytes7 = [ + 0, 0x10000, 0x800, 0x10800, 0x20000000, 0x20010000, 0x20000800, 0x20010800, 0x20000, 0x30000, + 0x20800, 0x30800, 0x20020000, 0x20030000, 0x20020800, 0x20030800 + ]; + const pc2bytes8 = [ + 0, 0x40000, 0, 0x40000, 0x2, 0x40002, 0x2, 0x40002, 0x2000000, 0x2040000, 0x2000000, 0x2040000, + 0x2000002, 0x2040002, 0x2000002, 0x2040002 + ]; + const pc2bytes9 = [ + 0, 0x10000000, 0x8, 0x10000008, 0, 0x10000000, 0x8, 0x10000008, 0x400, 0x10000400, 0x408, + 0x10000408, 0x400, 0x10000400, 0x408, 0x10000408 + ]; + const pc2bytes10 = [ + 0, 0x20, 0, 0x20, 0x100000, 0x100020, 0x100000, 0x100020, 0x2000, 0x2020, 0x2000, 0x2020, + 0x102000, 0x102020, 0x102000, 0x102020 + ]; + const pc2bytes11 = [ + 0, 0x1000000, 0x200, 0x1000200, 0x200000, 0x1200000, 0x200200, 0x1200200, 0x4000000, 0x5000000, + 0x4000200, 0x5000200, 0x4200000, 0x5200000, 0x4200200, 0x5200200 + ]; + const pc2bytes12 = [ + 0, 0x1000, 0x8000000, 0x8001000, 0x80000, 0x81000, 0x8080000, 0x8081000, 0x10, 0x1010, + 0x8000010, 0x8001010, 0x80010, 0x81010, 0x8080010, 0x8081010 + ]; + const pc2bytes13 = [0, 0x4, 0x100, 0x104, 0, 0x4, 0x100, 0x104, 0x1, 0x5, 0x101, 0x105, 0x1, 0x5, 0x101, 0x105]; + + //how many iterations (1 for des, 3 for triple des) + const iterations = key.length > 8 ? 3 : 1; //changed by Paul 16/6/2007 to use Triple DES for 9+ byte keys + //stores the return keys + const keys = new Array(32 * iterations); + //now define the left shifts which need to be done + const shifts = [0, 0, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 0]; + //other variables + let lefttemp; + let righttemp; + let m = 0; + let n = 0; + let temp; + + for (let j = 0; j < iterations; j++) { //either 1 or 3 iterations + let left = (key[m++] << 24) | (key[m++] << 16) | (key[m++] << 8) | key[m++]; + let right = (key[m++] << 24) | (key[m++] << 16) | (key[m++] << 8) | key[m++]; + + temp = ((left >>> 4) ^ right) & 0x0f0f0f0f; + right ^= temp; + left ^= (temp << 4); + temp = ((right >>> -16) ^ left) & 0x0000ffff; + left ^= temp; + right ^= (temp << -16); + temp = ((left >>> 2) ^ right) & 0x33333333; + right ^= temp; + left ^= (temp << 2); + temp = ((right >>> -16) ^ left) & 0x0000ffff; + left ^= temp; + right ^= (temp << -16); + temp = ((left >>> 1) ^ right) & 0x55555555; + right ^= temp; + left ^= (temp << 1); + temp = ((right >>> 8) ^ left) & 0x00ff00ff; + left ^= temp; + right ^= (temp << 8); + temp = ((left >>> 1) ^ right) & 0x55555555; + right ^= temp; + left ^= (temp << 1); + + //the right side needs to be shifted and to get the last four bits of the left side + temp = (left << 8) | ((right >>> 20) & 0x000000f0); + //left needs to be put upside down + left = (right << 24) | ((right << 8) & 0xff0000) | ((right >>> 8) & 0xff00) | ((right >>> 24) & 0xf0); + right = temp; + + //now go through and perform these shifts on the left and right keys + for (let i = 0; i < shifts.length; i++) { + //shift the keys either one or two bits to the left + if (shifts[i]) { + left = (left << 2) | (left >>> 26); + right = (right << 2) | (right >>> 26); + } else { + left = (left << 1) | (left >>> 27); + right = (right << 1) | (right >>> 27); + } + left &= -0xf; + right &= -0xf; + + //now apply PC-2, in such a way that E is easier when encrypting or decrypting + //this conversion will look like PC-2 except only the last 6 bits of each byte are used + //rather than 48 consecutive bits and the order of lines will be according to + //how the S selection functions will be applied: S2, S4, S6, S8, S1, S3, S5, S7 + lefttemp = pc2bytes0[left >>> 28] | pc2bytes1[(left >>> 24) & 0xf] | pc2bytes2[(left >>> 20) & 0xf] | pc2bytes3[( + left >>> 16) & 0xf] | pc2bytes4[(left >>> 12) & 0xf] | pc2bytes5[(left >>> 8) & 0xf] | pc2bytes6[(left >>> 4) & + 0xf]; + righttemp = pc2bytes7[right >>> 28] | pc2bytes8[(right >>> 24) & 0xf] | pc2bytes9[(right >>> 20) & 0xf] | + pc2bytes10[(right >>> 16) & 0xf] | pc2bytes11[(right >>> 12) & 0xf] | pc2bytes12[(right >>> 8) & 0xf] | + pc2bytes13[(right >>> 4) & 0xf]; + temp = ((righttemp >>> 16) ^ lefttemp) & 0x0000ffff; + keys[n++] = lefttemp ^ temp; + keys[n++] = righttemp ^ (temp << 16); + } + } //for each iterations + //return the keys we've created + return keys; + } //end of desCreateKeys + + + function desAddPadding(message, padding) { + const padLength = 8 - (message.length % 8); + + let pad; + if (padding === 2 && (padLength < 8)) { //pad the message with spaces + pad = ' '.charCodeAt(0); + } else if (padding === 1) { //PKCS7 padding + pad = padLength; + } else if (!padding && (padLength < 8)) { //pad the message out with null bytes + pad = 0; + } else if (padLength === 8) { + return message; + } else { + throw new Error('des: invalid padding'); + } + + const paddedMessage = new Uint8Array(message.length + padLength); + for (let i = 0; i < message.length; i++) { + paddedMessage[i] = message[i]; + } + for (let j = 0; j < padLength; j++) { + paddedMessage[message.length + j] = pad; + } + + return paddedMessage; + } + + function desRemovePadding(message, padding) { + let padLength = null; + let pad; + if (padding === 2) { // space padded + pad = ' '.charCodeAt(0); + } else if (padding === 1) { // PKCS7 + padLength = message[message.length - 1]; + } else if (!padding) { // null padding + pad = 0; + } else { + throw new Error('des: invalid padding'); + } + + if (!padLength) { + padLength = 1; + while (message[message.length - padLength] === pad) { + padLength++; + } + padLength--; + } + + return message.subarray(0, message.length - padLength); + } + + // added by Recurity Labs + + function TripleDES(key) { + this.key = []; + + for (let i = 0; i < 3; i++) { + this.key.push(new Uint8Array(key.subarray(i * 8, (i * 8) + 8))); + } + + this.encrypt = function(block) { + return des( + desCreateKeys(this.key[2]), + des( + desCreateKeys(this.key[1]), + des( + desCreateKeys(this.key[0]), + block, true, 0, null, null + ), + false, 0, null, null + ), true, 0, null, null + ); + }; + } + + TripleDES.keySize = TripleDES.prototype.keySize = 24; + TripleDES.blockSize = TripleDES.prototype.blockSize = 8; + + // This is "original" DES + + function DES(key) { + this.key = key; + + this.encrypt = function(block, padding) { + const keys = desCreateKeys(this.key); + return des(keys, block, true, 0, null, padding); + }; + + this.decrypt = function(block, padding) { + const keys = desCreateKeys(this.key); + return des(keys, block, false, 0, null, padding); + }; + } + + // Use of this source code is governed by a BSD-style + // license that can be found in the LICENSE file. + + // Copyright 2010 pjacobs@xeekr.com . All rights reserved. + + // Modified by Recurity Labs GmbH + + // fixed/modified by Herbert Hanewinkel, www.haneWIN.de + // check www.haneWIN.de for the latest version + + // cast5.js is a Javascript implementation of CAST-128, as defined in RFC 2144. + // CAST-128 is a common OpenPGP cipher. + + + // CAST5 constructor + + function OpenPGPSymEncCAST5() { + this.BlockSize = 8; + this.KeySize = 16; + + this.setKey = function(key) { + this.masking = new Array(16); + this.rotate = new Array(16); + + this.reset(); + + if (key.length === this.KeySize) { + this.keySchedule(key); + } else { + throw new Error('CAST-128: keys must be 16 bytes'); + } + return true; + }; + + this.reset = function() { + for (let i = 0; i < 16; i++) { + this.masking[i] = 0; + this.rotate[i] = 0; + } + }; + + this.getBlockSize = function() { + return this.BlockSize; + }; + + this.encrypt = function(src) { + const dst = new Array(src.length); + + for (let i = 0; i < src.length; i += 8) { + let l = (src[i] << 24) | (src[i + 1] << 16) | (src[i + 2] << 8) | src[i + 3]; + let r = (src[i + 4] << 24) | (src[i + 5] << 16) | (src[i + 6] << 8) | src[i + 7]; + let t; + + t = r; + r = l ^ f1(r, this.masking[0], this.rotate[0]); + l = t; + t = r; + r = l ^ f2(r, this.masking[1], this.rotate[1]); + l = t; + t = r; + r = l ^ f3(r, this.masking[2], this.rotate[2]); + l = t; + t = r; + r = l ^ f1(r, this.masking[3], this.rotate[3]); + l = t; + + t = r; + r = l ^ f2(r, this.masking[4], this.rotate[4]); + l = t; + t = r; + r = l ^ f3(r, this.masking[5], this.rotate[5]); + l = t; + t = r; + r = l ^ f1(r, this.masking[6], this.rotate[6]); + l = t; + t = r; + r = l ^ f2(r, this.masking[7], this.rotate[7]); + l = t; + + t = r; + r = l ^ f3(r, this.masking[8], this.rotate[8]); + l = t; + t = r; + r = l ^ f1(r, this.masking[9], this.rotate[9]); + l = t; + t = r; + r = l ^ f2(r, this.masking[10], this.rotate[10]); + l = t; + t = r; + r = l ^ f3(r, this.masking[11], this.rotate[11]); + l = t; + + t = r; + r = l ^ f1(r, this.masking[12], this.rotate[12]); + l = t; + t = r; + r = l ^ f2(r, this.masking[13], this.rotate[13]); + l = t; + t = r; + r = l ^ f3(r, this.masking[14], this.rotate[14]); + l = t; + t = r; + r = l ^ f1(r, this.masking[15], this.rotate[15]); + l = t; + + dst[i] = (r >>> 24) & 255; + dst[i + 1] = (r >>> 16) & 255; + dst[i + 2] = (r >>> 8) & 255; + dst[i + 3] = r & 255; + dst[i + 4] = (l >>> 24) & 255; + dst[i + 5] = (l >>> 16) & 255; + dst[i + 6] = (l >>> 8) & 255; + dst[i + 7] = l & 255; + } + + return dst; + }; + + this.decrypt = function(src) { + const dst = new Array(src.length); + + for (let i = 0; i < src.length; i += 8) { + let l = (src[i] << 24) | (src[i + 1] << 16) | (src[i + 2] << 8) | src[i + 3]; + let r = (src[i + 4] << 24) | (src[i + 5] << 16) | (src[i + 6] << 8) | src[i + 7]; + let t; + + t = r; + r = l ^ f1(r, this.masking[15], this.rotate[15]); + l = t; + t = r; + r = l ^ f3(r, this.masking[14], this.rotate[14]); + l = t; + t = r; + r = l ^ f2(r, this.masking[13], this.rotate[13]); + l = t; + t = r; + r = l ^ f1(r, this.masking[12], this.rotate[12]); + l = t; + + t = r; + r = l ^ f3(r, this.masking[11], this.rotate[11]); + l = t; + t = r; + r = l ^ f2(r, this.masking[10], this.rotate[10]); + l = t; + t = r; + r = l ^ f1(r, this.masking[9], this.rotate[9]); + l = t; + t = r; + r = l ^ f3(r, this.masking[8], this.rotate[8]); + l = t; + + t = r; + r = l ^ f2(r, this.masking[7], this.rotate[7]); + l = t; + t = r; + r = l ^ f1(r, this.masking[6], this.rotate[6]); + l = t; + t = r; + r = l ^ f3(r, this.masking[5], this.rotate[5]); + l = t; + t = r; + r = l ^ f2(r, this.masking[4], this.rotate[4]); + l = t; + + t = r; + r = l ^ f1(r, this.masking[3], this.rotate[3]); + l = t; + t = r; + r = l ^ f3(r, this.masking[2], this.rotate[2]); + l = t; + t = r; + r = l ^ f2(r, this.masking[1], this.rotate[1]); + l = t; + t = r; + r = l ^ f1(r, this.masking[0], this.rotate[0]); + l = t; + + dst[i] = (r >>> 24) & 255; + dst[i + 1] = (r >>> 16) & 255; + dst[i + 2] = (r >>> 8) & 255; + dst[i + 3] = r & 255; + dst[i + 4] = (l >>> 24) & 255; + dst[i + 5] = (l >> 16) & 255; + dst[i + 6] = (l >> 8) & 255; + dst[i + 7] = l & 255; + } + + return dst; + }; + const scheduleA = new Array(4); + + scheduleA[0] = new Array(4); + scheduleA[0][0] = [4, 0, 0xd, 0xf, 0xc, 0xe, 0x8]; + scheduleA[0][1] = [5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa]; + scheduleA[0][2] = [6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9]; + scheduleA[0][3] = [7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb]; + + scheduleA[1] = new Array(4); + scheduleA[1][0] = [0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0]; + scheduleA[1][1] = [1, 4, 0, 2, 1, 3, 16 + 2]; + scheduleA[1][2] = [2, 5, 7, 6, 5, 4, 16 + 1]; + scheduleA[1][3] = [3, 7, 0xa, 9, 0xb, 8, 16 + 3]; + + scheduleA[2] = new Array(4); + scheduleA[2][0] = [4, 0, 0xd, 0xf, 0xc, 0xe, 8]; + scheduleA[2][1] = [5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa]; + scheduleA[2][2] = [6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9]; + scheduleA[2][3] = [7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb]; + + + scheduleA[3] = new Array(4); + scheduleA[3][0] = [0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0]; + scheduleA[3][1] = [1, 4, 0, 2, 1, 3, 16 + 2]; + scheduleA[3][2] = [2, 5, 7, 6, 5, 4, 16 + 1]; + scheduleA[3][3] = [3, 7, 0xa, 9, 0xb, 8, 16 + 3]; + + const scheduleB = new Array(4); + + scheduleB[0] = new Array(4); + scheduleB[0][0] = [16 + 8, 16 + 9, 16 + 7, 16 + 6, 16 + 2]; + scheduleB[0][1] = [16 + 0xa, 16 + 0xb, 16 + 5, 16 + 4, 16 + 6]; + scheduleB[0][2] = [16 + 0xc, 16 + 0xd, 16 + 3, 16 + 2, 16 + 9]; + scheduleB[0][3] = [16 + 0xe, 16 + 0xf, 16 + 1, 16 + 0, 16 + 0xc]; + + scheduleB[1] = new Array(4); + scheduleB[1][0] = [3, 2, 0xc, 0xd, 8]; + scheduleB[1][1] = [1, 0, 0xe, 0xf, 0xd]; + scheduleB[1][2] = [7, 6, 8, 9, 3]; + scheduleB[1][3] = [5, 4, 0xa, 0xb, 7]; + + + scheduleB[2] = new Array(4); + scheduleB[2][0] = [16 + 3, 16 + 2, 16 + 0xc, 16 + 0xd, 16 + 9]; + scheduleB[2][1] = [16 + 1, 16 + 0, 16 + 0xe, 16 + 0xf, 16 + 0xc]; + scheduleB[2][2] = [16 + 7, 16 + 6, 16 + 8, 16 + 9, 16 + 2]; + scheduleB[2][3] = [16 + 5, 16 + 4, 16 + 0xa, 16 + 0xb, 16 + 6]; + + + scheduleB[3] = new Array(4); + scheduleB[3][0] = [8, 9, 7, 6, 3]; + scheduleB[3][1] = [0xa, 0xb, 5, 4, 7]; + scheduleB[3][2] = [0xc, 0xd, 3, 2, 8]; + scheduleB[3][3] = [0xe, 0xf, 1, 0, 0xd]; + + // changed 'in' to 'inn' (in javascript 'in' is a reserved word) + this.keySchedule = function(inn) { + const t = new Array(8); + const k = new Array(32); + + let j; + + for (let i = 0; i < 4; i++) { + j = i * 4; + t[i] = (inn[j] << 24) | (inn[j + 1] << 16) | (inn[j + 2] << 8) | inn[j + 3]; + } + + const x = [6, 7, 4, 5]; + let ki = 0; + let w; + + for (let half = 0; half < 2; half++) { + for (let round = 0; round < 4; round++) { + for (j = 0; j < 4; j++) { + const a = scheduleA[round][j]; + w = t[a[1]]; + + w ^= sBox[4][(t[a[2] >>> 2] >>> (24 - 8 * (a[2] & 3))) & 0xff]; + w ^= sBox[5][(t[a[3] >>> 2] >>> (24 - 8 * (a[3] & 3))) & 0xff]; + w ^= sBox[6][(t[a[4] >>> 2] >>> (24 - 8 * (a[4] & 3))) & 0xff]; + w ^= sBox[7][(t[a[5] >>> 2] >>> (24 - 8 * (a[5] & 3))) & 0xff]; + w ^= sBox[x[j]][(t[a[6] >>> 2] >>> (24 - 8 * (a[6] & 3))) & 0xff]; + t[a[0]] = w; + } + + for (j = 0; j < 4; j++) { + const b = scheduleB[round][j]; + w = sBox[4][(t[b[0] >>> 2] >>> (24 - 8 * (b[0] & 3))) & 0xff]; + + w ^= sBox[5][(t[b[1] >>> 2] >>> (24 - 8 * (b[1] & 3))) & 0xff]; + w ^= sBox[6][(t[b[2] >>> 2] >>> (24 - 8 * (b[2] & 3))) & 0xff]; + w ^= sBox[7][(t[b[3] >>> 2] >>> (24 - 8 * (b[3] & 3))) & 0xff]; + w ^= sBox[4 + j][(t[b[4] >>> 2] >>> (24 - 8 * (b[4] & 3))) & 0xff]; + k[ki] = w; + ki++; + } + } + } + + for (let i = 0; i < 16; i++) { + this.masking[i] = k[i]; + this.rotate[i] = k[16 + i] & 0x1f; + } + }; + + // These are the three 'f' functions. See RFC 2144, section 2.2. + + function f1(d, m, r) { + const t = m + d; + const I = (t << r) | (t >>> (32 - r)); + return ((sBox[0][I >>> 24] ^ sBox[1][(I >>> 16) & 255]) - sBox[2][(I >>> 8) & 255]) + sBox[3][I & 255]; + } + + function f2(d, m, r) { + const t = m ^ d; + const I = (t << r) | (t >>> (32 - r)); + return ((sBox[0][I >>> 24] - sBox[1][(I >>> 16) & 255]) + sBox[2][(I >>> 8) & 255]) ^ sBox[3][I & 255]; + } + + function f3(d, m, r) { + const t = m - d; + const I = (t << r) | (t >>> (32 - r)); + return ((sBox[0][I >>> 24] + sBox[1][(I >>> 16) & 255]) ^ sBox[2][(I >>> 8) & 255]) - sBox[3][I & 255]; + } + + const sBox = new Array(8); + sBox[0] = [ + 0x30fb40d4, 0x9fa0ff0b, 0x6beccd2f, 0x3f258c7a, 0x1e213f2f, 0x9c004dd3, 0x6003e540, 0xcf9fc949, + 0xbfd4af27, 0x88bbbdb5, 0xe2034090, 0x98d09675, 0x6e63a0e0, 0x15c361d2, 0xc2e7661d, 0x22d4ff8e, + 0x28683b6f, 0xc07fd059, 0xff2379c8, 0x775f50e2, 0x43c340d3, 0xdf2f8656, 0x887ca41a, 0xa2d2bd2d, + 0xa1c9e0d6, 0x346c4819, 0x61b76d87, 0x22540f2f, 0x2abe32e1, 0xaa54166b, 0x22568e3a, 0xa2d341d0, + 0x66db40c8, 0xa784392f, 0x004dff2f, 0x2db9d2de, 0x97943fac, 0x4a97c1d8, 0x527644b7, 0xb5f437a7, + 0xb82cbaef, 0xd751d159, 0x6ff7f0ed, 0x5a097a1f, 0x827b68d0, 0x90ecf52e, 0x22b0c054, 0xbc8e5935, + 0x4b6d2f7f, 0x50bb64a2, 0xd2664910, 0xbee5812d, 0xb7332290, 0xe93b159f, 0xb48ee411, 0x4bff345d, + 0xfd45c240, 0xad31973f, 0xc4f6d02e, 0x55fc8165, 0xd5b1caad, 0xa1ac2dae, 0xa2d4b76d, 0xc19b0c50, + 0x882240f2, 0x0c6e4f38, 0xa4e4bfd7, 0x4f5ba272, 0x564c1d2f, 0xc59c5319, 0xb949e354, 0xb04669fe, + 0xb1b6ab8a, 0xc71358dd, 0x6385c545, 0x110f935d, 0x57538ad5, 0x6a390493, 0xe63d37e0, 0x2a54f6b3, + 0x3a787d5f, 0x6276a0b5, 0x19a6fcdf, 0x7a42206a, 0x29f9d4d5, 0xf61b1891, 0xbb72275e, 0xaa508167, + 0x38901091, 0xc6b505eb, 0x84c7cb8c, 0x2ad75a0f, 0x874a1427, 0xa2d1936b, 0x2ad286af, 0xaa56d291, + 0xd7894360, 0x425c750d, 0x93b39e26, 0x187184c9, 0x6c00b32d, 0x73e2bb14, 0xa0bebc3c, 0x54623779, + 0x64459eab, 0x3f328b82, 0x7718cf82, 0x59a2cea6, 0x04ee002e, 0x89fe78e6, 0x3fab0950, 0x325ff6c2, + 0x81383f05, 0x6963c5c8, 0x76cb5ad6, 0xd49974c9, 0xca180dcf, 0x380782d5, 0xc7fa5cf6, 0x8ac31511, + 0x35e79e13, 0x47da91d0, 0xf40f9086, 0xa7e2419e, 0x31366241, 0x051ef495, 0xaa573b04, 0x4a805d8d, + 0x548300d0, 0x00322a3c, 0xbf64cddf, 0xba57a68e, 0x75c6372b, 0x50afd341, 0xa7c13275, 0x915a0bf5, + 0x6b54bfab, 0x2b0b1426, 0xab4cc9d7, 0x449ccd82, 0xf7fbf265, 0xab85c5f3, 0x1b55db94, 0xaad4e324, + 0xcfa4bd3f, 0x2deaa3e2, 0x9e204d02, 0xc8bd25ac, 0xeadf55b3, 0xd5bd9e98, 0xe31231b2, 0x2ad5ad6c, + 0x954329de, 0xadbe4528, 0xd8710f69, 0xaa51c90f, 0xaa786bf6, 0x22513f1e, 0xaa51a79b, 0x2ad344cc, + 0x7b5a41f0, 0xd37cfbad, 0x1b069505, 0x41ece491, 0xb4c332e6, 0x032268d4, 0xc9600acc, 0xce387e6d, + 0xbf6bb16c, 0x6a70fb78, 0x0d03d9c9, 0xd4df39de, 0xe01063da, 0x4736f464, 0x5ad328d8, 0xb347cc96, + 0x75bb0fc3, 0x98511bfb, 0x4ffbcc35, 0xb58bcf6a, 0xe11f0abc, 0xbfc5fe4a, 0xa70aec10, 0xac39570a, + 0x3f04442f, 0x6188b153, 0xe0397a2e, 0x5727cb79, 0x9ceb418f, 0x1cacd68d, 0x2ad37c96, 0x0175cb9d, + 0xc69dff09, 0xc75b65f0, 0xd9db40d8, 0xec0e7779, 0x4744ead4, 0xb11c3274, 0xdd24cb9e, 0x7e1c54bd, + 0xf01144f9, 0xd2240eb1, 0x9675b3fd, 0xa3ac3755, 0xd47c27af, 0x51c85f4d, 0x56907596, 0xa5bb15e6, + 0x580304f0, 0xca042cf1, 0x011a37ea, 0x8dbfaadb, 0x35ba3e4a, 0x3526ffa0, 0xc37b4d09, 0xbc306ed9, + 0x98a52666, 0x5648f725, 0xff5e569d, 0x0ced63d0, 0x7c63b2cf, 0x700b45e1, 0xd5ea50f1, 0x85a92872, + 0xaf1fbda7, 0xd4234870, 0xa7870bf3, 0x2d3b4d79, 0x42e04198, 0x0cd0ede7, 0x26470db8, 0xf881814c, + 0x474d6ad7, 0x7c0c5e5c, 0xd1231959, 0x381b7298, 0xf5d2f4db, 0xab838653, 0x6e2f1e23, 0x83719c9e, + 0xbd91e046, 0x9a56456e, 0xdc39200c, 0x20c8c571, 0x962bda1c, 0xe1e696ff, 0xb141ab08, 0x7cca89b9, + 0x1a69e783, 0x02cc4843, 0xa2f7c579, 0x429ef47d, 0x427b169c, 0x5ac9f049, 0xdd8f0f00, 0x5c8165bf + ]; + + sBox[1] = [ + 0x1f201094, 0xef0ba75b, 0x69e3cf7e, 0x393f4380, 0xfe61cf7a, 0xeec5207a, 0x55889c94, 0x72fc0651, + 0xada7ef79, 0x4e1d7235, 0xd55a63ce, 0xde0436ba, 0x99c430ef, 0x5f0c0794, 0x18dcdb7d, 0xa1d6eff3, + 0xa0b52f7b, 0x59e83605, 0xee15b094, 0xe9ffd909, 0xdc440086, 0xef944459, 0xba83ccb3, 0xe0c3cdfb, + 0xd1da4181, 0x3b092ab1, 0xf997f1c1, 0xa5e6cf7b, 0x01420ddb, 0xe4e7ef5b, 0x25a1ff41, 0xe180f806, + 0x1fc41080, 0x179bee7a, 0xd37ac6a9, 0xfe5830a4, 0x98de8b7f, 0x77e83f4e, 0x79929269, 0x24fa9f7b, + 0xe113c85b, 0xacc40083, 0xd7503525, 0xf7ea615f, 0x62143154, 0x0d554b63, 0x5d681121, 0xc866c359, + 0x3d63cf73, 0xcee234c0, 0xd4d87e87, 0x5c672b21, 0x071f6181, 0x39f7627f, 0x361e3084, 0xe4eb573b, + 0x602f64a4, 0xd63acd9c, 0x1bbc4635, 0x9e81032d, 0x2701f50c, 0x99847ab4, 0xa0e3df79, 0xba6cf38c, + 0x10843094, 0x2537a95e, 0xf46f6ffe, 0xa1ff3b1f, 0x208cfb6a, 0x8f458c74, 0xd9e0a227, 0x4ec73a34, + 0xfc884f69, 0x3e4de8df, 0xef0e0088, 0x3559648d, 0x8a45388c, 0x1d804366, 0x721d9bfd, 0xa58684bb, + 0xe8256333, 0x844e8212, 0x128d8098, 0xfed33fb4, 0xce280ae1, 0x27e19ba5, 0xd5a6c252, 0xe49754bd, + 0xc5d655dd, 0xeb667064, 0x77840b4d, 0xa1b6a801, 0x84db26a9, 0xe0b56714, 0x21f043b7, 0xe5d05860, + 0x54f03084, 0x066ff472, 0xa31aa153, 0xdadc4755, 0xb5625dbf, 0x68561be6, 0x83ca6b94, 0x2d6ed23b, + 0xeccf01db, 0xa6d3d0ba, 0xb6803d5c, 0xaf77a709, 0x33b4a34c, 0x397bc8d6, 0x5ee22b95, 0x5f0e5304, + 0x81ed6f61, 0x20e74364, 0xb45e1378, 0xde18639b, 0x881ca122, 0xb96726d1, 0x8049a7e8, 0x22b7da7b, + 0x5e552d25, 0x5272d237, 0x79d2951c, 0xc60d894c, 0x488cb402, 0x1ba4fe5b, 0xa4b09f6b, 0x1ca815cf, + 0xa20c3005, 0x8871df63, 0xb9de2fcb, 0x0cc6c9e9, 0x0beeff53, 0xe3214517, 0xb4542835, 0x9f63293c, + 0xee41e729, 0x6e1d2d7c, 0x50045286, 0x1e6685f3, 0xf33401c6, 0x30a22c95, 0x31a70850, 0x60930f13, + 0x73f98417, 0xa1269859, 0xec645c44, 0x52c877a9, 0xcdff33a6, 0xa02b1741, 0x7cbad9a2, 0x2180036f, + 0x50d99c08, 0xcb3f4861, 0xc26bd765, 0x64a3f6ab, 0x80342676, 0x25a75e7b, 0xe4e6d1fc, 0x20c710e6, + 0xcdf0b680, 0x17844d3b, 0x31eef84d, 0x7e0824e4, 0x2ccb49eb, 0x846a3bae, 0x8ff77888, 0xee5d60f6, + 0x7af75673, 0x2fdd5cdb, 0xa11631c1, 0x30f66f43, 0xb3faec54, 0x157fd7fa, 0xef8579cc, 0xd152de58, + 0xdb2ffd5e, 0x8f32ce19, 0x306af97a, 0x02f03ef8, 0x99319ad5, 0xc242fa0f, 0xa7e3ebb0, 0xc68e4906, + 0xb8da230c, 0x80823028, 0xdcdef3c8, 0xd35fb171, 0x088a1bc8, 0xbec0c560, 0x61a3c9e8, 0xbca8f54d, + 0xc72feffa, 0x22822e99, 0x82c570b4, 0xd8d94e89, 0x8b1c34bc, 0x301e16e6, 0x273be979, 0xb0ffeaa6, + 0x61d9b8c6, 0x00b24869, 0xb7ffce3f, 0x08dc283b, 0x43daf65a, 0xf7e19798, 0x7619b72f, 0x8f1c9ba4, + 0xdc8637a0, 0x16a7d3b1, 0x9fc393b7, 0xa7136eeb, 0xc6bcc63e, 0x1a513742, 0xef6828bc, 0x520365d6, + 0x2d6a77ab, 0x3527ed4b, 0x821fd216, 0x095c6e2e, 0xdb92f2fb, 0x5eea29cb, 0x145892f5, 0x91584f7f, + 0x5483697b, 0x2667a8cc, 0x85196048, 0x8c4bacea, 0x833860d4, 0x0d23e0f9, 0x6c387e8a, 0x0ae6d249, + 0xb284600c, 0xd835731d, 0xdcb1c647, 0xac4c56ea, 0x3ebd81b3, 0x230eabb0, 0x6438bc87, 0xf0b5b1fa, + 0x8f5ea2b3, 0xfc184642, 0x0a036b7a, 0x4fb089bd, 0x649da589, 0xa345415e, 0x5c038323, 0x3e5d3bb9, + 0x43d79572, 0x7e6dd07c, 0x06dfdf1e, 0x6c6cc4ef, 0x7160a539, 0x73bfbe70, 0x83877605, 0x4523ecf1 + ]; + + sBox[2] = [ + 0x8defc240, 0x25fa5d9f, 0xeb903dbf, 0xe810c907, 0x47607fff, 0x369fe44b, 0x8c1fc644, 0xaececa90, + 0xbeb1f9bf, 0xeefbcaea, 0xe8cf1950, 0x51df07ae, 0x920e8806, 0xf0ad0548, 0xe13c8d83, 0x927010d5, + 0x11107d9f, 0x07647db9, 0xb2e3e4d4, 0x3d4f285e, 0xb9afa820, 0xfade82e0, 0xa067268b, 0x8272792e, + 0x553fb2c0, 0x489ae22b, 0xd4ef9794, 0x125e3fbc, 0x21fffcee, 0x825b1bfd, 0x9255c5ed, 0x1257a240, + 0x4e1a8302, 0xbae07fff, 0x528246e7, 0x8e57140e, 0x3373f7bf, 0x8c9f8188, 0xa6fc4ee8, 0xc982b5a5, + 0xa8c01db7, 0x579fc264, 0x67094f31, 0xf2bd3f5f, 0x40fff7c1, 0x1fb78dfc, 0x8e6bd2c1, 0x437be59b, + 0x99b03dbf, 0xb5dbc64b, 0x638dc0e6, 0x55819d99, 0xa197c81c, 0x4a012d6e, 0xc5884a28, 0xccc36f71, + 0xb843c213, 0x6c0743f1, 0x8309893c, 0x0feddd5f, 0x2f7fe850, 0xd7c07f7e, 0x02507fbf, 0x5afb9a04, + 0xa747d2d0, 0x1651192e, 0xaf70bf3e, 0x58c31380, 0x5f98302e, 0x727cc3c4, 0x0a0fb402, 0x0f7fef82, + 0x8c96fdad, 0x5d2c2aae, 0x8ee99a49, 0x50da88b8, 0x8427f4a0, 0x1eac5790, 0x796fb449, 0x8252dc15, + 0xefbd7d9b, 0xa672597d, 0xada840d8, 0x45f54504, 0xfa5d7403, 0xe83ec305, 0x4f91751a, 0x925669c2, + 0x23efe941, 0xa903f12e, 0x60270df2, 0x0276e4b6, 0x94fd6574, 0x927985b2, 0x8276dbcb, 0x02778176, + 0xf8af918d, 0x4e48f79e, 0x8f616ddf, 0xe29d840e, 0x842f7d83, 0x340ce5c8, 0x96bbb682, 0x93b4b148, + 0xef303cab, 0x984faf28, 0x779faf9b, 0x92dc560d, 0x224d1e20, 0x8437aa88, 0x7d29dc96, 0x2756d3dc, + 0x8b907cee, 0xb51fd240, 0xe7c07ce3, 0xe566b4a1, 0xc3e9615e, 0x3cf8209d, 0x6094d1e3, 0xcd9ca341, + 0x5c76460e, 0x00ea983b, 0xd4d67881, 0xfd47572c, 0xf76cedd9, 0xbda8229c, 0x127dadaa, 0x438a074e, + 0x1f97c090, 0x081bdb8a, 0x93a07ebe, 0xb938ca15, 0x97b03cff, 0x3dc2c0f8, 0x8d1ab2ec, 0x64380e51, + 0x68cc7bfb, 0xd90f2788, 0x12490181, 0x5de5ffd4, 0xdd7ef86a, 0x76a2e214, 0xb9a40368, 0x925d958f, + 0x4b39fffa, 0xba39aee9, 0xa4ffd30b, 0xfaf7933b, 0x6d498623, 0x193cbcfa, 0x27627545, 0x825cf47a, + 0x61bd8ba0, 0xd11e42d1, 0xcead04f4, 0x127ea392, 0x10428db7, 0x8272a972, 0x9270c4a8, 0x127de50b, + 0x285ba1c8, 0x3c62f44f, 0x35c0eaa5, 0xe805d231, 0x428929fb, 0xb4fcdf82, 0x4fb66a53, 0x0e7dc15b, + 0x1f081fab, 0x108618ae, 0xfcfd086d, 0xf9ff2889, 0x694bcc11, 0x236a5cae, 0x12deca4d, 0x2c3f8cc5, + 0xd2d02dfe, 0xf8ef5896, 0xe4cf52da, 0x95155b67, 0x494a488c, 0xb9b6a80c, 0x5c8f82bc, 0x89d36b45, + 0x3a609437, 0xec00c9a9, 0x44715253, 0x0a874b49, 0xd773bc40, 0x7c34671c, 0x02717ef6, 0x4feb5536, + 0xa2d02fff, 0xd2bf60c4, 0xd43f03c0, 0x50b4ef6d, 0x07478cd1, 0x006e1888, 0xa2e53f55, 0xb9e6d4bc, + 0xa2048016, 0x97573833, 0xd7207d67, 0xde0f8f3d, 0x72f87b33, 0xabcc4f33, 0x7688c55d, 0x7b00a6b0, + 0x947b0001, 0x570075d2, 0xf9bb88f8, 0x8942019e, 0x4264a5ff, 0x856302e0, 0x72dbd92b, 0xee971b69, + 0x6ea22fde, 0x5f08ae2b, 0xaf7a616d, 0xe5c98767, 0xcf1febd2, 0x61efc8c2, 0xf1ac2571, 0xcc8239c2, + 0x67214cb8, 0xb1e583d1, 0xb7dc3e62, 0x7f10bdce, 0xf90a5c38, 0x0ff0443d, 0x606e6dc6, 0x60543a49, + 0x5727c148, 0x2be98a1d, 0x8ab41738, 0x20e1be24, 0xaf96da0f, 0x68458425, 0x99833be5, 0x600d457d, + 0x282f9350, 0x8334b362, 0xd91d1120, 0x2b6d8da0, 0x642b1e31, 0x9c305a00, 0x52bce688, 0x1b03588a, + 0xf7baefd5, 0x4142ed9c, 0xa4315c11, 0x83323ec5, 0xdfef4636, 0xa133c501, 0xe9d3531c, 0xee353783 + ]; + + sBox[3] = [ + 0x9db30420, 0x1fb6e9de, 0xa7be7bef, 0xd273a298, 0x4a4f7bdb, 0x64ad8c57, 0x85510443, 0xfa020ed1, + 0x7e287aff, 0xe60fb663, 0x095f35a1, 0x79ebf120, 0xfd059d43, 0x6497b7b1, 0xf3641f63, 0x241e4adf, + 0x28147f5f, 0x4fa2b8cd, 0xc9430040, 0x0cc32220, 0xfdd30b30, 0xc0a5374f, 0x1d2d00d9, 0x24147b15, + 0xee4d111a, 0x0fca5167, 0x71ff904c, 0x2d195ffe, 0x1a05645f, 0x0c13fefe, 0x081b08ca, 0x05170121, + 0x80530100, 0xe83e5efe, 0xac9af4f8, 0x7fe72701, 0xd2b8ee5f, 0x06df4261, 0xbb9e9b8a, 0x7293ea25, + 0xce84ffdf, 0xf5718801, 0x3dd64b04, 0xa26f263b, 0x7ed48400, 0x547eebe6, 0x446d4ca0, 0x6cf3d6f5, + 0x2649abdf, 0xaea0c7f5, 0x36338cc1, 0x503f7e93, 0xd3772061, 0x11b638e1, 0x72500e03, 0xf80eb2bb, + 0xabe0502e, 0xec8d77de, 0x57971e81, 0xe14f6746, 0xc9335400, 0x6920318f, 0x081dbb99, 0xffc304a5, + 0x4d351805, 0x7f3d5ce3, 0xa6c866c6, 0x5d5bcca9, 0xdaec6fea, 0x9f926f91, 0x9f46222f, 0x3991467d, + 0xa5bf6d8e, 0x1143c44f, 0x43958302, 0xd0214eeb, 0x022083b8, 0x3fb6180c, 0x18f8931e, 0x281658e6, + 0x26486e3e, 0x8bd78a70, 0x7477e4c1, 0xb506e07c, 0xf32d0a25, 0x79098b02, 0xe4eabb81, 0x28123b23, + 0x69dead38, 0x1574ca16, 0xdf871b62, 0x211c40b7, 0xa51a9ef9, 0x0014377b, 0x041e8ac8, 0x09114003, + 0xbd59e4d2, 0xe3d156d5, 0x4fe876d5, 0x2f91a340, 0x557be8de, 0x00eae4a7, 0x0ce5c2ec, 0x4db4bba6, + 0xe756bdff, 0xdd3369ac, 0xec17b035, 0x06572327, 0x99afc8b0, 0x56c8c391, 0x6b65811c, 0x5e146119, + 0x6e85cb75, 0xbe07c002, 0xc2325577, 0x893ff4ec, 0x5bbfc92d, 0xd0ec3b25, 0xb7801ab7, 0x8d6d3b24, + 0x20c763ef, 0xc366a5fc, 0x9c382880, 0x0ace3205, 0xaac9548a, 0xeca1d7c7, 0x041afa32, 0x1d16625a, + 0x6701902c, 0x9b757a54, 0x31d477f7, 0x9126b031, 0x36cc6fdb, 0xc70b8b46, 0xd9e66a48, 0x56e55a79, + 0x026a4ceb, 0x52437eff, 0x2f8f76b4, 0x0df980a5, 0x8674cde3, 0xedda04eb, 0x17a9be04, 0x2c18f4df, + 0xb7747f9d, 0xab2af7b4, 0xefc34d20, 0x2e096b7c, 0x1741a254, 0xe5b6a035, 0x213d42f6, 0x2c1c7c26, + 0x61c2f50f, 0x6552daf9, 0xd2c231f8, 0x25130f69, 0xd8167fa2, 0x0418f2c8, 0x001a96a6, 0x0d1526ab, + 0x63315c21, 0x5e0a72ec, 0x49bafefd, 0x187908d9, 0x8d0dbd86, 0x311170a7, 0x3e9b640c, 0xcc3e10d7, + 0xd5cad3b6, 0x0caec388, 0xf73001e1, 0x6c728aff, 0x71eae2a1, 0x1f9af36e, 0xcfcbd12f, 0xc1de8417, + 0xac07be6b, 0xcb44a1d8, 0x8b9b0f56, 0x013988c3, 0xb1c52fca, 0xb4be31cd, 0xd8782806, 0x12a3a4e2, + 0x6f7de532, 0x58fd7eb6, 0xd01ee900, 0x24adffc2, 0xf4990fc5, 0x9711aac5, 0x001d7b95, 0x82e5e7d2, + 0x109873f6, 0x00613096, 0xc32d9521, 0xada121ff, 0x29908415, 0x7fbb977f, 0xaf9eb3db, 0x29c9ed2a, + 0x5ce2a465, 0xa730f32c, 0xd0aa3fe8, 0x8a5cc091, 0xd49e2ce7, 0x0ce454a9, 0xd60acd86, 0x015f1919, + 0x77079103, 0xdea03af6, 0x78a8565e, 0xdee356df, 0x21f05cbe, 0x8b75e387, 0xb3c50651, 0xb8a5c3ef, + 0xd8eeb6d2, 0xe523be77, 0xc2154529, 0x2f69efdf, 0xafe67afb, 0xf470c4b2, 0xf3e0eb5b, 0xd6cc9876, + 0x39e4460c, 0x1fda8538, 0x1987832f, 0xca007367, 0xa99144f8, 0x296b299e, 0x492fc295, 0x9266beab, + 0xb5676e69, 0x9bd3ddda, 0xdf7e052f, 0xdb25701c, 0x1b5e51ee, 0xf65324e6, 0x6afce36c, 0x0316cc04, + 0x8644213e, 0xb7dc59d0, 0x7965291f, 0xccd6fd43, 0x41823979, 0x932bcdf6, 0xb657c34d, 0x4edfd282, + 0x7ae5290c, 0x3cb9536b, 0x851e20fe, 0x9833557e, 0x13ecf0b0, 0xd3ffb372, 0x3f85c5c1, 0x0aef7ed2 + ]; + + sBox[4] = [ + 0x7ec90c04, 0x2c6e74b9, 0x9b0e66df, 0xa6337911, 0xb86a7fff, 0x1dd358f5, 0x44dd9d44, 0x1731167f, + 0x08fbf1fa, 0xe7f511cc, 0xd2051b00, 0x735aba00, 0x2ab722d8, 0x386381cb, 0xacf6243a, 0x69befd7a, + 0xe6a2e77f, 0xf0c720cd, 0xc4494816, 0xccf5c180, 0x38851640, 0x15b0a848, 0xe68b18cb, 0x4caadeff, + 0x5f480a01, 0x0412b2aa, 0x259814fc, 0x41d0efe2, 0x4e40b48d, 0x248eb6fb, 0x8dba1cfe, 0x41a99b02, + 0x1a550a04, 0xba8f65cb, 0x7251f4e7, 0x95a51725, 0xc106ecd7, 0x97a5980a, 0xc539b9aa, 0x4d79fe6a, + 0xf2f3f763, 0x68af8040, 0xed0c9e56, 0x11b4958b, 0xe1eb5a88, 0x8709e6b0, 0xd7e07156, 0x4e29fea7, + 0x6366e52d, 0x02d1c000, 0xc4ac8e05, 0x9377f571, 0x0c05372a, 0x578535f2, 0x2261be02, 0xd642a0c9, + 0xdf13a280, 0x74b55bd2, 0x682199c0, 0xd421e5ec, 0x53fb3ce8, 0xc8adedb3, 0x28a87fc9, 0x3d959981, + 0x5c1ff900, 0xfe38d399, 0x0c4eff0b, 0x062407ea, 0xaa2f4fb1, 0x4fb96976, 0x90c79505, 0xb0a8a774, + 0xef55a1ff, 0xe59ca2c2, 0xa6b62d27, 0xe66a4263, 0xdf65001f, 0x0ec50966, 0xdfdd55bc, 0x29de0655, + 0x911e739a, 0x17af8975, 0x32c7911c, 0x89f89468, 0x0d01e980, 0x524755f4, 0x03b63cc9, 0x0cc844b2, + 0xbcf3f0aa, 0x87ac36e9, 0xe53a7426, 0x01b3d82b, 0x1a9e7449, 0x64ee2d7e, 0xcddbb1da, 0x01c94910, + 0xb868bf80, 0x0d26f3fd, 0x9342ede7, 0x04a5c284, 0x636737b6, 0x50f5b616, 0xf24766e3, 0x8eca36c1, + 0x136e05db, 0xfef18391, 0xfb887a37, 0xd6e7f7d4, 0xc7fb7dc9, 0x3063fcdf, 0xb6f589de, 0xec2941da, + 0x26e46695, 0xb7566419, 0xf654efc5, 0xd08d58b7, 0x48925401, 0xc1bacb7f, 0xe5ff550f, 0xb6083049, + 0x5bb5d0e8, 0x87d72e5a, 0xab6a6ee1, 0x223a66ce, 0xc62bf3cd, 0x9e0885f9, 0x68cb3e47, 0x086c010f, + 0xa21de820, 0xd18b69de, 0xf3f65777, 0xfa02c3f6, 0x407edac3, 0xcbb3d550, 0x1793084d, 0xb0d70eba, + 0x0ab378d5, 0xd951fb0c, 0xded7da56, 0x4124bbe4, 0x94ca0b56, 0x0f5755d1, 0xe0e1e56e, 0x6184b5be, + 0x580a249f, 0x94f74bc0, 0xe327888e, 0x9f7b5561, 0xc3dc0280, 0x05687715, 0x646c6bd7, 0x44904db3, + 0x66b4f0a3, 0xc0f1648a, 0x697ed5af, 0x49e92ff6, 0x309e374f, 0x2cb6356a, 0x85808573, 0x4991f840, + 0x76f0ae02, 0x083be84d, 0x28421c9a, 0x44489406, 0x736e4cb8, 0xc1092910, 0x8bc95fc6, 0x7d869cf4, + 0x134f616f, 0x2e77118d, 0xb31b2be1, 0xaa90b472, 0x3ca5d717, 0x7d161bba, 0x9cad9010, 0xaf462ba2, + 0x9fe459d2, 0x45d34559, 0xd9f2da13, 0xdbc65487, 0xf3e4f94e, 0x176d486f, 0x097c13ea, 0x631da5c7, + 0x445f7382, 0x175683f4, 0xcdc66a97, 0x70be0288, 0xb3cdcf72, 0x6e5dd2f3, 0x20936079, 0x459b80a5, + 0xbe60e2db, 0xa9c23101, 0xeba5315c, 0x224e42f2, 0x1c5c1572, 0xf6721b2c, 0x1ad2fff3, 0x8c25404e, + 0x324ed72f, 0x4067b7fd, 0x0523138e, 0x5ca3bc78, 0xdc0fd66e, 0x75922283, 0x784d6b17, 0x58ebb16e, + 0x44094f85, 0x3f481d87, 0xfcfeae7b, 0x77b5ff76, 0x8c2302bf, 0xaaf47556, 0x5f46b02a, 0x2b092801, + 0x3d38f5f7, 0x0ca81f36, 0x52af4a8a, 0x66d5e7c0, 0xdf3b0874, 0x95055110, 0x1b5ad7a8, 0xf61ed5ad, + 0x6cf6e479, 0x20758184, 0xd0cefa65, 0x88f7be58, 0x4a046826, 0x0ff6f8f3, 0xa09c7f70, 0x5346aba0, + 0x5ce96c28, 0xe176eda3, 0x6bac307f, 0x376829d2, 0x85360fa9, 0x17e3fe2a, 0x24b79767, 0xf5a96b20, + 0xd6cd2595, 0x68ff1ebf, 0x7555442c, 0xf19f06be, 0xf9e0659a, 0xeeb9491d, 0x34010718, 0xbb30cab8, + 0xe822fe15, 0x88570983, 0x750e6249, 0xda627e55, 0x5e76ffa8, 0xb1534546, 0x6d47de08, 0xefe9e7d4 + ]; + + sBox[5] = [ + 0xf6fa8f9d, 0x2cac6ce1, 0x4ca34867, 0xe2337f7c, 0x95db08e7, 0x016843b4, 0xeced5cbc, 0x325553ac, + 0xbf9f0960, 0xdfa1e2ed, 0x83f0579d, 0x63ed86b9, 0x1ab6a6b8, 0xde5ebe39, 0xf38ff732, 0x8989b138, + 0x33f14961, 0xc01937bd, 0xf506c6da, 0xe4625e7e, 0xa308ea99, 0x4e23e33c, 0x79cbd7cc, 0x48a14367, + 0xa3149619, 0xfec94bd5, 0xa114174a, 0xeaa01866, 0xa084db2d, 0x09a8486f, 0xa888614a, 0x2900af98, + 0x01665991, 0xe1992863, 0xc8f30c60, 0x2e78ef3c, 0xd0d51932, 0xcf0fec14, 0xf7ca07d2, 0xd0a82072, + 0xfd41197e, 0x9305a6b0, 0xe86be3da, 0x74bed3cd, 0x372da53c, 0x4c7f4448, 0xdab5d440, 0x6dba0ec3, + 0x083919a7, 0x9fbaeed9, 0x49dbcfb0, 0x4e670c53, 0x5c3d9c01, 0x64bdb941, 0x2c0e636a, 0xba7dd9cd, + 0xea6f7388, 0xe70bc762, 0x35f29adb, 0x5c4cdd8d, 0xf0d48d8c, 0xb88153e2, 0x08a19866, 0x1ae2eac8, + 0x284caf89, 0xaa928223, 0x9334be53, 0x3b3a21bf, 0x16434be3, 0x9aea3906, 0xefe8c36e, 0xf890cdd9, + 0x80226dae, 0xc340a4a3, 0xdf7e9c09, 0xa694a807, 0x5b7c5ecc, 0x221db3a6, 0x9a69a02f, 0x68818a54, + 0xceb2296f, 0x53c0843a, 0xfe893655, 0x25bfe68a, 0xb4628abc, 0xcf222ebf, 0x25ac6f48, 0xa9a99387, + 0x53bddb65, 0xe76ffbe7, 0xe967fd78, 0x0ba93563, 0x8e342bc1, 0xe8a11be9, 0x4980740d, 0xc8087dfc, + 0x8de4bf99, 0xa11101a0, 0x7fd37975, 0xda5a26c0, 0xe81f994f, 0x9528cd89, 0xfd339fed, 0xb87834bf, + 0x5f04456d, 0x22258698, 0xc9c4c83b, 0x2dc156be, 0x4f628daa, 0x57f55ec5, 0xe2220abe, 0xd2916ebf, + 0x4ec75b95, 0x24f2c3c0, 0x42d15d99, 0xcd0d7fa0, 0x7b6e27ff, 0xa8dc8af0, 0x7345c106, 0xf41e232f, + 0x35162386, 0xe6ea8926, 0x3333b094, 0x157ec6f2, 0x372b74af, 0x692573e4, 0xe9a9d848, 0xf3160289, + 0x3a62ef1d, 0xa787e238, 0xf3a5f676, 0x74364853, 0x20951063, 0x4576698d, 0xb6fad407, 0x592af950, + 0x36f73523, 0x4cfb6e87, 0x7da4cec0, 0x6c152daa, 0xcb0396a8, 0xc50dfe5d, 0xfcd707ab, 0x0921c42f, + 0x89dff0bb, 0x5fe2be78, 0x448f4f33, 0x754613c9, 0x2b05d08d, 0x48b9d585, 0xdc049441, 0xc8098f9b, + 0x7dede786, 0xc39a3373, 0x42410005, 0x6a091751, 0x0ef3c8a6, 0x890072d6, 0x28207682, 0xa9a9f7be, + 0xbf32679d, 0xd45b5b75, 0xb353fd00, 0xcbb0e358, 0x830f220a, 0x1f8fb214, 0xd372cf08, 0xcc3c4a13, + 0x8cf63166, 0x061c87be, 0x88c98f88, 0x6062e397, 0x47cf8e7a, 0xb6c85283, 0x3cc2acfb, 0x3fc06976, + 0x4e8f0252, 0x64d8314d, 0xda3870e3, 0x1e665459, 0xc10908f0, 0x513021a5, 0x6c5b68b7, 0x822f8aa0, + 0x3007cd3e, 0x74719eef, 0xdc872681, 0x073340d4, 0x7e432fd9, 0x0c5ec241, 0x8809286c, 0xf592d891, + 0x08a930f6, 0x957ef305, 0xb7fbffbd, 0xc266e96f, 0x6fe4ac98, 0xb173ecc0, 0xbc60b42a, 0x953498da, + 0xfba1ae12, 0x2d4bd736, 0x0f25faab, 0xa4f3fceb, 0xe2969123, 0x257f0c3d, 0x9348af49, 0x361400bc, + 0xe8816f4a, 0x3814f200, 0xa3f94043, 0x9c7a54c2, 0xbc704f57, 0xda41e7f9, 0xc25ad33a, 0x54f4a084, + 0xb17f5505, 0x59357cbe, 0xedbd15c8, 0x7f97c5ab, 0xba5ac7b5, 0xb6f6deaf, 0x3a479c3a, 0x5302da25, + 0x653d7e6a, 0x54268d49, 0x51a477ea, 0x5017d55b, 0xd7d25d88, 0x44136c76, 0x0404a8c8, 0xb8e5a121, + 0xb81a928a, 0x60ed5869, 0x97c55b96, 0xeaec991b, 0x29935913, 0x01fdb7f1, 0x088e8dfa, 0x9ab6f6f5, + 0x3b4cbf9f, 0x4a5de3ab, 0xe6051d35, 0xa0e1d855, 0xd36b4cf1, 0xf544edeb, 0xb0e93524, 0xbebb8fbd, + 0xa2d762cf, 0x49c92f54, 0x38b5f331, 0x7128a454, 0x48392905, 0xa65b1db8, 0x851c97bd, 0xd675cf2f + ]; + + sBox[6] = [ + 0x85e04019, 0x332bf567, 0x662dbfff, 0xcfc65693, 0x2a8d7f6f, 0xab9bc912, 0xde6008a1, 0x2028da1f, + 0x0227bce7, 0x4d642916, 0x18fac300, 0x50f18b82, 0x2cb2cb11, 0xb232e75c, 0x4b3695f2, 0xb28707de, + 0xa05fbcf6, 0xcd4181e9, 0xe150210c, 0xe24ef1bd, 0xb168c381, 0xfde4e789, 0x5c79b0d8, 0x1e8bfd43, + 0x4d495001, 0x38be4341, 0x913cee1d, 0x92a79c3f, 0x089766be, 0xbaeeadf4, 0x1286becf, 0xb6eacb19, + 0x2660c200, 0x7565bde4, 0x64241f7a, 0x8248dca9, 0xc3b3ad66, 0x28136086, 0x0bd8dfa8, 0x356d1cf2, + 0x107789be, 0xb3b2e9ce, 0x0502aa8f, 0x0bc0351e, 0x166bf52a, 0xeb12ff82, 0xe3486911, 0xd34d7516, + 0x4e7b3aff, 0x5f43671b, 0x9cf6e037, 0x4981ac83, 0x334266ce, 0x8c9341b7, 0xd0d854c0, 0xcb3a6c88, + 0x47bc2829, 0x4725ba37, 0xa66ad22b, 0x7ad61f1e, 0x0c5cbafa, 0x4437f107, 0xb6e79962, 0x42d2d816, + 0x0a961288, 0xe1a5c06e, 0x13749e67, 0x72fc081a, 0xb1d139f7, 0xf9583745, 0xcf19df58, 0xbec3f756, + 0xc06eba30, 0x07211b24, 0x45c28829, 0xc95e317f, 0xbc8ec511, 0x38bc46e9, 0xc6e6fa14, 0xbae8584a, + 0xad4ebc46, 0x468f508b, 0x7829435f, 0xf124183b, 0x821dba9f, 0xaff60ff4, 0xea2c4e6d, 0x16e39264, + 0x92544a8b, 0x009b4fc3, 0xaba68ced, 0x9ac96f78, 0x06a5b79a, 0xb2856e6e, 0x1aec3ca9, 0xbe838688, + 0x0e0804e9, 0x55f1be56, 0xe7e5363b, 0xb3a1f25d, 0xf7debb85, 0x61fe033c, 0x16746233, 0x3c034c28, + 0xda6d0c74, 0x79aac56c, 0x3ce4e1ad, 0x51f0c802, 0x98f8f35a, 0x1626a49f, 0xeed82b29, 0x1d382fe3, + 0x0c4fb99a, 0xbb325778, 0x3ec6d97b, 0x6e77a6a9, 0xcb658b5c, 0xd45230c7, 0x2bd1408b, 0x60c03eb7, + 0xb9068d78, 0xa33754f4, 0xf430c87d, 0xc8a71302, 0xb96d8c32, 0xebd4e7be, 0xbe8b9d2d, 0x7979fb06, + 0xe7225308, 0x8b75cf77, 0x11ef8da4, 0xe083c858, 0x8d6b786f, 0x5a6317a6, 0xfa5cf7a0, 0x5dda0033, + 0xf28ebfb0, 0xf5b9c310, 0xa0eac280, 0x08b9767a, 0xa3d9d2b0, 0x79d34217, 0x021a718d, 0x9ac6336a, + 0x2711fd60, 0x438050e3, 0x069908a8, 0x3d7fedc4, 0x826d2bef, 0x4eeb8476, 0x488dcf25, 0x36c9d566, + 0x28e74e41, 0xc2610aca, 0x3d49a9cf, 0xbae3b9df, 0xb65f8de6, 0x92aeaf64, 0x3ac7d5e6, 0x9ea80509, + 0xf22b017d, 0xa4173f70, 0xdd1e16c3, 0x15e0d7f9, 0x50b1b887, 0x2b9f4fd5, 0x625aba82, 0x6a017962, + 0x2ec01b9c, 0x15488aa9, 0xd716e740, 0x40055a2c, 0x93d29a22, 0xe32dbf9a, 0x058745b9, 0x3453dc1e, + 0xd699296e, 0x496cff6f, 0x1c9f4986, 0xdfe2ed07, 0xb87242d1, 0x19de7eae, 0x053e561a, 0x15ad6f8c, + 0x66626c1c, 0x7154c24c, 0xea082b2a, 0x93eb2939, 0x17dcb0f0, 0x58d4f2ae, 0x9ea294fb, 0x52cf564c, + 0x9883fe66, 0x2ec40581, 0x763953c3, 0x01d6692e, 0xd3a0c108, 0xa1e7160e, 0xe4f2dfa6, 0x693ed285, + 0x74904698, 0x4c2b0edd, 0x4f757656, 0x5d393378, 0xa132234f, 0x3d321c5d, 0xc3f5e194, 0x4b269301, + 0xc79f022f, 0x3c997e7e, 0x5e4f9504, 0x3ffafbbd, 0x76f7ad0e, 0x296693f4, 0x3d1fce6f, 0xc61e45be, + 0xd3b5ab34, 0xf72bf9b7, 0x1b0434c0, 0x4e72b567, 0x5592a33d, 0xb5229301, 0xcfd2a87f, 0x60aeb767, + 0x1814386b, 0x30bcc33d, 0x38a0c07d, 0xfd1606f2, 0xc363519b, 0x589dd390, 0x5479f8e6, 0x1cb8d647, + 0x97fd61a9, 0xea7759f4, 0x2d57539d, 0x569a58cf, 0xe84e63ad, 0x462e1b78, 0x6580f87e, 0xf3817914, + 0x91da55f4, 0x40a230f3, 0xd1988f35, 0xb6e318d2, 0x3ffa50bc, 0x3d40f021, 0xc3c0bdae, 0x4958c24c, + 0x518f36b2, 0x84b1d370, 0x0fedce83, 0x878ddada, 0xf2a279c7, 0x94e01be8, 0x90716f4b, 0x954b8aa3 + ]; + + sBox[7] = [ + 0xe216300d, 0xbbddfffc, 0xa7ebdabd, 0x35648095, 0x7789f8b7, 0xe6c1121b, 0x0e241600, 0x052ce8b5, + 0x11a9cfb0, 0xe5952f11, 0xece7990a, 0x9386d174, 0x2a42931c, 0x76e38111, 0xb12def3a, 0x37ddddfc, + 0xde9adeb1, 0x0a0cc32c, 0xbe197029, 0x84a00940, 0xbb243a0f, 0xb4d137cf, 0xb44e79f0, 0x049eedfd, + 0x0b15a15d, 0x480d3168, 0x8bbbde5a, 0x669ded42, 0xc7ece831, 0x3f8f95e7, 0x72df191b, 0x7580330d, + 0x94074251, 0x5c7dcdfa, 0xabbe6d63, 0xaa402164, 0xb301d40a, 0x02e7d1ca, 0x53571dae, 0x7a3182a2, + 0x12a8ddec, 0xfdaa335d, 0x176f43e8, 0x71fb46d4, 0x38129022, 0xce949ad4, 0xb84769ad, 0x965bd862, + 0x82f3d055, 0x66fb9767, 0x15b80b4e, 0x1d5b47a0, 0x4cfde06f, 0xc28ec4b8, 0x57e8726e, 0x647a78fc, + 0x99865d44, 0x608bd593, 0x6c200e03, 0x39dc5ff6, 0x5d0b00a3, 0xae63aff2, 0x7e8bd632, 0x70108c0c, + 0xbbd35049, 0x2998df04, 0x980cf42a, 0x9b6df491, 0x9e7edd53, 0x06918548, 0x58cb7e07, 0x3b74ef2e, + 0x522fffb1, 0xd24708cc, 0x1c7e27cd, 0xa4eb215b, 0x3cf1d2e2, 0x19b47a38, 0x424f7618, 0x35856039, + 0x9d17dee7, 0x27eb35e6, 0xc9aff67b, 0x36baf5b8, 0x09c467cd, 0xc18910b1, 0xe11dbf7b, 0x06cd1af8, + 0x7170c608, 0x2d5e3354, 0xd4de495a, 0x64c6d006, 0xbcc0c62c, 0x3dd00db3, 0x708f8f34, 0x77d51b42, + 0x264f620f, 0x24b8d2bf, 0x15c1b79e, 0x46a52564, 0xf8d7e54e, 0x3e378160, 0x7895cda5, 0x859c15a5, + 0xe6459788, 0xc37bc75f, 0xdb07ba0c, 0x0676a3ab, 0x7f229b1e, 0x31842e7b, 0x24259fd7, 0xf8bef472, + 0x835ffcb8, 0x6df4c1f2, 0x96f5b195, 0xfd0af0fc, 0xb0fe134c, 0xe2506d3d, 0x4f9b12ea, 0xf215f225, + 0xa223736f, 0x9fb4c428, 0x25d04979, 0x34c713f8, 0xc4618187, 0xea7a6e98, 0x7cd16efc, 0x1436876c, + 0xf1544107, 0xbedeee14, 0x56e9af27, 0xa04aa441, 0x3cf7c899, 0x92ecbae6, 0xdd67016d, 0x151682eb, + 0xa842eedf, 0xfdba60b4, 0xf1907b75, 0x20e3030f, 0x24d8c29e, 0xe139673b, 0xefa63fb8, 0x71873054, + 0xb6f2cf3b, 0x9f326442, 0xcb15a4cc, 0xb01a4504, 0xf1e47d8d, 0x844a1be5, 0xbae7dfdc, 0x42cbda70, + 0xcd7dae0a, 0x57e85b7a, 0xd53f5af6, 0x20cf4d8c, 0xcea4d428, 0x79d130a4, 0x3486ebfb, 0x33d3cddc, + 0x77853b53, 0x37effcb5, 0xc5068778, 0xe580b3e6, 0x4e68b8f4, 0xc5c8b37e, 0x0d809ea2, 0x398feb7c, + 0x132a4f94, 0x43b7950e, 0x2fee7d1c, 0x223613bd, 0xdd06caa2, 0x37df932b, 0xc4248289, 0xacf3ebc3, + 0x5715f6b7, 0xef3478dd, 0xf267616f, 0xc148cbe4, 0x9052815e, 0x5e410fab, 0xb48a2465, 0x2eda7fa4, + 0xe87b40e4, 0xe98ea084, 0x5889e9e1, 0xefd390fc, 0xdd07d35b, 0xdb485694, 0x38d7e5b2, 0x57720101, + 0x730edebc, 0x5b643113, 0x94917e4f, 0x503c2fba, 0x646f1282, 0x7523d24a, 0xe0779695, 0xf9c17a8f, + 0x7a5b2121, 0xd187b896, 0x29263a4d, 0xba510cdf, 0x81f47c9f, 0xad1163ed, 0xea7b5965, 0x1a00726e, + 0x11403092, 0x00da6d77, 0x4a0cdd61, 0xad1f4603, 0x605bdfb0, 0x9eedc364, 0x22ebe6a8, 0xcee7d28a, + 0xa0e736a0, 0x5564a6b9, 0x10853209, 0xc7eb8f37, 0x2de705ca, 0x8951570f, 0xdf09822b, 0xbd691a6c, + 0xaa12e4f2, 0x87451c0f, 0xe0f6a27a, 0x3ada4819, 0x4cf1764f, 0x0d771c2b, 0x67cdb156, 0x350d8384, + 0x5938fa0f, 0x42399ef3, 0x36997b07, 0x0e84093d, 0x4aa93e61, 0x8360d87b, 0x1fa98b0c, 0x1149382c, + 0xe97625a5, 0x0614d1b7, 0x0e25244b, 0x0c768347, 0x589e8d82, 0x0d2059d1, 0xa466bb1e, 0xf8da0a82, + 0x04f19130, 0xba6e4ec0, 0x99265164, 0x1ee7230d, 0x50b2ad80, 0xeaee6801, 0x8db2a283, 0xea8bf59e + ]; + } + + function CAST5(key) { + this.cast5 = new OpenPGPSymEncCAST5(); + this.cast5.setKey(key); + + this.encrypt = function(block) { + return this.cast5.encrypt(block); + }; + } + + CAST5.blockSize = CAST5.prototype.blockSize = 8; + CAST5.keySize = CAST5.prototype.keySize = 16; + + /* eslint-disable no-mixed-operators, no-fallthrough */ + + + /* Modified by Recurity Labs GmbH + * + * Cipher.js + * A block-cipher algorithm implementation on JavaScript + * See Cipher.readme.txt for further information. + * + * Copyright(c) 2009 Atsushi Oka [ http://oka.nu/ ] + * This script file is distributed under the LGPL + * + * ACKNOWLEDGMENT + * + * The main subroutines are written by Michiel van Everdingen. + * + * Michiel van Everdingen + * http://home.versatel.nl/MAvanEverdingen/index.html + * + * All rights for these routines are reserved to Michiel van Everdingen. + * + */ + + //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //Math + //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + const MAXINT = 0xFFFFFFFF; + + function rotw(w, n) { + return (w << n | w >>> (32 - n)) & MAXINT; + } + + function getW(a, i) { + return a[i] | a[i + 1] << 8 | a[i + 2] << 16 | a[i + 3] << 24; + } + + function setW(a, i, w) { + a.splice(i, 4, w & 0xFF, (w >>> 8) & 0xFF, (w >>> 16) & 0xFF, (w >>> 24) & 0xFF); + } + + function getB(x, n) { + return (x >>> (n * 8)) & 0xFF; + } + + // ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + // Twofish + // ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + function createTwofish() { + // + let keyBytes = null; + let dataBytes = null; + let dataOffset = -1; + // var dataLength = -1; + // var idx2 = -1; + // + + let tfsKey = []; + let tfsM = [ + [], + [], + [], + [] + ]; + + function tfsInit(key) { + keyBytes = key; + let i; + let a; + let b; + let c; + let d; + const meKey = []; + const moKey = []; + const inKey = []; + let kLen; + const sKey = []; + let f01; + let f5b; + let fef; + + const q0 = [ + [8, 1, 7, 13, 6, 15, 3, 2, 0, 11, 5, 9, 14, 12, 10, 4], + [2, 8, 11, 13, 15, 7, 6, 14, 3, 1, 9, 4, 0, 10, 12, 5] + ]; + const q1 = [ + [14, 12, 11, 8, 1, 2, 3, 5, 15, 4, 10, 6, 7, 0, 9, 13], + [1, 14, 2, 11, 4, 12, 3, 7, 6, 13, 10, 5, 15, 9, 0, 8] + ]; + const q2 = [ + [11, 10, 5, 14, 6, 13, 9, 0, 12, 8, 15, 3, 2, 4, 7, 1], + [4, 12, 7, 5, 1, 6, 9, 10, 0, 14, 13, 8, 2, 11, 3, 15] + ]; + const q3 = [ + [13, 7, 15, 4, 1, 2, 6, 14, 9, 11, 3, 0, 8, 5, 12, 10], + [11, 9, 5, 1, 12, 3, 13, 14, 6, 4, 7, 15, 2, 0, 8, 10] + ]; + const ror4 = [0, 8, 1, 9, 2, 10, 3, 11, 4, 12, 5, 13, 6, 14, 7, 15]; + const ashx = [0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 5, 14, 7]; + const q = [ + [], + [] + ]; + const m = [ + [], + [], + [], + [] + ]; + + function ffm5b(x) { + return x ^ (x >> 2) ^ [0, 90, 180, 238][x & 3]; + } + + function ffmEf(x) { + return x ^ (x >> 1) ^ (x >> 2) ^ [0, 238, 180, 90][x & 3]; + } + + function mdsRem(p, q) { + let i; + let t; + let u; + for (i = 0; i < 8; i++) { + t = q >>> 24; + q = ((q << 8) & MAXINT) | p >>> 24; + p = (p << 8) & MAXINT; + u = t << 1; + if (t & 128) { + u ^= 333; + } + q ^= t ^ (u << 16); + u ^= t >>> 1; + if (t & 1) { + u ^= 166; + } + q ^= u << 24 | u << 8; + } + return q; + } + + function qp(n, x) { + const a = x >> 4; + const b = x & 15; + const c = q0[n][a ^ b]; + const d = q1[n][ror4[b] ^ ashx[a]]; + return q3[n][ror4[d] ^ ashx[c]] << 4 | q2[n][c ^ d]; + } + + function hFun(x, key) { + let a = getB(x, 0); + let b = getB(x, 1); + let c = getB(x, 2); + let d = getB(x, 3); + switch (kLen) { + case 4: + a = q[1][a] ^ getB(key[3], 0); + b = q[0][b] ^ getB(key[3], 1); + c = q[0][c] ^ getB(key[3], 2); + d = q[1][d] ^ getB(key[3], 3); + case 3: + a = q[1][a] ^ getB(key[2], 0); + b = q[1][b] ^ getB(key[2], 1); + c = q[0][c] ^ getB(key[2], 2); + d = q[0][d] ^ getB(key[2], 3); + case 2: + a = q[0][q[0][a] ^ getB(key[1], 0)] ^ getB(key[0], 0); + b = q[0][q[1][b] ^ getB(key[1], 1)] ^ getB(key[0], 1); + c = q[1][q[0][c] ^ getB(key[1], 2)] ^ getB(key[0], 2); + d = q[1][q[1][d] ^ getB(key[1], 3)] ^ getB(key[0], 3); + } + return m[0][a] ^ m[1][b] ^ m[2][c] ^ m[3][d]; + } + + keyBytes = keyBytes.slice(0, 32); + i = keyBytes.length; + while (i !== 16 && i !== 24 && i !== 32) { + keyBytes[i++] = 0; + } + + for (i = 0; i < keyBytes.length; i += 4) { + inKey[i >> 2] = getW(keyBytes, i); + } + for (i = 0; i < 256; i++) { + q[0][i] = qp(0, i); + q[1][i] = qp(1, i); + } + for (i = 0; i < 256; i++) { + f01 = q[1][i]; + f5b = ffm5b(f01); + fef = ffmEf(f01); + m[0][i] = f01 + (f5b << 8) + (fef << 16) + (fef << 24); + m[2][i] = f5b + (fef << 8) + (f01 << 16) + (fef << 24); + f01 = q[0][i]; + f5b = ffm5b(f01); + fef = ffmEf(f01); + m[1][i] = fef + (fef << 8) + (f5b << 16) + (f01 << 24); + m[3][i] = f5b + (f01 << 8) + (fef << 16) + (f5b << 24); + } + + kLen = inKey.length / 2; + for (i = 0; i < kLen; i++) { + a = inKey[i + i]; + meKey[i] = a; + b = inKey[i + i + 1]; + moKey[i] = b; + sKey[kLen - i - 1] = mdsRem(a, b); + } + for (i = 0; i < 40; i += 2) { + a = 0x1010101 * i; + b = a + 0x1010101; + a = hFun(a, meKey); + b = rotw(hFun(b, moKey), 8); + tfsKey[i] = (a + b) & MAXINT; + tfsKey[i + 1] = rotw(a + 2 * b, 9); + } + for (i = 0; i < 256; i++) { + a = b = c = d = i; + switch (kLen) { + case 4: + a = q[1][a] ^ getB(sKey[3], 0); + b = q[0][b] ^ getB(sKey[3], 1); + c = q[0][c] ^ getB(sKey[3], 2); + d = q[1][d] ^ getB(sKey[3], 3); + case 3: + a = q[1][a] ^ getB(sKey[2], 0); + b = q[1][b] ^ getB(sKey[2], 1); + c = q[0][c] ^ getB(sKey[2], 2); + d = q[0][d] ^ getB(sKey[2], 3); + case 2: + tfsM[0][i] = m[0][q[0][q[0][a] ^ getB(sKey[1], 0)] ^ getB(sKey[0], 0)]; + tfsM[1][i] = m[1][q[0][q[1][b] ^ getB(sKey[1], 1)] ^ getB(sKey[0], 1)]; + tfsM[2][i] = m[2][q[1][q[0][c] ^ getB(sKey[1], 2)] ^ getB(sKey[0], 2)]; + tfsM[3][i] = m[3][q[1][q[1][d] ^ getB(sKey[1], 3)] ^ getB(sKey[0], 3)]; + } + } + } + + function tfsG0(x) { + return tfsM[0][getB(x, 0)] ^ tfsM[1][getB(x, 1)] ^ tfsM[2][getB(x, 2)] ^ tfsM[3][getB(x, 3)]; + } + + function tfsG1(x) { + return tfsM[0][getB(x, 3)] ^ tfsM[1][getB(x, 0)] ^ tfsM[2][getB(x, 1)] ^ tfsM[3][getB(x, 2)]; + } + + function tfsFrnd(r, blk) { + let a = tfsG0(blk[0]); + let b = tfsG1(blk[1]); + blk[2] = rotw(blk[2] ^ (a + b + tfsKey[4 * r + 8]) & MAXINT, 31); + blk[3] = rotw(blk[3], 1) ^ (a + 2 * b + tfsKey[4 * r + 9]) & MAXINT; + a = tfsG0(blk[2]); + b = tfsG1(blk[3]); + blk[0] = rotw(blk[0] ^ (a + b + tfsKey[4 * r + 10]) & MAXINT, 31); + blk[1] = rotw(blk[1], 1) ^ (a + 2 * b + tfsKey[4 * r + 11]) & MAXINT; + } + + function tfsIrnd(i, blk) { + let a = tfsG0(blk[0]); + let b = tfsG1(blk[1]); + blk[2] = rotw(blk[2], 1) ^ (a + b + tfsKey[4 * i + 10]) & MAXINT; + blk[3] = rotw(blk[3] ^ (a + 2 * b + tfsKey[4 * i + 11]) & MAXINT, 31); + a = tfsG0(blk[2]); + b = tfsG1(blk[3]); + blk[0] = rotw(blk[0], 1) ^ (a + b + tfsKey[4 * i + 8]) & MAXINT; + blk[1] = rotw(blk[1] ^ (a + 2 * b + tfsKey[4 * i + 9]) & MAXINT, 31); + } + + function tfsClose() { + tfsKey = []; + tfsM = [ + [], + [], + [], + [] + ]; + } + + function tfsEncrypt(data, offset) { + dataBytes = data; + dataOffset = offset; + const blk = [getW(dataBytes, dataOffset) ^ tfsKey[0], + getW(dataBytes, dataOffset + 4) ^ tfsKey[1], + getW(dataBytes, dataOffset + 8) ^ tfsKey[2], + getW(dataBytes, dataOffset + 12) ^ tfsKey[3]]; + for (let j = 0; j < 8; j++) { + tfsFrnd(j, blk); + } + setW(dataBytes, dataOffset, blk[2] ^ tfsKey[4]); + setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[5]); + setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[6]); + setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[7]); + dataOffset += 16; + return dataBytes; + } + + function tfsDecrypt(data, offset) { + dataBytes = data; + dataOffset = offset; + const blk = [getW(dataBytes, dataOffset) ^ tfsKey[4], + getW(dataBytes, dataOffset + 4) ^ tfsKey[5], + getW(dataBytes, dataOffset + 8) ^ tfsKey[6], + getW(dataBytes, dataOffset + 12) ^ tfsKey[7]]; + for (let j = 7; j >= 0; j--) { + tfsIrnd(j, blk); + } + setW(dataBytes, dataOffset, blk[2] ^ tfsKey[0]); + setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[1]); + setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[2]); + setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[3]); + dataOffset += 16; + } + + // added by Recurity Labs + + function tfsFinal() { + return dataBytes; + } + + return { + name: 'twofish', + blocksize: 128 / 8, + open: tfsInit, + close: tfsClose, + encrypt: tfsEncrypt, + decrypt: tfsDecrypt, + // added by Recurity Labs + finalize: tfsFinal + }; + } + + // added by Recurity Labs + + function TF(key) { + this.tf = createTwofish(); + this.tf.open(Array.from(key), 0); + + this.encrypt = function(block) { + return this.tf.encrypt(Array.from(block), 0); + }; + } + + TF.keySize = TF.prototype.keySize = 32; + TF.blockSize = TF.prototype.blockSize = 16; + + /* Modified by Recurity Labs GmbH + * + * Originally written by nklein software (nklein.com) + */ + + /* + * Javascript implementation based on Bruce Schneier's reference implementation. + * + * + * The constructor doesn't do much of anything. It's just here + * so we can start defining properties and methods and such. + */ + function Blowfish() {} + + /* + * Declare the block size so that protocols know what size + * Initialization Vector (IV) they will need. + */ + Blowfish.prototype.BLOCKSIZE = 8; + + /* + * These are the default SBOXES. + */ + Blowfish.prototype.SBOXES = [ + [ + 0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96, + 0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x0801f2e2, 0x858efc16, + 0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0x0d95748f, 0x728eb658, + 0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013, + 0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e, + 0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60, + 0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6, + 0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a, + 0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c, + 0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193, + 0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1, + 0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0x0f6d6ff3, 0x83f44239, + 0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a, + 0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3, + 0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176, + 0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe, + 0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706, + 0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b, + 0x075372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b, + 0x976ce0bd, 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463, + 0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c, + 0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3, + 0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a, + 0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8, + 0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760, + 0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db, + 0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8, + 0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b, + 0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33, + 0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4, + 0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0, + 0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c, + 0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777, + 0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299, + 0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705, + 0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf, + 0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x00250e2d, 0x2071b35e, + 0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa, + 0x78c14389, 0xd95a537f, 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9, + 0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915, + 0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5, 0x571be91f, + 0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664, + 0x53b02d5d, 0xa99f8fa1, 0x08ba4799, 0x6e85076a + ], + [ + 0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d, + 0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1, + 0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65, + 0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1, + 0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f, 0x3ebaefc9, + 0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737, + 0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d, + 0xf01c1f04, 0x0200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd, + 0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc, + 0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e, 0xa4751e41, + 0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908, + 0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af, + 0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124, + 0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c, + 0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908, + 0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd, + 0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x043556f1, 0xd7a3c76b, + 0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e, + 0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa, + 0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a, + 0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d, + 0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66, + 0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5, + 0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84, + 0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96, + 0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14, + 0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca, + 0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7, + 0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77, + 0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99, + 0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054, + 0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73, + 0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea, + 0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105, + 0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646, + 0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285, + 0x095bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0x0c55f5ea, + 0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb, + 0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e, + 0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc, + 0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9, 0xdb73dbd3, 0x105588cd, + 0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20, + 0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7 + ], + [ + 0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7, + 0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af, + 0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af, + 0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0, 0x31cb8504, + 0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4, + 0x0a2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee, + 0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec, + 0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b, + 0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332, + 0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527, + 0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58, + 0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c, + 0xfdf8e802, 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22, + 0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17, + 0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60, + 0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2, 0x02e1329e, 0xaf664fd1, 0xcad18115, + 0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99, + 0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0, + 0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341, 0x992eff74, + 0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d, + 0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3, + 0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3, + 0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979, + 0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c, + 0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa, + 0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0x0a121386, 0xd90cec6e, 0xd5abea2a, + 0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086, + 0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc, + 0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24, + 0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2, + 0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84, + 0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c, + 0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09, + 0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025, 0x1d6efe10, + 0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe, + 0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0x0de6d027, + 0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0, + 0x006058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634, + 0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188, + 0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc, + 0xed545578, 0x08fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8, + 0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837, + 0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0 + ], + [ + 0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742, + 0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b, + 0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79, + 0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6, + 0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a, + 0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4, + 0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1, + 0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59, + 0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797, + 0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a, 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28, + 0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6, + 0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28, + 0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba, + 0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a, + 0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5, + 0x03563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f, + 0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce, + 0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680, + 0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166, 0xb39a460a, 0x6445c0dd, + 0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb, + 0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb, + 0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370, + 0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc, + 0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d, 0x06b89fb4, 0xce6ea048, + 0x6f3f3b82, 0x3520ab82, 0x011a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc, + 0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9, + 0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a, + 0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f, + 0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a, + 0x0f91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1, + 0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b, + 0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e, + 0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e, + 0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f, + 0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623, + 0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc, + 0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a, + 0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6, + 0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3, + 0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060, + 0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c, + 0x01c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f, + 0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6 + ] + ]; + + //* + //* This is the default PARRAY + //* + Blowfish.prototype.PARRAY = [ + 0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822, 0x299f31d0, + 0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c, + 0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b + ]; + + //* + //* This is the number of rounds the cipher will go + //* + Blowfish.prototype.NN = 16; + + //* + //* This function is needed to get rid of problems + //* with the high-bit getting set. If we don't do + //* this, then sometimes ( aa & 0x00FFFFFFFF ) is not + //* equal to ( bb & 0x00FFFFFFFF ) even when they + //* agree bit-for-bit for the first 32 bits. + //* + Blowfish.prototype._clean = function(xx) { + if (xx < 0) { + const yy = xx & 0x7FFFFFFF; + xx = yy + 0x80000000; + } + return xx; + }; + + //* + //* This is the mixing function that uses the sboxes + //* + Blowfish.prototype._F = function(xx) { + let yy; + + const dd = xx & 0x00FF; + xx >>>= 8; + const cc = xx & 0x00FF; + xx >>>= 8; + const bb = xx & 0x00FF; + xx >>>= 8; + const aa = xx & 0x00FF; + + yy = this.sboxes[0][aa] + this.sboxes[1][bb]; + yy ^= this.sboxes[2][cc]; + yy += this.sboxes[3][dd]; + + return yy; + }; + + //* + //* This method takes an array with two values, left and right + //* and does NN rounds of Blowfish on them. + //* + Blowfish.prototype._encryptBlock = function(vals) { + let dataL = vals[0]; + let dataR = vals[1]; + + let ii; + + for (ii = 0; ii < this.NN; ++ii) { + dataL ^= this.parray[ii]; + dataR = this._F(dataL) ^ dataR; + + const tmp = dataL; + dataL = dataR; + dataR = tmp; + } + + dataL ^= this.parray[this.NN + 0]; + dataR ^= this.parray[this.NN + 1]; + + vals[0] = this._clean(dataR); + vals[1] = this._clean(dataL); + }; + + //* + //* This method takes a vector of numbers and turns them + //* into long words so that they can be processed by the + //* real algorithm. + //* + //* Maybe I should make the real algorithm above take a vector + //* instead. That will involve more looping, but it won't require + //* the F() method to deconstruct the vector. + //* + Blowfish.prototype.encryptBlock = function(vector) { + let ii; + const vals = [0, 0]; + const off = this.BLOCKSIZE / 2; + for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) { + vals[0] = (vals[0] << 8) | (vector[ii + 0] & 0x00FF); + vals[1] = (vals[1] << 8) | (vector[ii + off] & 0x00FF); + } + + this._encryptBlock(vals); + + const ret = []; + for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) { + ret[ii + 0] = ((vals[0] >>> (24 - 8 * (ii))) & 0x00FF); + ret[ii + off] = ((vals[1] >>> (24 - 8 * (ii))) & 0x00FF); + // vals[ 0 ] = ( vals[ 0 ] >>> 8 ); + // vals[ 1 ] = ( vals[ 1 ] >>> 8 ); + } + + return ret; + }; + + //* + //* This method takes an array with two values, left and right + //* and undoes NN rounds of Blowfish on them. + //* + Blowfish.prototype._decryptBlock = function(vals) { + let dataL = vals[0]; + let dataR = vals[1]; + + let ii; + + for (ii = this.NN + 1; ii > 1; --ii) { + dataL ^= this.parray[ii]; + dataR = this._F(dataL) ^ dataR; + + const tmp = dataL; + dataL = dataR; + dataR = tmp; + } + + dataL ^= this.parray[1]; + dataR ^= this.parray[0]; + + vals[0] = this._clean(dataR); + vals[1] = this._clean(dataL); + }; + + //* + //* This method takes a key array and initializes the + //* sboxes and parray for this encryption. + //* + Blowfish.prototype.init = function(key) { + let ii; + let jj = 0; + + this.parray = []; + for (ii = 0; ii < this.NN + 2; ++ii) { + let data = 0x00000000; + for (let kk = 0; kk < 4; ++kk) { + data = (data << 8) | (key[jj] & 0x00FF); + if (++jj >= key.length) { + jj = 0; + } + } + this.parray[ii] = this.PARRAY[ii] ^ data; + } + + this.sboxes = []; + for (ii = 0; ii < 4; ++ii) { + this.sboxes[ii] = []; + for (jj = 0; jj < 256; ++jj) { + this.sboxes[ii][jj] = this.SBOXES[ii][jj]; + } + } + + const vals = [0x00000000, 0x00000000]; + + for (ii = 0; ii < this.NN + 2; ii += 2) { + this._encryptBlock(vals); + this.parray[ii + 0] = vals[0]; + this.parray[ii + 1] = vals[1]; + } + + for (ii = 0; ii < 4; ++ii) { + for (jj = 0; jj < 256; jj += 2) { + this._encryptBlock(vals); + this.sboxes[ii][jj + 0] = vals[0]; + this.sboxes[ii][jj + 1] = vals[1]; + } + } + }; + + // added by Recurity Labs + function BF(key) { + this.bf = new Blowfish(); + this.bf.init(key); + + this.encrypt = function(block) { + return this.bf.encryptBlock(block); + }; + } + + BF.keySize = BF.prototype.keySize = 16; + BF.blockSize = BF.prototype.blockSize = 8; + + /** + * @fileoverview Symmetric cryptography functions + * @module crypto/cipher + * @private + */ + + /** + * AES-128 encryption and decryption (ID 7) + * @function + * @param {String} key - 128-bit key + * @see {@link https://github.com/asmcrypto/asmcrypto.js|asmCrypto} + * @see {@link https://csrc.nist.gov/publications/fips/fips197/fips-197.pdf|NIST FIPS-197} + * @returns {Object} + */ + const aes128 = aes(128); + /** + * AES-128 Block Cipher (ID 8) + * @function + * @param {String} key - 192-bit key + * @see {@link https://github.com/asmcrypto/asmcrypto.js|asmCrypto} + * @see {@link https://csrc.nist.gov/publications/fips/fips197/fips-197.pdf|NIST FIPS-197} + * @returns {Object} + */ + const aes192 = aes(192); + /** + * AES-128 Block Cipher (ID 9) + * @function + * @param {String} key - 256-bit key + * @see {@link https://github.com/asmcrypto/asmcrypto.js|asmCrypto} + * @see {@link https://csrc.nist.gov/publications/fips/fips197/fips-197.pdf|NIST FIPS-197} + * @returns {Object} + */ + const aes256 = aes(256); + // Not in OpenPGP specifications + const des$1 = DES; + /** + * Triple DES Block Cipher (ID 2) + * @function + * @param {String} key - 192-bit key + * @see {@link https://nvlpubs.nist.gov/nistpubs/SpecialPublications/NIST.SP.800-67r2.pdf|NIST SP 800-67} + * @returns {Object} + */ + const tripledes = TripleDES; + /** + * CAST-128 Block Cipher (ID 3) + * @function + * @param {String} key - 128-bit key + * @see {@link https://tools.ietf.org/html/rfc2144|The CAST-128 Encryption Algorithm} + * @returns {Object} + */ + const cast5 = CAST5; + /** + * Twofish Block Cipher (ID 10) + * @function + * @param {String} key - 256-bit key + * @see {@link https://tools.ietf.org/html/rfc4880#ref-TWOFISH|TWOFISH} + * @returns {Object} + */ + const twofish = TF; + /** + * Blowfish Block Cipher (ID 4) + * @function + * @param {String} key - 128-bit key + * @see {@link https://tools.ietf.org/html/rfc4880#ref-BLOWFISH|BLOWFISH} + * @returns {Object} + */ + const blowfish = BF; + /** + * Not implemented + * @function + * @throws {Error} + */ + const idea = function() { + throw new Error('IDEA symmetric-key algorithm not implemented'); + }; + + var cipher = /*#__PURE__*/Object.freeze({ + __proto__: null, + aes128: aes128, + aes192: aes192, + aes256: aes256, + des: des$1, + tripledes: tripledes, + cast5: cast5, + twofish: twofish, + blowfish: blowfish, + idea: idea + }); + + var sha1_asm = function ( stdlib, foreign, buffer ) { + "use asm"; + + // SHA256 state + var H0 = 0, H1 = 0, H2 = 0, H3 = 0, H4 = 0, + TOTAL0 = 0, TOTAL1 = 0; + + // HMAC state + var I0 = 0, I1 = 0, I2 = 0, I3 = 0, I4 = 0, + O0 = 0, O1 = 0, O2 = 0, O3 = 0, O4 = 0; + + // I/O buffer + var HEAP = new stdlib.Uint8Array(buffer); + + function _core ( w0, w1, w2, w3, w4, w5, w6, w7, w8, w9, w10, w11, w12, w13, w14, w15 ) { + w0 = w0|0; + w1 = w1|0; + w2 = w2|0; + w3 = w3|0; + w4 = w4|0; + w5 = w5|0; + w6 = w6|0; + w7 = w7|0; + w8 = w8|0; + w9 = w9|0; + w10 = w10|0; + w11 = w11|0; + w12 = w12|0; + w13 = w13|0; + w14 = w14|0; + w15 = w15|0; + + var a = 0, b = 0, c = 0, d = 0, e = 0, n = 0, t = 0, + w16 = 0, w17 = 0, w18 = 0, w19 = 0, + w20 = 0, w21 = 0, w22 = 0, w23 = 0, w24 = 0, w25 = 0, w26 = 0, w27 = 0, w28 = 0, w29 = 0, + w30 = 0, w31 = 0, w32 = 0, w33 = 0, w34 = 0, w35 = 0, w36 = 0, w37 = 0, w38 = 0, w39 = 0, + w40 = 0, w41 = 0, w42 = 0, w43 = 0, w44 = 0, w45 = 0, w46 = 0, w47 = 0, w48 = 0, w49 = 0, + w50 = 0, w51 = 0, w52 = 0, w53 = 0, w54 = 0, w55 = 0, w56 = 0, w57 = 0, w58 = 0, w59 = 0, + w60 = 0, w61 = 0, w62 = 0, w63 = 0, w64 = 0, w65 = 0, w66 = 0, w67 = 0, w68 = 0, w69 = 0, + w70 = 0, w71 = 0, w72 = 0, w73 = 0, w74 = 0, w75 = 0, w76 = 0, w77 = 0, w78 = 0, w79 = 0; + + a = H0; + b = H1; + c = H2; + d = H3; + e = H4; + + // 0 + t = ( w0 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 1 + t = ( w1 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 2 + t = ( w2 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 3 + t = ( w3 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 4 + t = ( w4 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 5 + t = ( w5 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 6 + t = ( w6 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 7 + t = ( w7 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 8 + t = ( w8 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 9 + t = ( w9 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 10 + t = ( w10 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 11 + t = ( w11 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 12 + t = ( w12 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 13 + t = ( w13 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 14 + t = ( w14 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 15 + t = ( w15 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 16 + n = w13 ^ w8 ^ w2 ^ w0; + w16 = (n << 1) | (n >>> 31); + t = (w16 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 17 + n = w14 ^ w9 ^ w3 ^ w1; + w17 = (n << 1) | (n >>> 31); + t = (w17 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 18 + n = w15 ^ w10 ^ w4 ^ w2; + w18 = (n << 1) | (n >>> 31); + t = (w18 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 19 + n = w16 ^ w11 ^ w5 ^ w3; + w19 = (n << 1) | (n >>> 31); + t = (w19 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (~b & d)) + 0x5a827999 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 20 + n = w17 ^ w12 ^ w6 ^ w4; + w20 = (n << 1) | (n >>> 31); + t = (w20 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 21 + n = w18 ^ w13 ^ w7 ^ w5; + w21 = (n << 1) | (n >>> 31); + t = (w21 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 22 + n = w19 ^ w14 ^ w8 ^ w6; + w22 = (n << 1) | (n >>> 31); + t = (w22 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 23 + n = w20 ^ w15 ^ w9 ^ w7; + w23 = (n << 1) | (n >>> 31); + t = (w23 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 24 + n = w21 ^ w16 ^ w10 ^ w8; + w24 = (n << 1) | (n >>> 31); + t = (w24 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 25 + n = w22 ^ w17 ^ w11 ^ w9; + w25 = (n << 1) | (n >>> 31); + t = (w25 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 26 + n = w23 ^ w18 ^ w12 ^ w10; + w26 = (n << 1) | (n >>> 31); + t = (w26 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 27 + n = w24 ^ w19 ^ w13 ^ w11; + w27 = (n << 1) | (n >>> 31); + t = (w27 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 28 + n = w25 ^ w20 ^ w14 ^ w12; + w28 = (n << 1) | (n >>> 31); + t = (w28 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 29 + n = w26 ^ w21 ^ w15 ^ w13; + w29 = (n << 1) | (n >>> 31); + t = (w29 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 30 + n = w27 ^ w22 ^ w16 ^ w14; + w30 = (n << 1) | (n >>> 31); + t = (w30 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 31 + n = w28 ^ w23 ^ w17 ^ w15; + w31 = (n << 1) | (n >>> 31); + t = (w31 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 32 + n = w29 ^ w24 ^ w18 ^ w16; + w32 = (n << 1) | (n >>> 31); + t = (w32 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 33 + n = w30 ^ w25 ^ w19 ^ w17; + w33 = (n << 1) | (n >>> 31); + t = (w33 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 34 + n = w31 ^ w26 ^ w20 ^ w18; + w34 = (n << 1) | (n >>> 31); + t = (w34 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 35 + n = w32 ^ w27 ^ w21 ^ w19; + w35 = (n << 1) | (n >>> 31); + t = (w35 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 36 + n = w33 ^ w28 ^ w22 ^ w20; + w36 = (n << 1) | (n >>> 31); + t = (w36 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 37 + n = w34 ^ w29 ^ w23 ^ w21; + w37 = (n << 1) | (n >>> 31); + t = (w37 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 38 + n = w35 ^ w30 ^ w24 ^ w22; + w38 = (n << 1) | (n >>> 31); + t = (w38 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 39 + n = w36 ^ w31 ^ w25 ^ w23; + w39 = (n << 1) | (n >>> 31); + t = (w39 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) + 0x6ed9eba1 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 40 + n = w37 ^ w32 ^ w26 ^ w24; + w40 = (n << 1) | (n >>> 31); + t = (w40 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 41 + n = w38 ^ w33 ^ w27 ^ w25; + w41 = (n << 1) | (n >>> 31); + t = (w41 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 42 + n = w39 ^ w34 ^ w28 ^ w26; + w42 = (n << 1) | (n >>> 31); + t = (w42 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 43 + n = w40 ^ w35 ^ w29 ^ w27; + w43 = (n << 1) | (n >>> 31); + t = (w43 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 44 + n = w41 ^ w36 ^ w30 ^ w28; + w44 = (n << 1) | (n >>> 31); + t = (w44 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 45 + n = w42 ^ w37 ^ w31 ^ w29; + w45 = (n << 1) | (n >>> 31); + t = (w45 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 46 + n = w43 ^ w38 ^ w32 ^ w30; + w46 = (n << 1) | (n >>> 31); + t = (w46 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 47 + n = w44 ^ w39 ^ w33 ^ w31; + w47 = (n << 1) | (n >>> 31); + t = (w47 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 48 + n = w45 ^ w40 ^ w34 ^ w32; + w48 = (n << 1) | (n >>> 31); + t = (w48 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 49 + n = w46 ^ w41 ^ w35 ^ w33; + w49 = (n << 1) | (n >>> 31); + t = (w49 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 50 + n = w47 ^ w42 ^ w36 ^ w34; + w50 = (n << 1) | (n >>> 31); + t = (w50 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 51 + n = w48 ^ w43 ^ w37 ^ w35; + w51 = (n << 1) | (n >>> 31); + t = (w51 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 52 + n = w49 ^ w44 ^ w38 ^ w36; + w52 = (n << 1) | (n >>> 31); + t = (w52 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 53 + n = w50 ^ w45 ^ w39 ^ w37; + w53 = (n << 1) | (n >>> 31); + t = (w53 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 54 + n = w51 ^ w46 ^ w40 ^ w38; + w54 = (n << 1) | (n >>> 31); + t = (w54 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 55 + n = w52 ^ w47 ^ w41 ^ w39; + w55 = (n << 1) | (n >>> 31); + t = (w55 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 56 + n = w53 ^ w48 ^ w42 ^ w40; + w56 = (n << 1) | (n >>> 31); + t = (w56 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 57 + n = w54 ^ w49 ^ w43 ^ w41; + w57 = (n << 1) | (n >>> 31); + t = (w57 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 58 + n = w55 ^ w50 ^ w44 ^ w42; + w58 = (n << 1) | (n >>> 31); + t = (w58 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 59 + n = w56 ^ w51 ^ w45 ^ w43; + w59 = (n << 1) | (n >>> 31); + t = (w59 + ((a << 5) | (a >>> 27)) + e + ((b & c) | (b & d) | (c & d)) - 0x70e44324 )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 60 + n = w57 ^ w52 ^ w46 ^ w44; + w60 = (n << 1) | (n >>> 31); + t = (w60 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 61 + n = w58 ^ w53 ^ w47 ^ w45; + w61 = (n << 1) | (n >>> 31); + t = (w61 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 62 + n = w59 ^ w54 ^ w48 ^ w46; + w62 = (n << 1) | (n >>> 31); + t = (w62 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 63 + n = w60 ^ w55 ^ w49 ^ w47; + w63 = (n << 1) | (n >>> 31); + t = (w63 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 64 + n = w61 ^ w56 ^ w50 ^ w48; + w64 = (n << 1) | (n >>> 31); + t = (w64 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 65 + n = w62 ^ w57 ^ w51 ^ w49; + w65 = (n << 1) | (n >>> 31); + t = (w65 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 66 + n = w63 ^ w58 ^ w52 ^ w50; + w66 = (n << 1) | (n >>> 31); + t = (w66 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 67 + n = w64 ^ w59 ^ w53 ^ w51; + w67 = (n << 1) | (n >>> 31); + t = (w67 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 68 + n = w65 ^ w60 ^ w54 ^ w52; + w68 = (n << 1) | (n >>> 31); + t = (w68 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 69 + n = w66 ^ w61 ^ w55 ^ w53; + w69 = (n << 1) | (n >>> 31); + t = (w69 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 70 + n = w67 ^ w62 ^ w56 ^ w54; + w70 = (n << 1) | (n >>> 31); + t = (w70 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 71 + n = w68 ^ w63 ^ w57 ^ w55; + w71 = (n << 1) | (n >>> 31); + t = (w71 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 72 + n = w69 ^ w64 ^ w58 ^ w56; + w72 = (n << 1) | (n >>> 31); + t = (w72 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 73 + n = w70 ^ w65 ^ w59 ^ w57; + w73 = (n << 1) | (n >>> 31); + t = (w73 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 74 + n = w71 ^ w66 ^ w60 ^ w58; + w74 = (n << 1) | (n >>> 31); + t = (w74 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 75 + n = w72 ^ w67 ^ w61 ^ w59; + w75 = (n << 1) | (n >>> 31); + t = (w75 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 76 + n = w73 ^ w68 ^ w62 ^ w60; + w76 = (n << 1) | (n >>> 31); + t = (w76 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 77 + n = w74 ^ w69 ^ w63 ^ w61; + w77 = (n << 1) | (n >>> 31); + t = (w77 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 78 + n = w75 ^ w70 ^ w64 ^ w62; + w78 = (n << 1) | (n >>> 31); + t = (w78 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + // 79 + n = w76 ^ w71 ^ w65 ^ w63; + w79 = (n << 1) | (n >>> 31); + t = (w79 + ((a << 5) | (a >>> 27)) + e + (b ^ c ^ d) - 0x359d3e2a )|0; + e = d; d = c; c = (b << 30) | (b >>> 2); b = a; a = t; + + H0 = ( H0 + a )|0; + H1 = ( H1 + b )|0; + H2 = ( H2 + c )|0; + H3 = ( H3 + d )|0; + H4 = ( H4 + e )|0; + + } + + function _core_heap ( offset ) { + offset = offset|0; + + _core( + HEAP[offset|0]<<24 | HEAP[offset|1]<<16 | HEAP[offset|2]<<8 | HEAP[offset|3], + HEAP[offset|4]<<24 | HEAP[offset|5]<<16 | HEAP[offset|6]<<8 | HEAP[offset|7], + HEAP[offset|8]<<24 | HEAP[offset|9]<<16 | HEAP[offset|10]<<8 | HEAP[offset|11], + HEAP[offset|12]<<24 | HEAP[offset|13]<<16 | HEAP[offset|14]<<8 | HEAP[offset|15], + HEAP[offset|16]<<24 | HEAP[offset|17]<<16 | HEAP[offset|18]<<8 | HEAP[offset|19], + HEAP[offset|20]<<24 | HEAP[offset|21]<<16 | HEAP[offset|22]<<8 | HEAP[offset|23], + HEAP[offset|24]<<24 | HEAP[offset|25]<<16 | HEAP[offset|26]<<8 | HEAP[offset|27], + HEAP[offset|28]<<24 | HEAP[offset|29]<<16 | HEAP[offset|30]<<8 | HEAP[offset|31], + HEAP[offset|32]<<24 | HEAP[offset|33]<<16 | HEAP[offset|34]<<8 | HEAP[offset|35], + HEAP[offset|36]<<24 | HEAP[offset|37]<<16 | HEAP[offset|38]<<8 | HEAP[offset|39], + HEAP[offset|40]<<24 | HEAP[offset|41]<<16 | HEAP[offset|42]<<8 | HEAP[offset|43], + HEAP[offset|44]<<24 | HEAP[offset|45]<<16 | HEAP[offset|46]<<8 | HEAP[offset|47], + HEAP[offset|48]<<24 | HEAP[offset|49]<<16 | HEAP[offset|50]<<8 | HEAP[offset|51], + HEAP[offset|52]<<24 | HEAP[offset|53]<<16 | HEAP[offset|54]<<8 | HEAP[offset|55], + HEAP[offset|56]<<24 | HEAP[offset|57]<<16 | HEAP[offset|58]<<8 | HEAP[offset|59], + HEAP[offset|60]<<24 | HEAP[offset|61]<<16 | HEAP[offset|62]<<8 | HEAP[offset|63] + ); + } + + // offset — multiple of 32 + function _state_to_heap ( output ) { + output = output|0; + + HEAP[output|0] = H0>>>24; + HEAP[output|1] = H0>>>16&255; + HEAP[output|2] = H0>>>8&255; + HEAP[output|3] = H0&255; + HEAP[output|4] = H1>>>24; + HEAP[output|5] = H1>>>16&255; + HEAP[output|6] = H1>>>8&255; + HEAP[output|7] = H1&255; + HEAP[output|8] = H2>>>24; + HEAP[output|9] = H2>>>16&255; + HEAP[output|10] = H2>>>8&255; + HEAP[output|11] = H2&255; + HEAP[output|12] = H3>>>24; + HEAP[output|13] = H3>>>16&255; + HEAP[output|14] = H3>>>8&255; + HEAP[output|15] = H3&255; + HEAP[output|16] = H4>>>24; + HEAP[output|17] = H4>>>16&255; + HEAP[output|18] = H4>>>8&255; + HEAP[output|19] = H4&255; + } + + function reset () { + H0 = 0x67452301; + H1 = 0xefcdab89; + H2 = 0x98badcfe; + H3 = 0x10325476; + H4 = 0xc3d2e1f0; + TOTAL0 = TOTAL1 = 0; + } + + function init ( h0, h1, h2, h3, h4, total0, total1 ) { + h0 = h0|0; + h1 = h1|0; + h2 = h2|0; + h3 = h3|0; + h4 = h4|0; + total0 = total0|0; + total1 = total1|0; + + H0 = h0; + H1 = h1; + H2 = h2; + H3 = h3; + H4 = h4; + TOTAL0 = total0; + TOTAL1 = total1; + } + + // offset — multiple of 64 + function process ( offset, length ) { + offset = offset|0; + length = length|0; + + var hashed = 0; + + if ( offset & 63 ) + return -1; + + while ( (length|0) >= 64 ) { + _core_heap(offset); + + offset = ( offset + 64 )|0; + length = ( length - 64 )|0; + + hashed = ( hashed + 64 )|0; + } + + TOTAL0 = ( TOTAL0 + hashed )|0; + if ( TOTAL0>>>0 < hashed>>>0 ) TOTAL1 = ( TOTAL1 + 1 )|0; + + return hashed|0; + } + + // offset — multiple of 64 + // output — multiple of 32 + function finish ( offset, length, output ) { + offset = offset|0; + length = length|0; + output = output|0; + + var hashed = 0, + i = 0; + + if ( offset & 63 ) + return -1; + + if ( ~output ) + if ( output & 31 ) + return -1; + + if ( (length|0) >= 64 ) { + hashed = process( offset, length )|0; + if ( (hashed|0) == -1 ) + return -1; + + offset = ( offset + hashed )|0; + length = ( length - hashed )|0; + } + + hashed = ( hashed + length )|0; + TOTAL0 = ( TOTAL0 + length )|0; + if ( TOTAL0>>>0 < length>>>0 ) TOTAL1 = (TOTAL1 + 1)|0; + + HEAP[offset|length] = 0x80; + + if ( (length|0) >= 56 ) { + for ( i = (length+1)|0; (i|0) < 64; i = (i+1)|0 ) + HEAP[offset|i] = 0x00; + _core_heap(offset); + + length = 0; + + HEAP[offset|0] = 0; + } + + for ( i = (length+1)|0; (i|0) < 59; i = (i+1)|0 ) + HEAP[offset|i] = 0; + + HEAP[offset|56] = TOTAL1>>>21&255; + HEAP[offset|57] = TOTAL1>>>13&255; + HEAP[offset|58] = TOTAL1>>>5&255; + HEAP[offset|59] = TOTAL1<<3&255 | TOTAL0>>>29; + HEAP[offset|60] = TOTAL0>>>21&255; + HEAP[offset|61] = TOTAL0>>>13&255; + HEAP[offset|62] = TOTAL0>>>5&255; + HEAP[offset|63] = TOTAL0<<3&255; + _core_heap(offset); + + if ( ~output ) + _state_to_heap(output); + + return hashed|0; + } + + function hmac_reset () { + H0 = I0; + H1 = I1; + H2 = I2; + H3 = I3; + H4 = I4; + TOTAL0 = 64; + TOTAL1 = 0; + } + + function _hmac_opad () { + H0 = O0; + H1 = O1; + H2 = O2; + H3 = O3; + H4 = O4; + TOTAL0 = 64; + TOTAL1 = 0; + } + + function hmac_init ( p0, p1, p2, p3, p4, p5, p6, p7, p8, p9, p10, p11, p12, p13, p14, p15 ) { + p0 = p0|0; + p1 = p1|0; + p2 = p2|0; + p3 = p3|0; + p4 = p4|0; + p5 = p5|0; + p6 = p6|0; + p7 = p7|0; + p8 = p8|0; + p9 = p9|0; + p10 = p10|0; + p11 = p11|0; + p12 = p12|0; + p13 = p13|0; + p14 = p14|0; + p15 = p15|0; + + // opad + reset(); + _core( + p0 ^ 0x5c5c5c5c, + p1 ^ 0x5c5c5c5c, + p2 ^ 0x5c5c5c5c, + p3 ^ 0x5c5c5c5c, + p4 ^ 0x5c5c5c5c, + p5 ^ 0x5c5c5c5c, + p6 ^ 0x5c5c5c5c, + p7 ^ 0x5c5c5c5c, + p8 ^ 0x5c5c5c5c, + p9 ^ 0x5c5c5c5c, + p10 ^ 0x5c5c5c5c, + p11 ^ 0x5c5c5c5c, + p12 ^ 0x5c5c5c5c, + p13 ^ 0x5c5c5c5c, + p14 ^ 0x5c5c5c5c, + p15 ^ 0x5c5c5c5c + ); + O0 = H0; + O1 = H1; + O2 = H2; + O3 = H3; + O4 = H4; + + // ipad + reset(); + _core( + p0 ^ 0x36363636, + p1 ^ 0x36363636, + p2 ^ 0x36363636, + p3 ^ 0x36363636, + p4 ^ 0x36363636, + p5 ^ 0x36363636, + p6 ^ 0x36363636, + p7 ^ 0x36363636, + p8 ^ 0x36363636, + p9 ^ 0x36363636, + p10 ^ 0x36363636, + p11 ^ 0x36363636, + p12 ^ 0x36363636, + p13 ^ 0x36363636, + p14 ^ 0x36363636, + p15 ^ 0x36363636 + ); + I0 = H0; + I1 = H1; + I2 = H2; + I3 = H3; + I4 = H4; + + TOTAL0 = 64; + TOTAL1 = 0; + } + + // offset — multiple of 64 + // output — multiple of 32 + function hmac_finish ( offset, length, output ) { + offset = offset|0; + length = length|0; + output = output|0; + + var t0 = 0, t1 = 0, t2 = 0, t3 = 0, t4 = 0, hashed = 0; + + if ( offset & 63 ) + return -1; + + if ( ~output ) + if ( output & 31 ) + return -1; + + hashed = finish( offset, length, -1 )|0; + t0 = H0, t1 = H1, t2 = H2, t3 = H3, t4 = H4; + + _hmac_opad(); + _core( t0, t1, t2, t3, t4, 0x80000000, 0, 0, 0, 0, 0, 0, 0, 0, 0, 672 ); + + if ( ~output ) + _state_to_heap(output); + + return hashed|0; + } + + // salt is assumed to be already processed + // offset — multiple of 64 + // output — multiple of 32 + function pbkdf2_generate_block ( offset, length, block, count, output ) { + offset = offset|0; + length = length|0; + block = block|0; + count = count|0; + output = output|0; + + var h0 = 0, h1 = 0, h2 = 0, h3 = 0, h4 = 0, + t0 = 0, t1 = 0, t2 = 0, t3 = 0, t4 = 0; + + if ( offset & 63 ) + return -1; + + if ( ~output ) + if ( output & 31 ) + return -1; + + // pad block number into heap + // FIXME probable OOB write + HEAP[(offset+length)|0] = block>>>24; + HEAP[(offset+length+1)|0] = block>>>16&255; + HEAP[(offset+length+2)|0] = block>>>8&255; + HEAP[(offset+length+3)|0] = block&255; + + // finish first iteration + hmac_finish( offset, (length+4)|0, -1 )|0; + h0 = t0 = H0, h1 = t1 = H1, h2 = t2 = H2, h3 = t3 = H3, h4 = t4 = H4; + count = (count-1)|0; + + // perform the rest iterations + while ( (count|0) > 0 ) { + hmac_reset(); + _core( t0, t1, t2, t3, t4, 0x80000000, 0, 0, 0, 0, 0, 0, 0, 0, 0, 672 ); + t0 = H0, t1 = H1, t2 = H2, t3 = H3, t4 = H4; + + _hmac_opad(); + _core( t0, t1, t2, t3, t4, 0x80000000, 0, 0, 0, 0, 0, 0, 0, 0, 0, 672 ); + t0 = H0, t1 = H1, t2 = H2, t3 = H3, t4 = H4; + + h0 = h0 ^ H0; + h1 = h1 ^ H1; + h2 = h2 ^ H2; + h3 = h3 ^ H3; + h4 = h4 ^ H4; + + count = (count-1)|0; + } + + H0 = h0; + H1 = h1; + H2 = h2; + H3 = h3; + H4 = h4; + + if ( ~output ) + _state_to_heap(output); + + return 0; + } + + return { + // SHA1 + reset: reset, + init: init, + process: process, + finish: finish, + + // HMAC-SHA1 + hmac_reset: hmac_reset, + hmac_init: hmac_init, + hmac_finish: hmac_finish, + + // PBKDF2-HMAC-SHA1 + pbkdf2_generate_block: pbkdf2_generate_block + } + }; + + class Hash { + constructor() { + this.pos = 0; + this.len = 0; + } + reset() { + const { asm } = this.acquire_asm(); + this.result = null; + this.pos = 0; + this.len = 0; + asm.reset(); + return this; + } + process(data) { + if (this.result !== null) + throw new IllegalStateError('state must be reset before processing new data'); + const { asm, heap } = this.acquire_asm(); + let hpos = this.pos; + let hlen = this.len; + let dpos = 0; + let dlen = data.length; + let wlen = 0; + while (dlen > 0) { + wlen = _heap_write(heap, hpos + hlen, data, dpos, dlen); + hlen += wlen; + dpos += wlen; + dlen -= wlen; + wlen = asm.process(hpos, hlen); + hpos += wlen; + hlen -= wlen; + if (!hlen) + hpos = 0; + } + this.pos = hpos; + this.len = hlen; + return this; + } + finish() { + if (this.result !== null) + throw new IllegalStateError('state must be reset before processing new data'); + const { asm, heap } = this.acquire_asm(); + asm.finish(this.pos, this.len, 0); + this.result = new Uint8Array(this.HASH_SIZE); + this.result.set(heap.subarray(0, this.HASH_SIZE)); + this.pos = 0; + this.len = 0; + this.release_asm(); + return this; + } + } + + const _sha1_block_size = 64; + const _sha1_hash_size = 20; + const heap_pool$1 = []; + const asm_pool$1 = []; + class Sha1 extends Hash { + constructor() { + super(); + this.NAME = 'sha1'; + this.BLOCK_SIZE = _sha1_block_size; + this.HASH_SIZE = _sha1_hash_size; + this.acquire_asm(); + } + acquire_asm() { + if (this.heap === undefined || this.asm === undefined) { + this.heap = heap_pool$1.pop() || _heap_init(); + this.asm = asm_pool$1.pop() || sha1_asm({ Uint8Array: Uint8Array }, null, this.heap.buffer); + this.reset(); + } + return { heap: this.heap, asm: this.asm }; + } + release_asm() { + if (this.heap !== undefined && this.asm !== undefined) { + heap_pool$1.push(this.heap); + asm_pool$1.push(this.asm); + } + this.heap = undefined; + this.asm = undefined; + } + static bytes(data) { + return new Sha1().process(data).finish().result; + } + } + Sha1.NAME = 'sha1'; + Sha1.heap_pool = []; + Sha1.asm_pool = []; + Sha1.asm_function = sha1_asm; + + var sha256_asm = function ( stdlib, foreign, buffer ) { + "use asm"; + + // SHA256 state + var H0 = 0, H1 = 0, H2 = 0, H3 = 0, H4 = 0, H5 = 0, H6 = 0, H7 = 0, + TOTAL0 = 0, TOTAL1 = 0; + + // HMAC state + var I0 = 0, I1 = 0, I2 = 0, I3 = 0, I4 = 0, I5 = 0, I6 = 0, I7 = 0, + O0 = 0, O1 = 0, O2 = 0, O3 = 0, O4 = 0, O5 = 0, O6 = 0, O7 = 0; + + // I/O buffer + var HEAP = new stdlib.Uint8Array(buffer); + + function _core ( w0, w1, w2, w3, w4, w5, w6, w7, w8, w9, w10, w11, w12, w13, w14, w15 ) { + w0 = w0|0; + w1 = w1|0; + w2 = w2|0; + w3 = w3|0; + w4 = w4|0; + w5 = w5|0; + w6 = w6|0; + w7 = w7|0; + w8 = w8|0; + w9 = w9|0; + w10 = w10|0; + w11 = w11|0; + w12 = w12|0; + w13 = w13|0; + w14 = w14|0; + w15 = w15|0; + + var a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0; + + a = H0; + b = H1; + c = H2; + d = H3; + e = H4; + f = H5; + g = H6; + h = H7; + + // 0 + h = ( w0 + h + ( e>>>6 ^ e>>>11 ^ e>>>25 ^ e<<26 ^ e<<21 ^ e<<7 ) + ( g ^ e & (f^g) ) + 0x428a2f98 )|0; + d = ( d + h )|0; + h = ( h + ( (a & b) ^ ( c & (a ^ b) ) ) + ( a>>>2 ^ a>>>13 ^ a>>>22 ^ a<<30 ^ a<<19 ^ a<<10 ) )|0; + + // 1 + g = ( w1 + g + ( d>>>6 ^ d>>>11 ^ d>>>25 ^ d<<26 ^ d<<21 ^ d<<7 ) + ( f ^ d & (e^f) ) + 0x71374491 )|0; + c = ( c + g )|0; + g = ( g + ( (h & a) ^ ( b & (h ^ a) ) ) + ( h>>>2 ^ h>>>13 ^ h>>>22 ^ h<<30 ^ h<<19 ^ h<<10 ) )|0; + + // 2 + f = ( w2 + f + ( c>>>6 ^ c>>>11 ^ c>>>25 ^ c<<26 ^ c<<21 ^ c<<7 ) + ( e ^ c & (d^e) ) + 0xb5c0fbcf )|0; + b = ( b + f )|0; + f = ( f + ( (g & h) ^ ( a & (g ^ h) ) ) + ( g>>>2 ^ g>>>13 ^ g>>>22 ^ g<<30 ^ g<<19 ^ g<<10 ) )|0; + + // 3 + e = ( w3 + e + ( b>>>6 ^ b>>>11 ^ b>>>25 ^ b<<26 ^ b<<21 ^ b<<7 ) + ( d ^ b & (c^d) ) + 0xe9b5dba5 )|0; + a = ( a + e )|0; + e = ( e + ( (f & g) ^ ( h & (f ^ g) ) ) + ( f>>>2 ^ f>>>13 ^ f>>>22 ^ f<<30 ^ f<<19 ^ f<<10 ) )|0; + + // 4 + d = ( w4 + d + ( a>>>6 ^ a>>>11 ^ a>>>25 ^ a<<26 ^ a<<21 ^ a<<7 ) + ( c ^ a & (b^c) ) + 0x3956c25b )|0; + h = ( h + d )|0; + d = ( d + ( (e & f) ^ ( g & (e ^ f) ) ) + ( e>>>2 ^ e>>>13 ^ e>>>22 ^ e<<30 ^ e<<19 ^ e<<10 ) )|0; + + // 5 + c = ( w5 + c + ( h>>>6 ^ h>>>11 ^ h>>>25 ^ h<<26 ^ h<<21 ^ h<<7 ) + ( b ^ h & (a^b) ) + 0x59f111f1 )|0; + g = ( g + c )|0; + c = ( c + ( (d & e) ^ ( f & (d ^ e) ) ) + ( d>>>2 ^ d>>>13 ^ d>>>22 ^ d<<30 ^ d<<19 ^ d<<10 ) )|0; + + // 6 + b = ( w6 + b + ( g>>>6 ^ g>>>11 ^ g>>>25 ^ g<<26 ^ g<<21 ^ g<<7 ) + ( a ^ g & (h^a) ) + 0x923f82a4 )|0; + f = ( f + b )|0; + b = ( b + ( (c & d) ^ ( e & (c ^ d) ) ) + ( c>>>2 ^ c>>>13 ^ c>>>22 ^ c<<30 ^ c<<19 ^ c<<10 ) )|0; + + // 7 + a = ( w7 + a + ( f>>>6 ^ f>>>11 ^ f>>>25 ^ f<<26 ^ f<<21 ^ f<<7 ) + ( h ^ f & (g^h) ) + 0xab1c5ed5 )|0; + e = ( e + a )|0; + a = ( a + ( (b & c) ^ ( d & (b ^ c) ) ) + ( b>>>2 ^ b>>>13 ^ b>>>22 ^ b<<30 ^ b<<19 ^ b<<10 ) )|0; + + // 8 + h = ( w8 + h + ( e>>>6 ^ e>>>11 ^ e>>>25 ^ e<<26 ^ e<<21 ^ e<<7 ) + ( g ^ e & (f^g) ) + 0xd807aa98 )|0; + d = ( d + h )|0; + h = ( h + ( (a & b) ^ ( c & (a ^ b) ) ) + ( a>>>2 ^ a>>>13 ^ a>>>22 ^ a<<30 ^ a<<19 ^ a<<10 ) )|0; + + // 9 + g = ( w9 + g + ( d>>>6 ^ d>>>11 ^ d>>>25 ^ d<<26 ^ d<<21 ^ d<<7 ) + ( f ^ d & (e^f) ) + 0x12835b01 )|0; + c = ( c + g )|0; + g = ( g + ( (h & a) ^ ( b & (h ^ a) ) ) + ( h>>>2 ^ h>>>13 ^ h>>>22 ^ h<<30 ^ h<<19 ^ h<<10 ) )|0; + + // 10 + f = ( w10 + f + ( c>>>6 ^ c>>>11 ^ c>>>25 ^ c<<26 ^ c<<21 ^ c<<7 ) + ( e ^ c & (d^e) ) + 0x243185be )|0; + b = ( b + f )|0; + f = ( f + ( (g & h) ^ ( a & (g ^ h) ) ) + ( g>>>2 ^ g>>>13 ^ g>>>22 ^ g<<30 ^ g<<19 ^ g<<10 ) )|0; + + // 11 + e = ( w11 + e + ( b>>>6 ^ b>>>11 ^ b>>>25 ^ b<<26 ^ b<<21 ^ b<<7 ) + ( d ^ b & (c^d) ) + 0x550c7dc3 )|0; + a = ( a + e )|0; + e = ( e + ( (f & g) ^ ( h & (f ^ g) ) ) + ( f>>>2 ^ f>>>13 ^ f>>>22 ^ f<<30 ^ f<<19 ^ f<<10 ) )|0; + + // 12 + d = ( w12 + d + ( a>>>6 ^ a>>>11 ^ a>>>25 ^ a<<26 ^ a<<21 ^ a<<7 ) + ( c ^ a & (b^c) ) + 0x72be5d74 )|0; + h = ( h + d )|0; + d = ( d + ( (e & f) ^ ( g & (e ^ f) ) ) + ( e>>>2 ^ e>>>13 ^ e>>>22 ^ e<<30 ^ e<<19 ^ e<<10 ) )|0; + + // 13 + c = ( w13 + c + ( h>>>6 ^ h>>>11 ^ h>>>25 ^ h<<26 ^ h<<21 ^ h<<7 ) + ( b ^ h & (a^b) ) + 0x80deb1fe )|0; + g = ( g + c )|0; + c = ( c + ( (d & e) ^ ( f & (d ^ e) ) ) + ( d>>>2 ^ d>>>13 ^ d>>>22 ^ d<<30 ^ d<<19 ^ d<<10 ) )|0; + + // 14 + b = ( w14 + b + ( g>>>6 ^ g>>>11 ^ g>>>25 ^ g<<26 ^ g<<21 ^ g<<7 ) + ( a ^ g & (h^a) ) + 0x9bdc06a7 )|0; + f = ( f + b )|0; + b = ( b + ( (c & d) ^ ( e & (c ^ d) ) ) + ( c>>>2 ^ c>>>13 ^ c>>>22 ^ c<<30 ^ c<<19 ^ c<<10 ) )|0; + + // 15 + a = ( w15 + a + ( f>>>6 ^ f>>>11 ^ f>>>25 ^ f<<26 ^ f<<21 ^ f<<7 ) + ( h ^ f & (g^h) ) + 0xc19bf174 )|0; + e = ( e + a )|0; + a = ( a + ( (b & c) ^ ( d & (b ^ c) ) ) + ( b>>>2 ^ b>>>13 ^ b>>>22 ^ b<<30 ^ b<<19 ^ b<<10 ) )|0; + + // 16 + w0 = ( ( w1>>>7 ^ w1>>>18 ^ w1>>>3 ^ w1<<25 ^ w1<<14 ) + ( w14>>>17 ^ w14>>>19 ^ w14>>>10 ^ w14<<15 ^ w14<<13 ) + w0 + w9 )|0; + h = ( w0 + h + ( e>>>6 ^ e>>>11 ^ e>>>25 ^ e<<26 ^ e<<21 ^ e<<7 ) + ( g ^ e & (f^g) ) + 0xe49b69c1 )|0; + d = ( d + h )|0; + h = ( h + ( (a & b) ^ ( c & (a ^ b) ) ) + ( a>>>2 ^ a>>>13 ^ a>>>22 ^ a<<30 ^ a<<19 ^ a<<10 ) )|0; + + // 17 + w1 = ( ( w2>>>7 ^ w2>>>18 ^ w2>>>3 ^ w2<<25 ^ w2<<14 ) + ( w15>>>17 ^ w15>>>19 ^ w15>>>10 ^ w15<<15 ^ w15<<13 ) + w1 + w10 )|0; + g = ( w1 + g + ( d>>>6 ^ d>>>11 ^ d>>>25 ^ d<<26 ^ d<<21 ^ d<<7 ) + ( f ^ d & (e^f) ) + 0xefbe4786 )|0; + c = ( c + g )|0; + g = ( g + ( (h & a) ^ ( b & (h ^ a) ) ) + ( h>>>2 ^ h>>>13 ^ h>>>22 ^ h<<30 ^ h<<19 ^ h<<10 ) )|0; + + // 18 + w2 = ( ( w3>>>7 ^ w3>>>18 ^ w3>>>3 ^ w3<<25 ^ w3<<14 ) + ( w0>>>17 ^ w0>>>19 ^ w0>>>10 ^ w0<<15 ^ w0<<13 ) + w2 + w11 )|0; + f = ( w2 + f + ( c>>>6 ^ c>>>11 ^ c>>>25 ^ c<<26 ^ c<<21 ^ c<<7 ) + ( e ^ c & (d^e) ) + 0x0fc19dc6 )|0; + b = ( b + f )|0; + f = ( f + ( (g & h) ^ ( a & (g ^ h) ) ) + ( g>>>2 ^ g>>>13 ^ g>>>22 ^ g<<30 ^ g<<19 ^ g<<10 ) )|0; + + // 19 + w3 = ( ( w4>>>7 ^ w4>>>18 ^ w4>>>3 ^ w4<<25 ^ w4<<14 ) + ( w1>>>17 ^ w1>>>19 ^ w1>>>10 ^ w1<<15 ^ w1<<13 ) + w3 + w12 )|0; + e = ( w3 + e + ( b>>>6 ^ b>>>11 ^ b>>>25 ^ b<<26 ^ b<<21 ^ b<<7 ) + ( d ^ b & (c^d) ) + 0x240ca1cc )|0; + a = ( a + e )|0; + e = ( e + ( (f & g) ^ ( h & (f ^ g) ) ) + ( f>>>2 ^ f>>>13 ^ f>>>22 ^ f<<30 ^ f<<19 ^ f<<10 ) )|0; + + // 20 + w4 = ( ( w5>>>7 ^ w5>>>18 ^ w5>>>3 ^ w5<<25 ^ w5<<14 ) + ( w2>>>17 ^ w2>>>19 ^ w2>>>10 ^ w2<<15 ^ w2<<13 ) + w4 + w13 )|0; + d = ( w4 + d + ( a>>>6 ^ a>>>11 ^ a>>>25 ^ a<<26 ^ a<<21 ^ a<<7 ) + ( c ^ a & (b^c) ) + 0x2de92c6f )|0; + h = ( h + d )|0; + d = ( d + ( (e & f) ^ ( g & (e ^ f) ) ) + ( e>>>2 ^ e>>>13 ^ e>>>22 ^ e<<30 ^ e<<19 ^ e<<10 ) )|0; + + // 21 + w5 = ( ( w6>>>7 ^ w6>>>18 ^ w6>>>3 ^ w6<<25 ^ w6<<14 ) + ( w3>>>17 ^ w3>>>19 ^ w3>>>10 ^ w3<<15 ^ w3<<13 ) + w5 + w14 )|0; + c = ( w5 + c + ( h>>>6 ^ h>>>11 ^ h>>>25 ^ h<<26 ^ h<<21 ^ h<<7 ) + ( b ^ h & (a^b) ) + 0x4a7484aa )|0; + g = ( g + c )|0; + c = ( c + ( (d & e) ^ ( f & (d ^ e) ) ) + ( d>>>2 ^ d>>>13 ^ d>>>22 ^ d<<30 ^ d<<19 ^ d<<10 ) )|0; + + // 22 + w6 = ( ( w7>>>7 ^ w7>>>18 ^ w7>>>3 ^ w7<<25 ^ w7<<14 ) + ( w4>>>17 ^ w4>>>19 ^ w4>>>10 ^ w4<<15 ^ w4<<13 ) + w6 + w15 )|0; + b = ( w6 + b + ( g>>>6 ^ g>>>11 ^ g>>>25 ^ g<<26 ^ g<<21 ^ g<<7 ) + ( a ^ g & (h^a) ) + 0x5cb0a9dc )|0; + f = ( f + b )|0; + b = ( b + ( (c & d) ^ ( e & (c ^ d) ) ) + ( c>>>2 ^ c>>>13 ^ c>>>22 ^ c<<30 ^ c<<19 ^ c<<10 ) )|0; + + // 23 + w7 = ( ( w8>>>7 ^ w8>>>18 ^ w8>>>3 ^ w8<<25 ^ w8<<14 ) + ( w5>>>17 ^ w5>>>19 ^ w5>>>10 ^ w5<<15 ^ w5<<13 ) + w7 + w0 )|0; + a = ( w7 + a + ( f>>>6 ^ f>>>11 ^ f>>>25 ^ f<<26 ^ f<<21 ^ f<<7 ) + ( h ^ f & (g^h) ) + 0x76f988da )|0; + e = ( e + a )|0; + a = ( a + ( (b & c) ^ ( d & (b ^ c) ) ) + ( b>>>2 ^ b>>>13 ^ b>>>22 ^ b<<30 ^ b<<19 ^ b<<10 ) )|0; + + // 24 + w8 = ( ( w9>>>7 ^ w9>>>18 ^ w9>>>3 ^ w9<<25 ^ w9<<14 ) + ( w6>>>17 ^ w6>>>19 ^ w6>>>10 ^ w6<<15 ^ w6<<13 ) + w8 + w1 )|0; + h = ( w8 + h + ( e>>>6 ^ e>>>11 ^ e>>>25 ^ e<<26 ^ e<<21 ^ e<<7 ) + ( g ^ e & (f^g) ) + 0x983e5152 )|0; + d = ( d + h )|0; + h = ( h + ( (a & b) ^ ( c & (a ^ b) ) ) + ( a>>>2 ^ a>>>13 ^ a>>>22 ^ a<<30 ^ a<<19 ^ a<<10 ) )|0; + + // 25 + w9 = ( ( w10>>>7 ^ w10>>>18 ^ w10>>>3 ^ w10<<25 ^ w10<<14 ) + ( w7>>>17 ^ w7>>>19 ^ w7>>>10 ^ w7<<15 ^ w7<<13 ) + w9 + w2 )|0; + g = ( w9 + g + ( d>>>6 ^ d>>>11 ^ d>>>25 ^ d<<26 ^ d<<21 ^ d<<7 ) + ( f ^ d & (e^f) ) + 0xa831c66d )|0; + c = ( c + g )|0; + g = ( g + ( (h & a) ^ ( b & (h ^ a) ) ) + ( h>>>2 ^ h>>>13 ^ h>>>22 ^ h<<30 ^ h<<19 ^ h<<10 ) )|0; + + // 26 + w10 = ( ( w11>>>7 ^ w11>>>18 ^ w11>>>3 ^ w11<<25 ^ w11<<14 ) + ( w8>>>17 ^ w8>>>19 ^ w8>>>10 ^ w8<<15 ^ w8<<13 ) + w10 + w3 )|0; + f = ( w10 + f + ( c>>>6 ^ c>>>11 ^ c>>>25 ^ c<<26 ^ c<<21 ^ c<<7 ) + ( e ^ c & (d^e) ) + 0xb00327c8 )|0; + b = ( b + f )|0; + f = ( f + ( (g & h) ^ ( a & (g ^ h) ) ) + ( g>>>2 ^ g>>>13 ^ g>>>22 ^ g<<30 ^ g<<19 ^ g<<10 ) )|0; + + // 27 + w11 = ( ( w12>>>7 ^ w12>>>18 ^ w12>>>3 ^ w12<<25 ^ w12<<14 ) + ( w9>>>17 ^ w9>>>19 ^ w9>>>10 ^ w9<<15 ^ w9<<13 ) + w11 + w4 )|0; + e = ( w11 + e + ( b>>>6 ^ b>>>11 ^ b>>>25 ^ b<<26 ^ b<<21 ^ b<<7 ) + ( d ^ b & (c^d) ) + 0xbf597fc7 )|0; + a = ( a + e )|0; + e = ( e + ( (f & g) ^ ( h & (f ^ g) ) ) + ( f>>>2 ^ f>>>13 ^ f>>>22 ^ f<<30 ^ f<<19 ^ f<<10 ) )|0; + + // 28 + w12 = ( ( w13>>>7 ^ w13>>>18 ^ w13>>>3 ^ w13<<25 ^ w13<<14 ) + ( w10>>>17 ^ w10>>>19 ^ w10>>>10 ^ w10<<15 ^ w10<<13 ) + w12 + w5 )|0; + d = ( w12 + d + ( a>>>6 ^ a>>>11 ^ a>>>25 ^ a<<26 ^ a<<21 ^ a<<7 ) + ( c ^ a & (b^c) ) + 0xc6e00bf3 )|0; + h = ( h + d )|0; + d = ( d + ( (e & f) ^ ( g & (e ^ f) ) ) + ( e>>>2 ^ e>>>13 ^ e>>>22 ^ e<<30 ^ e<<19 ^ e<<10 ) )|0; + + // 29 + w13 = ( ( w14>>>7 ^ w14>>>18 ^ w14>>>3 ^ w14<<25 ^ w14<<14 ) + ( w11>>>17 ^ w11>>>19 ^ w11>>>10 ^ w11<<15 ^ w11<<13 ) + w13 + w6 )|0; + c = ( w13 + c + ( h>>>6 ^ h>>>11 ^ h>>>25 ^ h<<26 ^ h<<21 ^ h<<7 ) + ( b ^ h & (a^b) ) + 0xd5a79147 )|0; + g = ( g + c )|0; + c = ( c + ( (d & e) ^ ( f & (d ^ e) ) ) + ( d>>>2 ^ d>>>13 ^ d>>>22 ^ d<<30 ^ d<<19 ^ d<<10 ) )|0; + + // 30 + w14 = ( ( w15>>>7 ^ w15>>>18 ^ w15>>>3 ^ w15<<25 ^ w15<<14 ) + ( w12>>>17 ^ w12>>>19 ^ w12>>>10 ^ w12<<15 ^ w12<<13 ) + w14 + w7 )|0; + b = ( w14 + b + ( g>>>6 ^ g>>>11 ^ g>>>25 ^ g<<26 ^ g<<21 ^ g<<7 ) + ( a ^ g & (h^a) ) + 0x06ca6351 )|0; + f = ( f + b )|0; + b = ( b + ( (c & d) ^ ( e & (c ^ d) ) ) + ( c>>>2 ^ c>>>13 ^ c>>>22 ^ c<<30 ^ c<<19 ^ c<<10 ) )|0; + + // 31 + w15 = ( ( w0>>>7 ^ w0>>>18 ^ w0>>>3 ^ w0<<25 ^ w0<<14 ) + ( w13>>>17 ^ w13>>>19 ^ w13>>>10 ^ w13<<15 ^ w13<<13 ) + w15 + w8 )|0; + a = ( w15 + a + ( f>>>6 ^ f>>>11 ^ f>>>25 ^ f<<26 ^ f<<21 ^ f<<7 ) + ( h ^ f & (g^h) ) + 0x14292967 )|0; + e = ( e + a )|0; + a = ( a + ( (b & c) ^ ( d & (b ^ c) ) ) + ( b>>>2 ^ b>>>13 ^ b>>>22 ^ b<<30 ^ b<<19 ^ b<<10 ) )|0; + + // 32 + w0 = ( ( w1>>>7 ^ w1>>>18 ^ w1>>>3 ^ w1<<25 ^ w1<<14 ) + ( w14>>>17 ^ w14>>>19 ^ w14>>>10 ^ w14<<15 ^ w14<<13 ) + w0 + w9 )|0; + h = ( w0 + h + ( e>>>6 ^ e>>>11 ^ e>>>25 ^ e<<26 ^ e<<21 ^ e<<7 ) + ( g ^ e & (f^g) ) + 0x27b70a85 )|0; + d = ( d + h )|0; + h = ( h + ( (a & b) ^ ( c & (a ^ b) ) ) + ( a>>>2 ^ a>>>13 ^ a>>>22 ^ a<<30 ^ a<<19 ^ a<<10 ) )|0; + + // 33 + w1 = ( ( w2>>>7 ^ w2>>>18 ^ w2>>>3 ^ w2<<25 ^ w2<<14 ) + ( w15>>>17 ^ w15>>>19 ^ w15>>>10 ^ w15<<15 ^ w15<<13 ) + w1 + w10 )|0; + g = ( w1 + g + ( d>>>6 ^ d>>>11 ^ d>>>25 ^ d<<26 ^ d<<21 ^ d<<7 ) + ( f ^ d & (e^f) ) + 0x2e1b2138 )|0; + c = ( c + g )|0; + g = ( g + ( (h & a) ^ ( b & (h ^ a) ) ) + ( h>>>2 ^ h>>>13 ^ h>>>22 ^ h<<30 ^ h<<19 ^ h<<10 ) )|0; + + // 34 + w2 = ( ( w3>>>7 ^ w3>>>18 ^ w3>>>3 ^ w3<<25 ^ w3<<14 ) + ( w0>>>17 ^ w0>>>19 ^ w0>>>10 ^ w0<<15 ^ w0<<13 ) + w2 + w11 )|0; + f = ( w2 + f + ( c>>>6 ^ c>>>11 ^ c>>>25 ^ c<<26 ^ c<<21 ^ c<<7 ) + ( e ^ c & (d^e) ) + 0x4d2c6dfc )|0; + b = ( b + f )|0; + f = ( f + ( (g & h) ^ ( a & (g ^ h) ) ) + ( g>>>2 ^ g>>>13 ^ g>>>22 ^ g<<30 ^ g<<19 ^ g<<10 ) )|0; + + // 35 + w3 = ( ( w4>>>7 ^ w4>>>18 ^ w4>>>3 ^ w4<<25 ^ w4<<14 ) + ( w1>>>17 ^ w1>>>19 ^ w1>>>10 ^ w1<<15 ^ w1<<13 ) + w3 + w12 )|0; + e = ( w3 + e + ( b>>>6 ^ b>>>11 ^ b>>>25 ^ b<<26 ^ b<<21 ^ b<<7 ) + ( d ^ b & (c^d) ) + 0x53380d13 )|0; + a = ( a + e )|0; + e = ( e + ( (f & g) ^ ( h & (f ^ g) ) ) + ( f>>>2 ^ f>>>13 ^ f>>>22 ^ f<<30 ^ f<<19 ^ f<<10 ) )|0; + + // 36 + w4 = ( ( w5>>>7 ^ w5>>>18 ^ w5>>>3 ^ w5<<25 ^ w5<<14 ) + ( w2>>>17 ^ w2>>>19 ^ w2>>>10 ^ w2<<15 ^ w2<<13 ) + w4 + w13 )|0; + d = ( w4 + d + ( a>>>6 ^ a>>>11 ^ a>>>25 ^ a<<26 ^ a<<21 ^ a<<7 ) + ( c ^ a & (b^c) ) + 0x650a7354 )|0; + h = ( h + d )|0; + d = ( d + ( (e & f) ^ ( g & (e ^ f) ) ) + ( e>>>2 ^ e>>>13 ^ e>>>22 ^ e<<30 ^ e<<19 ^ e<<10 ) )|0; + + // 37 + w5 = ( ( w6>>>7 ^ w6>>>18 ^ w6>>>3 ^ w6<<25 ^ w6<<14 ) + ( w3>>>17 ^ w3>>>19 ^ w3>>>10 ^ w3<<15 ^ w3<<13 ) + w5 + w14 )|0; + c = ( w5 + c + ( h>>>6 ^ h>>>11 ^ h>>>25 ^ h<<26 ^ h<<21 ^ h<<7 ) + ( b ^ h & (a^b) ) + 0x766a0abb )|0; + g = ( g + c )|0; + c = ( c + ( (d & e) ^ ( f & (d ^ e) ) ) + ( d>>>2 ^ d>>>13 ^ d>>>22 ^ d<<30 ^ d<<19 ^ d<<10 ) )|0; + + // 38 + w6 = ( ( w7>>>7 ^ w7>>>18 ^ w7>>>3 ^ w7<<25 ^ w7<<14 ) + ( w4>>>17 ^ w4>>>19 ^ w4>>>10 ^ w4<<15 ^ w4<<13 ) + w6 + w15 )|0; + b = ( w6 + b + ( g>>>6 ^ g>>>11 ^ g>>>25 ^ g<<26 ^ g<<21 ^ g<<7 ) + ( a ^ g & (h^a) ) + 0x81c2c92e )|0; + f = ( f + b )|0; + b = ( b + ( (c & d) ^ ( e & (c ^ d) ) ) + ( c>>>2 ^ c>>>13 ^ c>>>22 ^ c<<30 ^ c<<19 ^ c<<10 ) )|0; + + // 39 + w7 = ( ( w8>>>7 ^ w8>>>18 ^ w8>>>3 ^ w8<<25 ^ w8<<14 ) + ( w5>>>17 ^ w5>>>19 ^ w5>>>10 ^ w5<<15 ^ w5<<13 ) + w7 + w0 )|0; + a = ( w7 + a + ( f>>>6 ^ f>>>11 ^ f>>>25 ^ f<<26 ^ f<<21 ^ f<<7 ) + ( h ^ f & (g^h) ) + 0x92722c85 )|0; + e = ( e + a )|0; + a = ( a + ( (b & c) ^ ( d & (b ^ c) ) ) + ( b>>>2 ^ b>>>13 ^ b>>>22 ^ b<<30 ^ b<<19 ^ b<<10 ) )|0; + + // 40 + w8 = ( ( w9>>>7 ^ w9>>>18 ^ w9>>>3 ^ w9<<25 ^ w9<<14 ) + ( w6>>>17 ^ w6>>>19 ^ w6>>>10 ^ w6<<15 ^ w6<<13 ) + w8 + w1 )|0; + h = ( w8 + h + ( e>>>6 ^ e>>>11 ^ e>>>25 ^ e<<26 ^ e<<21 ^ e<<7 ) + ( g ^ e & (f^g) ) + 0xa2bfe8a1 )|0; + d = ( d + h )|0; + h = ( h + ( (a & b) ^ ( c & (a ^ b) ) ) + ( a>>>2 ^ a>>>13 ^ a>>>22 ^ a<<30 ^ a<<19 ^ a<<10 ) )|0; + + // 41 + w9 = ( ( w10>>>7 ^ w10>>>18 ^ w10>>>3 ^ w10<<25 ^ w10<<14 ) + ( w7>>>17 ^ w7>>>19 ^ w7>>>10 ^ w7<<15 ^ w7<<13 ) + w9 + w2 )|0; + g = ( w9 + g + ( d>>>6 ^ d>>>11 ^ d>>>25 ^ d<<26 ^ d<<21 ^ d<<7 ) + ( f ^ d & (e^f) ) + 0xa81a664b )|0; + c = ( c + g )|0; + g = ( g + ( (h & a) ^ ( b & (h ^ a) ) ) + ( h>>>2 ^ h>>>13 ^ h>>>22 ^ h<<30 ^ h<<19 ^ h<<10 ) )|0; + + // 42 + w10 = ( ( w11>>>7 ^ w11>>>18 ^ w11>>>3 ^ w11<<25 ^ w11<<14 ) + ( w8>>>17 ^ w8>>>19 ^ w8>>>10 ^ w8<<15 ^ w8<<13 ) + w10 + w3 )|0; + f = ( w10 + f + ( c>>>6 ^ c>>>11 ^ c>>>25 ^ c<<26 ^ c<<21 ^ c<<7 ) + ( e ^ c & (d^e) ) + 0xc24b8b70 )|0; + b = ( b + f )|0; + f = ( f + ( (g & h) ^ ( a & (g ^ h) ) ) + ( g>>>2 ^ g>>>13 ^ g>>>22 ^ g<<30 ^ g<<19 ^ g<<10 ) )|0; + + // 43 + w11 = ( ( w12>>>7 ^ w12>>>18 ^ w12>>>3 ^ w12<<25 ^ w12<<14 ) + ( w9>>>17 ^ w9>>>19 ^ w9>>>10 ^ w9<<15 ^ w9<<13 ) + w11 + w4 )|0; + e = ( w11 + e + ( b>>>6 ^ b>>>11 ^ b>>>25 ^ b<<26 ^ b<<21 ^ b<<7 ) + ( d ^ b & (c^d) ) + 0xc76c51a3 )|0; + a = ( a + e )|0; + e = ( e + ( (f & g) ^ ( h & (f ^ g) ) ) + ( f>>>2 ^ f>>>13 ^ f>>>22 ^ f<<30 ^ f<<19 ^ f<<10 ) )|0; + + // 44 + w12 = ( ( w13>>>7 ^ w13>>>18 ^ w13>>>3 ^ w13<<25 ^ w13<<14 ) + ( w10>>>17 ^ w10>>>19 ^ w10>>>10 ^ w10<<15 ^ w10<<13 ) + w12 + w5 )|0; + d = ( w12 + d + ( a>>>6 ^ a>>>11 ^ a>>>25 ^ a<<26 ^ a<<21 ^ a<<7 ) + ( c ^ a & (b^c) ) + 0xd192e819 )|0; + h = ( h + d )|0; + d = ( d + ( (e & f) ^ ( g & (e ^ f) ) ) + ( e>>>2 ^ e>>>13 ^ e>>>22 ^ e<<30 ^ e<<19 ^ e<<10 ) )|0; + + // 45 + w13 = ( ( w14>>>7 ^ w14>>>18 ^ w14>>>3 ^ w14<<25 ^ w14<<14 ) + ( w11>>>17 ^ w11>>>19 ^ w11>>>10 ^ w11<<15 ^ w11<<13 ) + w13 + w6 )|0; + c = ( w13 + c + ( h>>>6 ^ h>>>11 ^ h>>>25 ^ h<<26 ^ h<<21 ^ h<<7 ) + ( b ^ h & (a^b) ) + 0xd6990624 )|0; + g = ( g + c )|0; + c = ( c + ( (d & e) ^ ( f & (d ^ e) ) ) + ( d>>>2 ^ d>>>13 ^ d>>>22 ^ d<<30 ^ d<<19 ^ d<<10 ) )|0; + + // 46 + w14 = ( ( w15>>>7 ^ w15>>>18 ^ w15>>>3 ^ w15<<25 ^ w15<<14 ) + ( w12>>>17 ^ w12>>>19 ^ w12>>>10 ^ w12<<15 ^ w12<<13 ) + w14 + w7 )|0; + b = ( w14 + b + ( g>>>6 ^ g>>>11 ^ g>>>25 ^ g<<26 ^ g<<21 ^ g<<7 ) + ( a ^ g & (h^a) ) + 0xf40e3585 )|0; + f = ( f + b )|0; + b = ( b + ( (c & d) ^ ( e & (c ^ d) ) ) + ( c>>>2 ^ c>>>13 ^ c>>>22 ^ c<<30 ^ c<<19 ^ c<<10 ) )|0; + + // 47 + w15 = ( ( w0>>>7 ^ w0>>>18 ^ w0>>>3 ^ w0<<25 ^ w0<<14 ) + ( w13>>>17 ^ w13>>>19 ^ w13>>>10 ^ w13<<15 ^ w13<<13 ) + w15 + w8 )|0; + a = ( w15 + a + ( f>>>6 ^ f>>>11 ^ f>>>25 ^ f<<26 ^ f<<21 ^ f<<7 ) + ( h ^ f & (g^h) ) + 0x106aa070 )|0; + e = ( e + a )|0; + a = ( a + ( (b & c) ^ ( d & (b ^ c) ) ) + ( b>>>2 ^ b>>>13 ^ b>>>22 ^ b<<30 ^ b<<19 ^ b<<10 ) )|0; + + // 48 + w0 = ( ( w1>>>7 ^ w1>>>18 ^ w1>>>3 ^ w1<<25 ^ w1<<14 ) + ( w14>>>17 ^ w14>>>19 ^ w14>>>10 ^ w14<<15 ^ w14<<13 ) + w0 + w9 )|0; + h = ( w0 + h + ( e>>>6 ^ e>>>11 ^ e>>>25 ^ e<<26 ^ e<<21 ^ e<<7 ) + ( g ^ e & (f^g) ) + 0x19a4c116 )|0; + d = ( d + h )|0; + h = ( h + ( (a & b) ^ ( c & (a ^ b) ) ) + ( a>>>2 ^ a>>>13 ^ a>>>22 ^ a<<30 ^ a<<19 ^ a<<10 ) )|0; + + // 49 + w1 = ( ( w2>>>7 ^ w2>>>18 ^ w2>>>3 ^ w2<<25 ^ w2<<14 ) + ( w15>>>17 ^ w15>>>19 ^ w15>>>10 ^ w15<<15 ^ w15<<13 ) + w1 + w10 )|0; + g = ( w1 + g + ( d>>>6 ^ d>>>11 ^ d>>>25 ^ d<<26 ^ d<<21 ^ d<<7 ) + ( f ^ d & (e^f) ) + 0x1e376c08 )|0; + c = ( c + g )|0; + g = ( g + ( (h & a) ^ ( b & (h ^ a) ) ) + ( h>>>2 ^ h>>>13 ^ h>>>22 ^ h<<30 ^ h<<19 ^ h<<10 ) )|0; + + // 50 + w2 = ( ( w3>>>7 ^ w3>>>18 ^ w3>>>3 ^ w3<<25 ^ w3<<14 ) + ( w0>>>17 ^ w0>>>19 ^ w0>>>10 ^ w0<<15 ^ w0<<13 ) + w2 + w11 )|0; + f = ( w2 + f + ( c>>>6 ^ c>>>11 ^ c>>>25 ^ c<<26 ^ c<<21 ^ c<<7 ) + ( e ^ c & (d^e) ) + 0x2748774c )|0; + b = ( b + f )|0; + f = ( f + ( (g & h) ^ ( a & (g ^ h) ) ) + ( g>>>2 ^ g>>>13 ^ g>>>22 ^ g<<30 ^ g<<19 ^ g<<10 ) )|0; + + // 51 + w3 = ( ( w4>>>7 ^ w4>>>18 ^ w4>>>3 ^ w4<<25 ^ w4<<14 ) + ( w1>>>17 ^ w1>>>19 ^ w1>>>10 ^ w1<<15 ^ w1<<13 ) + w3 + w12 )|0; + e = ( w3 + e + ( b>>>6 ^ b>>>11 ^ b>>>25 ^ b<<26 ^ b<<21 ^ b<<7 ) + ( d ^ b & (c^d) ) + 0x34b0bcb5 )|0; + a = ( a + e )|0; + e = ( e + ( (f & g) ^ ( h & (f ^ g) ) ) + ( f>>>2 ^ f>>>13 ^ f>>>22 ^ f<<30 ^ f<<19 ^ f<<10 ) )|0; + + // 52 + w4 = ( ( w5>>>7 ^ w5>>>18 ^ w5>>>3 ^ w5<<25 ^ w5<<14 ) + ( w2>>>17 ^ w2>>>19 ^ w2>>>10 ^ w2<<15 ^ w2<<13 ) + w4 + w13 )|0; + d = ( w4 + d + ( a>>>6 ^ a>>>11 ^ a>>>25 ^ a<<26 ^ a<<21 ^ a<<7 ) + ( c ^ a & (b^c) ) + 0x391c0cb3 )|0; + h = ( h + d )|0; + d = ( d + ( (e & f) ^ ( g & (e ^ f) ) ) + ( e>>>2 ^ e>>>13 ^ e>>>22 ^ e<<30 ^ e<<19 ^ e<<10 ) )|0; + + // 53 + w5 = ( ( w6>>>7 ^ w6>>>18 ^ w6>>>3 ^ w6<<25 ^ w6<<14 ) + ( w3>>>17 ^ w3>>>19 ^ w3>>>10 ^ w3<<15 ^ w3<<13 ) + w5 + w14 )|0; + c = ( w5 + c + ( h>>>6 ^ h>>>11 ^ h>>>25 ^ h<<26 ^ h<<21 ^ h<<7 ) + ( b ^ h & (a^b) ) + 0x4ed8aa4a )|0; + g = ( g + c )|0; + c = ( c + ( (d & e) ^ ( f & (d ^ e) ) ) + ( d>>>2 ^ d>>>13 ^ d>>>22 ^ d<<30 ^ d<<19 ^ d<<10 ) )|0; + + // 54 + w6 = ( ( w7>>>7 ^ w7>>>18 ^ w7>>>3 ^ w7<<25 ^ w7<<14 ) + ( w4>>>17 ^ w4>>>19 ^ w4>>>10 ^ w4<<15 ^ w4<<13 ) + w6 + w15 )|0; + b = ( w6 + b + ( g>>>6 ^ g>>>11 ^ g>>>25 ^ g<<26 ^ g<<21 ^ g<<7 ) + ( a ^ g & (h^a) ) + 0x5b9cca4f )|0; + f = ( f + b )|0; + b = ( b + ( (c & d) ^ ( e & (c ^ d) ) ) + ( c>>>2 ^ c>>>13 ^ c>>>22 ^ c<<30 ^ c<<19 ^ c<<10 ) )|0; + + // 55 + w7 = ( ( w8>>>7 ^ w8>>>18 ^ w8>>>3 ^ w8<<25 ^ w8<<14 ) + ( w5>>>17 ^ w5>>>19 ^ w5>>>10 ^ w5<<15 ^ w5<<13 ) + w7 + w0 )|0; + a = ( w7 + a + ( f>>>6 ^ f>>>11 ^ f>>>25 ^ f<<26 ^ f<<21 ^ f<<7 ) + ( h ^ f & (g^h) ) + 0x682e6ff3 )|0; + e = ( e + a )|0; + a = ( a + ( (b & c) ^ ( d & (b ^ c) ) ) + ( b>>>2 ^ b>>>13 ^ b>>>22 ^ b<<30 ^ b<<19 ^ b<<10 ) )|0; + + // 56 + w8 = ( ( w9>>>7 ^ w9>>>18 ^ w9>>>3 ^ w9<<25 ^ w9<<14 ) + ( w6>>>17 ^ w6>>>19 ^ w6>>>10 ^ w6<<15 ^ w6<<13 ) + w8 + w1 )|0; + h = ( w8 + h + ( e>>>6 ^ e>>>11 ^ e>>>25 ^ e<<26 ^ e<<21 ^ e<<7 ) + ( g ^ e & (f^g) ) + 0x748f82ee )|0; + d = ( d + h )|0; + h = ( h + ( (a & b) ^ ( c & (a ^ b) ) ) + ( a>>>2 ^ a>>>13 ^ a>>>22 ^ a<<30 ^ a<<19 ^ a<<10 ) )|0; + + // 57 + w9 = ( ( w10>>>7 ^ w10>>>18 ^ w10>>>3 ^ w10<<25 ^ w10<<14 ) + ( w7>>>17 ^ w7>>>19 ^ w7>>>10 ^ w7<<15 ^ w7<<13 ) + w9 + w2 )|0; + g = ( w9 + g + ( d>>>6 ^ d>>>11 ^ d>>>25 ^ d<<26 ^ d<<21 ^ d<<7 ) + ( f ^ d & (e^f) ) + 0x78a5636f )|0; + c = ( c + g )|0; + g = ( g + ( (h & a) ^ ( b & (h ^ a) ) ) + ( h>>>2 ^ h>>>13 ^ h>>>22 ^ h<<30 ^ h<<19 ^ h<<10 ) )|0; + + // 58 + w10 = ( ( w11>>>7 ^ w11>>>18 ^ w11>>>3 ^ w11<<25 ^ w11<<14 ) + ( w8>>>17 ^ w8>>>19 ^ w8>>>10 ^ w8<<15 ^ w8<<13 ) + w10 + w3 )|0; + f = ( w10 + f + ( c>>>6 ^ c>>>11 ^ c>>>25 ^ c<<26 ^ c<<21 ^ c<<7 ) + ( e ^ c & (d^e) ) + 0x84c87814 )|0; + b = ( b + f )|0; + f = ( f + ( (g & h) ^ ( a & (g ^ h) ) ) + ( g>>>2 ^ g>>>13 ^ g>>>22 ^ g<<30 ^ g<<19 ^ g<<10 ) )|0; + + // 59 + w11 = ( ( w12>>>7 ^ w12>>>18 ^ w12>>>3 ^ w12<<25 ^ w12<<14 ) + ( w9>>>17 ^ w9>>>19 ^ w9>>>10 ^ w9<<15 ^ w9<<13 ) + w11 + w4 )|0; + e = ( w11 + e + ( b>>>6 ^ b>>>11 ^ b>>>25 ^ b<<26 ^ b<<21 ^ b<<7 ) + ( d ^ b & (c^d) ) + 0x8cc70208 )|0; + a = ( a + e )|0; + e = ( e + ( (f & g) ^ ( h & (f ^ g) ) ) + ( f>>>2 ^ f>>>13 ^ f>>>22 ^ f<<30 ^ f<<19 ^ f<<10 ) )|0; + + // 60 + w12 = ( ( w13>>>7 ^ w13>>>18 ^ w13>>>3 ^ w13<<25 ^ w13<<14 ) + ( w10>>>17 ^ w10>>>19 ^ w10>>>10 ^ w10<<15 ^ w10<<13 ) + w12 + w5 )|0; + d = ( w12 + d + ( a>>>6 ^ a>>>11 ^ a>>>25 ^ a<<26 ^ a<<21 ^ a<<7 ) + ( c ^ a & (b^c) ) + 0x90befffa )|0; + h = ( h + d )|0; + d = ( d + ( (e & f) ^ ( g & (e ^ f) ) ) + ( e>>>2 ^ e>>>13 ^ e>>>22 ^ e<<30 ^ e<<19 ^ e<<10 ) )|0; + + // 61 + w13 = ( ( w14>>>7 ^ w14>>>18 ^ w14>>>3 ^ w14<<25 ^ w14<<14 ) + ( w11>>>17 ^ w11>>>19 ^ w11>>>10 ^ w11<<15 ^ w11<<13 ) + w13 + w6 )|0; + c = ( w13 + c + ( h>>>6 ^ h>>>11 ^ h>>>25 ^ h<<26 ^ h<<21 ^ h<<7 ) + ( b ^ h & (a^b) ) + 0xa4506ceb )|0; + g = ( g + c )|0; + c = ( c + ( (d & e) ^ ( f & (d ^ e) ) ) + ( d>>>2 ^ d>>>13 ^ d>>>22 ^ d<<30 ^ d<<19 ^ d<<10 ) )|0; + + // 62 + w14 = ( ( w15>>>7 ^ w15>>>18 ^ w15>>>3 ^ w15<<25 ^ w15<<14 ) + ( w12>>>17 ^ w12>>>19 ^ w12>>>10 ^ w12<<15 ^ w12<<13 ) + w14 + w7 )|0; + b = ( w14 + b + ( g>>>6 ^ g>>>11 ^ g>>>25 ^ g<<26 ^ g<<21 ^ g<<7 ) + ( a ^ g & (h^a) ) + 0xbef9a3f7 )|0; + f = ( f + b )|0; + b = ( b + ( (c & d) ^ ( e & (c ^ d) ) ) + ( c>>>2 ^ c>>>13 ^ c>>>22 ^ c<<30 ^ c<<19 ^ c<<10 ) )|0; + + // 63 + w15 = ( ( w0>>>7 ^ w0>>>18 ^ w0>>>3 ^ w0<<25 ^ w0<<14 ) + ( w13>>>17 ^ w13>>>19 ^ w13>>>10 ^ w13<<15 ^ w13<<13 ) + w15 + w8 )|0; + a = ( w15 + a + ( f>>>6 ^ f>>>11 ^ f>>>25 ^ f<<26 ^ f<<21 ^ f<<7 ) + ( h ^ f & (g^h) ) + 0xc67178f2 )|0; + e = ( e + a )|0; + a = ( a + ( (b & c) ^ ( d & (b ^ c) ) ) + ( b>>>2 ^ b>>>13 ^ b>>>22 ^ b<<30 ^ b<<19 ^ b<<10 ) )|0; + + H0 = ( H0 + a )|0; + H1 = ( H1 + b )|0; + H2 = ( H2 + c )|0; + H3 = ( H3 + d )|0; + H4 = ( H4 + e )|0; + H5 = ( H5 + f )|0; + H6 = ( H6 + g )|0; + H7 = ( H7 + h )|0; + } + + function _core_heap ( offset ) { + offset = offset|0; + + _core( + HEAP[offset|0]<<24 | HEAP[offset|1]<<16 | HEAP[offset|2]<<8 | HEAP[offset|3], + HEAP[offset|4]<<24 | HEAP[offset|5]<<16 | HEAP[offset|6]<<8 | HEAP[offset|7], + HEAP[offset|8]<<24 | HEAP[offset|9]<<16 | HEAP[offset|10]<<8 | HEAP[offset|11], + HEAP[offset|12]<<24 | HEAP[offset|13]<<16 | HEAP[offset|14]<<8 | HEAP[offset|15], + HEAP[offset|16]<<24 | HEAP[offset|17]<<16 | HEAP[offset|18]<<8 | HEAP[offset|19], + HEAP[offset|20]<<24 | HEAP[offset|21]<<16 | HEAP[offset|22]<<8 | HEAP[offset|23], + HEAP[offset|24]<<24 | HEAP[offset|25]<<16 | HEAP[offset|26]<<8 | HEAP[offset|27], + HEAP[offset|28]<<24 | HEAP[offset|29]<<16 | HEAP[offset|30]<<8 | HEAP[offset|31], + HEAP[offset|32]<<24 | HEAP[offset|33]<<16 | HEAP[offset|34]<<8 | HEAP[offset|35], + HEAP[offset|36]<<24 | HEAP[offset|37]<<16 | HEAP[offset|38]<<8 | HEAP[offset|39], + HEAP[offset|40]<<24 | HEAP[offset|41]<<16 | HEAP[offset|42]<<8 | HEAP[offset|43], + HEAP[offset|44]<<24 | HEAP[offset|45]<<16 | HEAP[offset|46]<<8 | HEAP[offset|47], + HEAP[offset|48]<<24 | HEAP[offset|49]<<16 | HEAP[offset|50]<<8 | HEAP[offset|51], + HEAP[offset|52]<<24 | HEAP[offset|53]<<16 | HEAP[offset|54]<<8 | HEAP[offset|55], + HEAP[offset|56]<<24 | HEAP[offset|57]<<16 | HEAP[offset|58]<<8 | HEAP[offset|59], + HEAP[offset|60]<<24 | HEAP[offset|61]<<16 | HEAP[offset|62]<<8 | HEAP[offset|63] + ); + } + + // offset — multiple of 32 + function _state_to_heap ( output ) { + output = output|0; + + HEAP[output|0] = H0>>>24; + HEAP[output|1] = H0>>>16&255; + HEAP[output|2] = H0>>>8&255; + HEAP[output|3] = H0&255; + HEAP[output|4] = H1>>>24; + HEAP[output|5] = H1>>>16&255; + HEAP[output|6] = H1>>>8&255; + HEAP[output|7] = H1&255; + HEAP[output|8] = H2>>>24; + HEAP[output|9] = H2>>>16&255; + HEAP[output|10] = H2>>>8&255; + HEAP[output|11] = H2&255; + HEAP[output|12] = H3>>>24; + HEAP[output|13] = H3>>>16&255; + HEAP[output|14] = H3>>>8&255; + HEAP[output|15] = H3&255; + HEAP[output|16] = H4>>>24; + HEAP[output|17] = H4>>>16&255; + HEAP[output|18] = H4>>>8&255; + HEAP[output|19] = H4&255; + HEAP[output|20] = H5>>>24; + HEAP[output|21] = H5>>>16&255; + HEAP[output|22] = H5>>>8&255; + HEAP[output|23] = H5&255; + HEAP[output|24] = H6>>>24; + HEAP[output|25] = H6>>>16&255; + HEAP[output|26] = H6>>>8&255; + HEAP[output|27] = H6&255; + HEAP[output|28] = H7>>>24; + HEAP[output|29] = H7>>>16&255; + HEAP[output|30] = H7>>>8&255; + HEAP[output|31] = H7&255; + } + + function reset () { + H0 = 0x6a09e667; + H1 = 0xbb67ae85; + H2 = 0x3c6ef372; + H3 = 0xa54ff53a; + H4 = 0x510e527f; + H5 = 0x9b05688c; + H6 = 0x1f83d9ab; + H7 = 0x5be0cd19; + TOTAL0 = TOTAL1 = 0; + } + + function init ( h0, h1, h2, h3, h4, h5, h6, h7, total0, total1 ) { + h0 = h0|0; + h1 = h1|0; + h2 = h2|0; + h3 = h3|0; + h4 = h4|0; + h5 = h5|0; + h6 = h6|0; + h7 = h7|0; + total0 = total0|0; + total1 = total1|0; + + H0 = h0; + H1 = h1; + H2 = h2; + H3 = h3; + H4 = h4; + H5 = h5; + H6 = h6; + H7 = h7; + TOTAL0 = total0; + TOTAL1 = total1; + } + + // offset — multiple of 64 + function process ( offset, length ) { + offset = offset|0; + length = length|0; + + var hashed = 0; + + if ( offset & 63 ) + return -1; + + while ( (length|0) >= 64 ) { + _core_heap(offset); + + offset = ( offset + 64 )|0; + length = ( length - 64 )|0; + + hashed = ( hashed + 64 )|0; + } + + TOTAL0 = ( TOTAL0 + hashed )|0; + if ( TOTAL0>>>0 < hashed>>>0 ) TOTAL1 = ( TOTAL1 + 1 )|0; + + return hashed|0; + } + + // offset — multiple of 64 + // output — multiple of 32 + function finish ( offset, length, output ) { + offset = offset|0; + length = length|0; + output = output|0; + + var hashed = 0, + i = 0; + + if ( offset & 63 ) + return -1; + + if ( ~output ) + if ( output & 31 ) + return -1; + + if ( (length|0) >= 64 ) { + hashed = process( offset, length )|0; + if ( (hashed|0) == -1 ) + return -1; + + offset = ( offset + hashed )|0; + length = ( length - hashed )|0; + } + + hashed = ( hashed + length )|0; + TOTAL0 = ( TOTAL0 + length )|0; + if ( TOTAL0>>>0 < length>>>0 ) TOTAL1 = ( TOTAL1 + 1 )|0; + + HEAP[offset|length] = 0x80; + + if ( (length|0) >= 56 ) { + for ( i = (length+1)|0; (i|0) < 64; i = (i+1)|0 ) + HEAP[offset|i] = 0x00; + + _core_heap(offset); + + length = 0; + + HEAP[offset|0] = 0; + } + + for ( i = (length+1)|0; (i|0) < 59; i = (i+1)|0 ) + HEAP[offset|i] = 0; + + HEAP[offset|56] = TOTAL1>>>21&255; + HEAP[offset|57] = TOTAL1>>>13&255; + HEAP[offset|58] = TOTAL1>>>5&255; + HEAP[offset|59] = TOTAL1<<3&255 | TOTAL0>>>29; + HEAP[offset|60] = TOTAL0>>>21&255; + HEAP[offset|61] = TOTAL0>>>13&255; + HEAP[offset|62] = TOTAL0>>>5&255; + HEAP[offset|63] = TOTAL0<<3&255; + _core_heap(offset); + + if ( ~output ) + _state_to_heap(output); + + return hashed|0; + } + + function hmac_reset () { + H0 = I0; + H1 = I1; + H2 = I2; + H3 = I3; + H4 = I4; + H5 = I5; + H6 = I6; + H7 = I7; + TOTAL0 = 64; + TOTAL1 = 0; + } + + function _hmac_opad () { + H0 = O0; + H1 = O1; + H2 = O2; + H3 = O3; + H4 = O4; + H5 = O5; + H6 = O6; + H7 = O7; + TOTAL0 = 64; + TOTAL1 = 0; + } + + function hmac_init ( p0, p1, p2, p3, p4, p5, p6, p7, p8, p9, p10, p11, p12, p13, p14, p15 ) { + p0 = p0|0; + p1 = p1|0; + p2 = p2|0; + p3 = p3|0; + p4 = p4|0; + p5 = p5|0; + p6 = p6|0; + p7 = p7|0; + p8 = p8|0; + p9 = p9|0; + p10 = p10|0; + p11 = p11|0; + p12 = p12|0; + p13 = p13|0; + p14 = p14|0; + p15 = p15|0; + + // opad + reset(); + _core( + p0 ^ 0x5c5c5c5c, + p1 ^ 0x5c5c5c5c, + p2 ^ 0x5c5c5c5c, + p3 ^ 0x5c5c5c5c, + p4 ^ 0x5c5c5c5c, + p5 ^ 0x5c5c5c5c, + p6 ^ 0x5c5c5c5c, + p7 ^ 0x5c5c5c5c, + p8 ^ 0x5c5c5c5c, + p9 ^ 0x5c5c5c5c, + p10 ^ 0x5c5c5c5c, + p11 ^ 0x5c5c5c5c, + p12 ^ 0x5c5c5c5c, + p13 ^ 0x5c5c5c5c, + p14 ^ 0x5c5c5c5c, + p15 ^ 0x5c5c5c5c + ); + O0 = H0; + O1 = H1; + O2 = H2; + O3 = H3; + O4 = H4; + O5 = H5; + O6 = H6; + O7 = H7; + + // ipad + reset(); + _core( + p0 ^ 0x36363636, + p1 ^ 0x36363636, + p2 ^ 0x36363636, + p3 ^ 0x36363636, + p4 ^ 0x36363636, + p5 ^ 0x36363636, + p6 ^ 0x36363636, + p7 ^ 0x36363636, + p8 ^ 0x36363636, + p9 ^ 0x36363636, + p10 ^ 0x36363636, + p11 ^ 0x36363636, + p12 ^ 0x36363636, + p13 ^ 0x36363636, + p14 ^ 0x36363636, + p15 ^ 0x36363636 + ); + I0 = H0; + I1 = H1; + I2 = H2; + I3 = H3; + I4 = H4; + I5 = H5; + I6 = H6; + I7 = H7; + + TOTAL0 = 64; + TOTAL1 = 0; + } + + // offset — multiple of 64 + // output — multiple of 32 + function hmac_finish ( offset, length, output ) { + offset = offset|0; + length = length|0; + output = output|0; + + var t0 = 0, t1 = 0, t2 = 0, t3 = 0, t4 = 0, t5 = 0, t6 = 0, t7 = 0, + hashed = 0; + + if ( offset & 63 ) + return -1; + + if ( ~output ) + if ( output & 31 ) + return -1; + + hashed = finish( offset, length, -1 )|0; + t0 = H0, t1 = H1, t2 = H2, t3 = H3, t4 = H4, t5 = H5, t6 = H6, t7 = H7; + + _hmac_opad(); + _core( t0, t1, t2, t3, t4, t5, t6, t7, 0x80000000, 0, 0, 0, 0, 0, 0, 768 ); + + if ( ~output ) + _state_to_heap(output); + + return hashed|0; + } + + // salt is assumed to be already processed + // offset — multiple of 64 + // output — multiple of 32 + function pbkdf2_generate_block ( offset, length, block, count, output ) { + offset = offset|0; + length = length|0; + block = block|0; + count = count|0; + output = output|0; + + var h0 = 0, h1 = 0, h2 = 0, h3 = 0, h4 = 0, h5 = 0, h6 = 0, h7 = 0, + t0 = 0, t1 = 0, t2 = 0, t3 = 0, t4 = 0, t5 = 0, t6 = 0, t7 = 0; + + if ( offset & 63 ) + return -1; + + if ( ~output ) + if ( output & 31 ) + return -1; + + // pad block number into heap + // FIXME probable OOB write + HEAP[(offset+length)|0] = block>>>24; + HEAP[(offset+length+1)|0] = block>>>16&255; + HEAP[(offset+length+2)|0] = block>>>8&255; + HEAP[(offset+length+3)|0] = block&255; + + // finish first iteration + hmac_finish( offset, (length+4)|0, -1 )|0; + h0 = t0 = H0, h1 = t1 = H1, h2 = t2 = H2, h3 = t3 = H3, h4 = t4 = H4, h5 = t5 = H5, h6 = t6 = H6, h7 = t7 = H7; + count = (count-1)|0; + + // perform the rest iterations + while ( (count|0) > 0 ) { + hmac_reset(); + _core( t0, t1, t2, t3, t4, t5, t6, t7, 0x80000000, 0, 0, 0, 0, 0, 0, 768 ); + t0 = H0, t1 = H1, t2 = H2, t3 = H3, t4 = H4, t5 = H5, t6 = H6, t7 = H7; + + _hmac_opad(); + _core( t0, t1, t2, t3, t4, t5, t6, t7, 0x80000000, 0, 0, 0, 0, 0, 0, 768 ); + t0 = H0, t1 = H1, t2 = H2, t3 = H3, t4 = H4, t5 = H5, t6 = H6, t7 = H7; + + h0 = h0 ^ H0; + h1 = h1 ^ H1; + h2 = h2 ^ H2; + h3 = h3 ^ H3; + h4 = h4 ^ H4; + h5 = h5 ^ H5; + h6 = h6 ^ H6; + h7 = h7 ^ H7; + + count = (count-1)|0; + } + + H0 = h0; + H1 = h1; + H2 = h2; + H3 = h3; + H4 = h4; + H5 = h5; + H6 = h6; + H7 = h7; + + if ( ~output ) + _state_to_heap(output); + + return 0; + } + + return { + // SHA256 + reset: reset, + init: init, + process: process, + finish: finish, + + // HMAC-SHA256 + hmac_reset: hmac_reset, + hmac_init: hmac_init, + hmac_finish: hmac_finish, + + // PBKDF2-HMAC-SHA256 + pbkdf2_generate_block: pbkdf2_generate_block + } + }; + + const _sha256_block_size = 64; + const _sha256_hash_size = 32; + const heap_pool$2 = []; + const asm_pool$2 = []; + class Sha256 extends Hash { + constructor() { + super(); + this.NAME = 'sha256'; + this.BLOCK_SIZE = _sha256_block_size; + this.HASH_SIZE = _sha256_hash_size; + this.acquire_asm(); + } + acquire_asm() { + if (this.heap === undefined || this.asm === undefined) { + this.heap = heap_pool$2.pop() || _heap_init(); + this.asm = asm_pool$2.pop() || sha256_asm({ Uint8Array: Uint8Array }, null, this.heap.buffer); + this.reset(); + } + return { heap: this.heap, asm: this.asm }; + } + release_asm() { + if (this.heap !== undefined && this.asm !== undefined) { + heap_pool$2.push(this.heap); + asm_pool$2.push(this.asm); + } + this.heap = undefined; + this.asm = undefined; + } + static bytes(data) { + return new Sha256().process(data).finish().result; + } + } + Sha256.NAME = 'sha256'; + + var minimalisticAssert = assert; + + function assert(val, msg) { + if (!val) + throw new Error(msg || 'Assertion failed'); + } + + assert.equal = function assertEqual(l, r, msg) { + if (l != r) + throw new Error(msg || ('Assertion failed: ' + l + ' != ' + r)); + }; + + var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {}; + + function createCommonjsModule(fn, module) { + return module = { exports: {} }, fn(module, module.exports), module.exports; + } + + function commonjsRequire () { + throw new Error('Dynamic requires are not currently supported by @rollup/plugin-commonjs'); + } + + var inherits_browser = createCommonjsModule(function (module) { + if (typeof Object.create === 'function') { + // implementation from standard node.js 'util' module + module.exports = function inherits(ctor, superCtor) { + ctor.super_ = superCtor; + ctor.prototype = Object.create(superCtor.prototype, { + constructor: { + value: ctor, + enumerable: false, + writable: true, + configurable: true + } + }); + }; + } else { + // old school shim for old browsers + module.exports = function inherits(ctor, superCtor) { + ctor.super_ = superCtor; + var TempCtor = function () {}; + TempCtor.prototype = superCtor.prototype; + ctor.prototype = new TempCtor(); + ctor.prototype.constructor = ctor; + }; + } + }); + + var inherits_1 = inherits_browser; + + function toArray(msg, enc) { + if (Array.isArray(msg)) + return msg.slice(); + if (!msg) + return []; + var res = []; + if (typeof msg === 'string') { + if (!enc) { + for (var i = 0; i < msg.length; i++) { + var c = msg.charCodeAt(i); + var hi = c >> 8; + var lo = c & 0xff; + if (hi) + res.push(hi, lo); + else + res.push(lo); + } + } else if (enc === 'hex') { + msg = msg.replace(/[^a-z0-9]+/ig, ''); + if (msg.length % 2 !== 0) + msg = '0' + msg; + for (i = 0; i < msg.length; i += 2) + res.push(parseInt(msg[i] + msg[i + 1], 16)); + } + } else { + for (i = 0; i < msg.length; i++) + res[i] = msg[i] | 0; + } + return res; + } + var toArray_1 = toArray; + + function toHex(msg) { + var res = ''; + for (var i = 0; i < msg.length; i++) + res += zero2(msg[i].toString(16)); + return res; + } + var toHex_1 = toHex; + + function htonl(w) { + var res = (w >>> 24) | + ((w >>> 8) & 0xff00) | + ((w << 8) & 0xff0000) | + ((w & 0xff) << 24); + return res >>> 0; + } + var htonl_1 = htonl; + + function toHex32(msg, endian) { + var res = ''; + for (var i = 0; i < msg.length; i++) { + var w = msg[i]; + if (endian === 'little') + w = htonl(w); + res += zero8(w.toString(16)); + } + return res; + } + var toHex32_1 = toHex32; + + function zero2(word) { + if (word.length === 1) + return '0' + word; + else + return word; + } + var zero2_1 = zero2; + + function zero8(word) { + if (word.length === 7) + return '0' + word; + else if (word.length === 6) + return '00' + word; + else if (word.length === 5) + return '000' + word; + else if (word.length === 4) + return '0000' + word; + else if (word.length === 3) + return '00000' + word; + else if (word.length === 2) + return '000000' + word; + else if (word.length === 1) + return '0000000' + word; + else + return word; + } + var zero8_1 = zero8; + + function join32(msg, start, end, endian) { + var len = end - start; + minimalisticAssert(len % 4 === 0); + var res = new Array(len / 4); + for (var i = 0, k = start; i < res.length; i++, k += 4) { + var w; + if (endian === 'big') + w = (msg[k] << 24) | (msg[k + 1] << 16) | (msg[k + 2] << 8) | msg[k + 3]; + else + w = (msg[k + 3] << 24) | (msg[k + 2] << 16) | (msg[k + 1] << 8) | msg[k]; + res[i] = w >>> 0; + } + return res; + } + var join32_1 = join32; + + function split32(msg, endian) { + var res = new Array(msg.length * 4); + for (var i = 0, k = 0; i < msg.length; i++, k += 4) { + var m = msg[i]; + if (endian === 'big') { + res[k] = m >>> 24; + res[k + 1] = (m >>> 16) & 0xff; + res[k + 2] = (m >>> 8) & 0xff; + res[k + 3] = m & 0xff; + } else { + res[k + 3] = m >>> 24; + res[k + 2] = (m >>> 16) & 0xff; + res[k + 1] = (m >>> 8) & 0xff; + res[k] = m & 0xff; + } + } + return res; + } + var split32_1 = split32; + + function rotr32(w, b) { + return (w >>> b) | (w << (32 - b)); + } + var rotr32_1 = rotr32; + + function rotl32(w, b) { + return (w << b) | (w >>> (32 - b)); + } + var rotl32_1 = rotl32; + + function sum32(a, b) { + return (a + b) >>> 0; + } + var sum32_1 = sum32; + + function sum32_3(a, b, c) { + return (a + b + c) >>> 0; + } + var sum32_3_1 = sum32_3; + + function sum32_4(a, b, c, d) { + return (a + b + c + d) >>> 0; + } + var sum32_4_1 = sum32_4; + + function sum32_5(a, b, c, d, e) { + return (a + b + c + d + e) >>> 0; + } + var sum32_5_1 = sum32_5; + + function sum64(buf, pos, ah, al) { + var bh = buf[pos]; + var bl = buf[pos + 1]; + + var lo = (al + bl) >>> 0; + var hi = (lo < al ? 1 : 0) + ah + bh; + buf[pos] = hi >>> 0; + buf[pos + 1] = lo; + } + var sum64_1 = sum64; + + function sum64_hi(ah, al, bh, bl) { + var lo = (al + bl) >>> 0; + var hi = (lo < al ? 1 : 0) + ah + bh; + return hi >>> 0; + } + var sum64_hi_1 = sum64_hi; + + function sum64_lo(ah, al, bh, bl) { + var lo = al + bl; + return lo >>> 0; + } + var sum64_lo_1 = sum64_lo; + + function sum64_4_hi(ah, al, bh, bl, ch, cl, dh, dl) { + var carry = 0; + var lo = al; + lo = (lo + bl) >>> 0; + carry += lo < al ? 1 : 0; + lo = (lo + cl) >>> 0; + carry += lo < cl ? 1 : 0; + lo = (lo + dl) >>> 0; + carry += lo < dl ? 1 : 0; + + var hi = ah + bh + ch + dh + carry; + return hi >>> 0; + } + var sum64_4_hi_1 = sum64_4_hi; + + function sum64_4_lo(ah, al, bh, bl, ch, cl, dh, dl) { + var lo = al + bl + cl + dl; + return lo >>> 0; + } + var sum64_4_lo_1 = sum64_4_lo; + + function sum64_5_hi(ah, al, bh, bl, ch, cl, dh, dl, eh, el) { + var carry = 0; + var lo = al; + lo = (lo + bl) >>> 0; + carry += lo < al ? 1 : 0; + lo = (lo + cl) >>> 0; + carry += lo < cl ? 1 : 0; + lo = (lo + dl) >>> 0; + carry += lo < dl ? 1 : 0; + lo = (lo + el) >>> 0; + carry += lo < el ? 1 : 0; + + var hi = ah + bh + ch + dh + eh + carry; + return hi >>> 0; + } + var sum64_5_hi_1 = sum64_5_hi; + + function sum64_5_lo(ah, al, bh, bl, ch, cl, dh, dl, eh, el) { + var lo = al + bl + cl + dl + el; + + return lo >>> 0; + } + var sum64_5_lo_1 = sum64_5_lo; + + function rotr64_hi(ah, al, num) { + var r = (al << (32 - num)) | (ah >>> num); + return r >>> 0; + } + var rotr64_hi_1 = rotr64_hi; + + function rotr64_lo(ah, al, num) { + var r = (ah << (32 - num)) | (al >>> num); + return r >>> 0; + } + var rotr64_lo_1 = rotr64_lo; + + function shr64_hi(ah, al, num) { + return ah >>> num; + } + var shr64_hi_1 = shr64_hi; + + function shr64_lo(ah, al, num) { + var r = (ah << (32 - num)) | (al >>> num); + return r >>> 0; + } + var shr64_lo_1 = shr64_lo; + + var utils = { + inherits: inherits_1, + toArray: toArray_1, + toHex: toHex_1, + htonl: htonl_1, + toHex32: toHex32_1, + zero2: zero2_1, + zero8: zero8_1, + join32: join32_1, + split32: split32_1, + rotr32: rotr32_1, + rotl32: rotl32_1, + sum32: sum32_1, + sum32_3: sum32_3_1, + sum32_4: sum32_4_1, + sum32_5: sum32_5_1, + sum64: sum64_1, + sum64_hi: sum64_hi_1, + sum64_lo: sum64_lo_1, + sum64_4_hi: sum64_4_hi_1, + sum64_4_lo: sum64_4_lo_1, + sum64_5_hi: sum64_5_hi_1, + sum64_5_lo: sum64_5_lo_1, + rotr64_hi: rotr64_hi_1, + rotr64_lo: rotr64_lo_1, + shr64_hi: shr64_hi_1, + shr64_lo: shr64_lo_1 + }; + + function BlockHash() { + this.pending = null; + this.pendingTotal = 0; + this.blockSize = this.constructor.blockSize; + this.outSize = this.constructor.outSize; + this.hmacStrength = this.constructor.hmacStrength; + this.padLength = this.constructor.padLength / 8; + this.endian = 'big'; + + this._delta8 = this.blockSize / 8; + this._delta32 = this.blockSize / 32; + } + var BlockHash_1 = BlockHash; + + BlockHash.prototype.update = function update(msg, enc) { + // Convert message to array, pad it, and join into 32bit blocks + msg = utils.toArray(msg, enc); + if (!this.pending) + this.pending = msg; + else + this.pending = this.pending.concat(msg); + this.pendingTotal += msg.length; + + // Enough data, try updating + if (this.pending.length >= this._delta8) { + msg = this.pending; + + // Process pending data in blocks + var r = msg.length % this._delta8; + this.pending = msg.slice(msg.length - r, msg.length); + if (this.pending.length === 0) + this.pending = null; + + msg = utils.join32(msg, 0, msg.length - r, this.endian); + for (var i = 0; i < msg.length; i += this._delta32) + this._update(msg, i, i + this._delta32); + } + + return this; + }; + + BlockHash.prototype.digest = function digest(enc) { + this.update(this._pad()); + minimalisticAssert(this.pending === null); + + return this._digest(enc); + }; + + BlockHash.prototype._pad = function pad() { + var len = this.pendingTotal; + var bytes = this._delta8; + var k = bytes - ((len + this.padLength) % bytes); + var res = new Array(k + this.padLength); + res[0] = 0x80; + for (var i = 1; i < k; i++) + res[i] = 0; + + // Append length + len <<= 3; + if (this.endian === 'big') { + for (var t = 8; t < this.padLength; t++) + res[i++] = 0; + + res[i++] = 0; + res[i++] = 0; + res[i++] = 0; + res[i++] = 0; + res[i++] = (len >>> 24) & 0xff; + res[i++] = (len >>> 16) & 0xff; + res[i++] = (len >>> 8) & 0xff; + res[i++] = len & 0xff; + } else { + res[i++] = len & 0xff; + res[i++] = (len >>> 8) & 0xff; + res[i++] = (len >>> 16) & 0xff; + res[i++] = (len >>> 24) & 0xff; + res[i++] = 0; + res[i++] = 0; + res[i++] = 0; + res[i++] = 0; + + for (t = 8; t < this.padLength; t++) + res[i++] = 0; + } + + return res; + }; + + var common = { + BlockHash: BlockHash_1 + }; + + var rotr32$1 = utils.rotr32; + + function ft_1(s, x, y, z) { + if (s === 0) + return ch32(x, y, z); + if (s === 1 || s === 3) + return p32(x, y, z); + if (s === 2) + return maj32(x, y, z); + } + var ft_1_1 = ft_1; + + function ch32(x, y, z) { + return (x & y) ^ ((~x) & z); + } + var ch32_1 = ch32; + + function maj32(x, y, z) { + return (x & y) ^ (x & z) ^ (y & z); + } + var maj32_1 = maj32; + + function p32(x, y, z) { + return x ^ y ^ z; + } + var p32_1 = p32; + + function s0_256(x) { + return rotr32$1(x, 2) ^ rotr32$1(x, 13) ^ rotr32$1(x, 22); + } + var s0_256_1 = s0_256; + + function s1_256(x) { + return rotr32$1(x, 6) ^ rotr32$1(x, 11) ^ rotr32$1(x, 25); + } + var s1_256_1 = s1_256; + + function g0_256(x) { + return rotr32$1(x, 7) ^ rotr32$1(x, 18) ^ (x >>> 3); + } + var g0_256_1 = g0_256; + + function g1_256(x) { + return rotr32$1(x, 17) ^ rotr32$1(x, 19) ^ (x >>> 10); + } + var g1_256_1 = g1_256; + + var common$1 = { + ft_1: ft_1_1, + ch32: ch32_1, + maj32: maj32_1, + p32: p32_1, + s0_256: s0_256_1, + s1_256: s1_256_1, + g0_256: g0_256_1, + g1_256: g1_256_1 + }; + + var sum32$1 = utils.sum32; + var sum32_4$1 = utils.sum32_4; + var sum32_5$1 = utils.sum32_5; + var ch32$1 = common$1.ch32; + var maj32$1 = common$1.maj32; + var s0_256$1 = common$1.s0_256; + var s1_256$1 = common$1.s1_256; + var g0_256$1 = common$1.g0_256; + var g1_256$1 = common$1.g1_256; + + var BlockHash$1 = common.BlockHash; + + var sha256_K = [ + 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, + 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5, + 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, + 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, + 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, + 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, + 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, + 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967, + 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, + 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, + 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, + 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070, + 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, + 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3, + 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, + 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2 + ]; + + function SHA256() { + if (!(this instanceof SHA256)) + return new SHA256(); + + BlockHash$1.call(this); + this.h = [ + 0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a, + 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19 + ]; + this.k = sha256_K; + this.W = new Array(64); + } + utils.inherits(SHA256, BlockHash$1); + var _256 = SHA256; + + SHA256.blockSize = 512; + SHA256.outSize = 256; + SHA256.hmacStrength = 192; + SHA256.padLength = 64; + + SHA256.prototype._update = function _update(msg, start) { + var W = this.W; + + for (var i = 0; i < 16; i++) + W[i] = msg[start + i]; + for (; i < W.length; i++) + W[i] = sum32_4$1(g1_256$1(W[i - 2]), W[i - 7], g0_256$1(W[i - 15]), W[i - 16]); + + var a = this.h[0]; + var b = this.h[1]; + var c = this.h[2]; + var d = this.h[3]; + var e = this.h[4]; + var f = this.h[5]; + var g = this.h[6]; + var h = this.h[7]; + + minimalisticAssert(this.k.length === W.length); + for (i = 0; i < W.length; i++) { + var T1 = sum32_5$1(h, s1_256$1(e), ch32$1(e, f, g), this.k[i], W[i]); + var T2 = sum32$1(s0_256$1(a), maj32$1(a, b, c)); + h = g; + g = f; + f = e; + e = sum32$1(d, T1); + d = c; + c = b; + b = a; + a = sum32$1(T1, T2); + } + + this.h[0] = sum32$1(this.h[0], a); + this.h[1] = sum32$1(this.h[1], b); + this.h[2] = sum32$1(this.h[2], c); + this.h[3] = sum32$1(this.h[3], d); + this.h[4] = sum32$1(this.h[4], e); + this.h[5] = sum32$1(this.h[5], f); + this.h[6] = sum32$1(this.h[6], g); + this.h[7] = sum32$1(this.h[7], h); + }; + + SHA256.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h, 'big'); + else + return utils.split32(this.h, 'big'); + }; + + function SHA224() { + if (!(this instanceof SHA224)) + return new SHA224(); + + _256.call(this); + this.h = [ + 0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939, + 0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4 ]; + } + utils.inherits(SHA224, _256); + var _224 = SHA224; + + SHA224.blockSize = 512; + SHA224.outSize = 224; + SHA224.hmacStrength = 192; + SHA224.padLength = 64; + + SHA224.prototype._digest = function digest(enc) { + // Just truncate output + if (enc === 'hex') + return utils.toHex32(this.h.slice(0, 7), 'big'); + else + return utils.split32(this.h.slice(0, 7), 'big'); + }; + + var rotr64_hi$1 = utils.rotr64_hi; + var rotr64_lo$1 = utils.rotr64_lo; + var shr64_hi$1 = utils.shr64_hi; + var shr64_lo$1 = utils.shr64_lo; + var sum64$1 = utils.sum64; + var sum64_hi$1 = utils.sum64_hi; + var sum64_lo$1 = utils.sum64_lo; + var sum64_4_hi$1 = utils.sum64_4_hi; + var sum64_4_lo$1 = utils.sum64_4_lo; + var sum64_5_hi$1 = utils.sum64_5_hi; + var sum64_5_lo$1 = utils.sum64_5_lo; + + var BlockHash$2 = common.BlockHash; + + var sha512_K = [ + 0x428a2f98, 0xd728ae22, 0x71374491, 0x23ef65cd, + 0xb5c0fbcf, 0xec4d3b2f, 0xe9b5dba5, 0x8189dbbc, + 0x3956c25b, 0xf348b538, 0x59f111f1, 0xb605d019, + 0x923f82a4, 0xaf194f9b, 0xab1c5ed5, 0xda6d8118, + 0xd807aa98, 0xa3030242, 0x12835b01, 0x45706fbe, + 0x243185be, 0x4ee4b28c, 0x550c7dc3, 0xd5ffb4e2, + 0x72be5d74, 0xf27b896f, 0x80deb1fe, 0x3b1696b1, + 0x9bdc06a7, 0x25c71235, 0xc19bf174, 0xcf692694, + 0xe49b69c1, 0x9ef14ad2, 0xefbe4786, 0x384f25e3, + 0x0fc19dc6, 0x8b8cd5b5, 0x240ca1cc, 0x77ac9c65, + 0x2de92c6f, 0x592b0275, 0x4a7484aa, 0x6ea6e483, + 0x5cb0a9dc, 0xbd41fbd4, 0x76f988da, 0x831153b5, + 0x983e5152, 0xee66dfab, 0xa831c66d, 0x2db43210, + 0xb00327c8, 0x98fb213f, 0xbf597fc7, 0xbeef0ee4, + 0xc6e00bf3, 0x3da88fc2, 0xd5a79147, 0x930aa725, + 0x06ca6351, 0xe003826f, 0x14292967, 0x0a0e6e70, + 0x27b70a85, 0x46d22ffc, 0x2e1b2138, 0x5c26c926, + 0x4d2c6dfc, 0x5ac42aed, 0x53380d13, 0x9d95b3df, + 0x650a7354, 0x8baf63de, 0x766a0abb, 0x3c77b2a8, + 0x81c2c92e, 0x47edaee6, 0x92722c85, 0x1482353b, + 0xa2bfe8a1, 0x4cf10364, 0xa81a664b, 0xbc423001, + 0xc24b8b70, 0xd0f89791, 0xc76c51a3, 0x0654be30, + 0xd192e819, 0xd6ef5218, 0xd6990624, 0x5565a910, + 0xf40e3585, 0x5771202a, 0x106aa070, 0x32bbd1b8, + 0x19a4c116, 0xb8d2d0c8, 0x1e376c08, 0x5141ab53, + 0x2748774c, 0xdf8eeb99, 0x34b0bcb5, 0xe19b48a8, + 0x391c0cb3, 0xc5c95a63, 0x4ed8aa4a, 0xe3418acb, + 0x5b9cca4f, 0x7763e373, 0x682e6ff3, 0xd6b2b8a3, + 0x748f82ee, 0x5defb2fc, 0x78a5636f, 0x43172f60, + 0x84c87814, 0xa1f0ab72, 0x8cc70208, 0x1a6439ec, + 0x90befffa, 0x23631e28, 0xa4506ceb, 0xde82bde9, + 0xbef9a3f7, 0xb2c67915, 0xc67178f2, 0xe372532b, + 0xca273ece, 0xea26619c, 0xd186b8c7, 0x21c0c207, + 0xeada7dd6, 0xcde0eb1e, 0xf57d4f7f, 0xee6ed178, + 0x06f067aa, 0x72176fba, 0x0a637dc5, 0xa2c898a6, + 0x113f9804, 0xbef90dae, 0x1b710b35, 0x131c471b, + 0x28db77f5, 0x23047d84, 0x32caab7b, 0x40c72493, + 0x3c9ebe0a, 0x15c9bebc, 0x431d67c4, 0x9c100d4c, + 0x4cc5d4be, 0xcb3e42b6, 0x597f299c, 0xfc657e2a, + 0x5fcb6fab, 0x3ad6faec, 0x6c44198c, 0x4a475817 + ]; + + function SHA512() { + if (!(this instanceof SHA512)) + return new SHA512(); + + BlockHash$2.call(this); + this.h = [ + 0x6a09e667, 0xf3bcc908, + 0xbb67ae85, 0x84caa73b, + 0x3c6ef372, 0xfe94f82b, + 0xa54ff53a, 0x5f1d36f1, + 0x510e527f, 0xade682d1, + 0x9b05688c, 0x2b3e6c1f, + 0x1f83d9ab, 0xfb41bd6b, + 0x5be0cd19, 0x137e2179 ]; + this.k = sha512_K; + this.W = new Array(160); + } + utils.inherits(SHA512, BlockHash$2); + var _512 = SHA512; + + SHA512.blockSize = 1024; + SHA512.outSize = 512; + SHA512.hmacStrength = 192; + SHA512.padLength = 128; + + SHA512.prototype._prepareBlock = function _prepareBlock(msg, start) { + var W = this.W; + + // 32 x 32bit words + for (var i = 0; i < 32; i++) + W[i] = msg[start + i]; + for (; i < W.length; i += 2) { + var c0_hi = g1_512_hi(W[i - 4], W[i - 3]); // i - 2 + var c0_lo = g1_512_lo(W[i - 4], W[i - 3]); + var c1_hi = W[i - 14]; // i - 7 + var c1_lo = W[i - 13]; + var c2_hi = g0_512_hi(W[i - 30], W[i - 29]); // i - 15 + var c2_lo = g0_512_lo(W[i - 30], W[i - 29]); + var c3_hi = W[i - 32]; // i - 16 + var c3_lo = W[i - 31]; + + W[i] = sum64_4_hi$1( + c0_hi, c0_lo, + c1_hi, c1_lo, + c2_hi, c2_lo, + c3_hi, c3_lo); + W[i + 1] = sum64_4_lo$1( + c0_hi, c0_lo, + c1_hi, c1_lo, + c2_hi, c2_lo, + c3_hi, c3_lo); + } + }; + + SHA512.prototype._update = function _update(msg, start) { + this._prepareBlock(msg, start); + + var W = this.W; + + var ah = this.h[0]; + var al = this.h[1]; + var bh = this.h[2]; + var bl = this.h[3]; + var ch = this.h[4]; + var cl = this.h[5]; + var dh = this.h[6]; + var dl = this.h[7]; + var eh = this.h[8]; + var el = this.h[9]; + var fh = this.h[10]; + var fl = this.h[11]; + var gh = this.h[12]; + var gl = this.h[13]; + var hh = this.h[14]; + var hl = this.h[15]; + + minimalisticAssert(this.k.length === W.length); + for (var i = 0; i < W.length; i += 2) { + var c0_hi = hh; + var c0_lo = hl; + var c1_hi = s1_512_hi(eh, el); + var c1_lo = s1_512_lo(eh, el); + var c2_hi = ch64_hi(eh, el, fh, fl, gh); + var c2_lo = ch64_lo(eh, el, fh, fl, gh, gl); + var c3_hi = this.k[i]; + var c3_lo = this.k[i + 1]; + var c4_hi = W[i]; + var c4_lo = W[i + 1]; + + var T1_hi = sum64_5_hi$1( + c0_hi, c0_lo, + c1_hi, c1_lo, + c2_hi, c2_lo, + c3_hi, c3_lo, + c4_hi, c4_lo); + var T1_lo = sum64_5_lo$1( + c0_hi, c0_lo, + c1_hi, c1_lo, + c2_hi, c2_lo, + c3_hi, c3_lo, + c4_hi, c4_lo); + + c0_hi = s0_512_hi(ah, al); + c0_lo = s0_512_lo(ah, al); + c1_hi = maj64_hi(ah, al, bh, bl, ch); + c1_lo = maj64_lo(ah, al, bh, bl, ch, cl); + + var T2_hi = sum64_hi$1(c0_hi, c0_lo, c1_hi, c1_lo); + var T2_lo = sum64_lo$1(c0_hi, c0_lo, c1_hi, c1_lo); + + hh = gh; + hl = gl; + + gh = fh; + gl = fl; + + fh = eh; + fl = el; + + eh = sum64_hi$1(dh, dl, T1_hi, T1_lo); + el = sum64_lo$1(dl, dl, T1_hi, T1_lo); + + dh = ch; + dl = cl; + + ch = bh; + cl = bl; + + bh = ah; + bl = al; + + ah = sum64_hi$1(T1_hi, T1_lo, T2_hi, T2_lo); + al = sum64_lo$1(T1_hi, T1_lo, T2_hi, T2_lo); + } + + sum64$1(this.h, 0, ah, al); + sum64$1(this.h, 2, bh, bl); + sum64$1(this.h, 4, ch, cl); + sum64$1(this.h, 6, dh, dl); + sum64$1(this.h, 8, eh, el); + sum64$1(this.h, 10, fh, fl); + sum64$1(this.h, 12, gh, gl); + sum64$1(this.h, 14, hh, hl); + }; + + SHA512.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h, 'big'); + else + return utils.split32(this.h, 'big'); + }; + + function ch64_hi(xh, xl, yh, yl, zh) { + var r = (xh & yh) ^ ((~xh) & zh); + if (r < 0) + r += 0x100000000; + return r; + } + + function ch64_lo(xh, xl, yh, yl, zh, zl) { + var r = (xl & yl) ^ ((~xl) & zl); + if (r < 0) + r += 0x100000000; + return r; + } + + function maj64_hi(xh, xl, yh, yl, zh) { + var r = (xh & yh) ^ (xh & zh) ^ (yh & zh); + if (r < 0) + r += 0x100000000; + return r; + } + + function maj64_lo(xh, xl, yh, yl, zh, zl) { + var r = (xl & yl) ^ (xl & zl) ^ (yl & zl); + if (r < 0) + r += 0x100000000; + return r; + } + + function s0_512_hi(xh, xl) { + var c0_hi = rotr64_hi$1(xh, xl, 28); + var c1_hi = rotr64_hi$1(xl, xh, 2); // 34 + var c2_hi = rotr64_hi$1(xl, xh, 7); // 39 + + var r = c0_hi ^ c1_hi ^ c2_hi; + if (r < 0) + r += 0x100000000; + return r; + } + + function s0_512_lo(xh, xl) { + var c0_lo = rotr64_lo$1(xh, xl, 28); + var c1_lo = rotr64_lo$1(xl, xh, 2); // 34 + var c2_lo = rotr64_lo$1(xl, xh, 7); // 39 + + var r = c0_lo ^ c1_lo ^ c2_lo; + if (r < 0) + r += 0x100000000; + return r; + } + + function s1_512_hi(xh, xl) { + var c0_hi = rotr64_hi$1(xh, xl, 14); + var c1_hi = rotr64_hi$1(xh, xl, 18); + var c2_hi = rotr64_hi$1(xl, xh, 9); // 41 + + var r = c0_hi ^ c1_hi ^ c2_hi; + if (r < 0) + r += 0x100000000; + return r; + } + + function s1_512_lo(xh, xl) { + var c0_lo = rotr64_lo$1(xh, xl, 14); + var c1_lo = rotr64_lo$1(xh, xl, 18); + var c2_lo = rotr64_lo$1(xl, xh, 9); // 41 + + var r = c0_lo ^ c1_lo ^ c2_lo; + if (r < 0) + r += 0x100000000; + return r; + } + + function g0_512_hi(xh, xl) { + var c0_hi = rotr64_hi$1(xh, xl, 1); + var c1_hi = rotr64_hi$1(xh, xl, 8); + var c2_hi = shr64_hi$1(xh, xl, 7); + + var r = c0_hi ^ c1_hi ^ c2_hi; + if (r < 0) + r += 0x100000000; + return r; + } + + function g0_512_lo(xh, xl) { + var c0_lo = rotr64_lo$1(xh, xl, 1); + var c1_lo = rotr64_lo$1(xh, xl, 8); + var c2_lo = shr64_lo$1(xh, xl, 7); + + var r = c0_lo ^ c1_lo ^ c2_lo; + if (r < 0) + r += 0x100000000; + return r; + } + + function g1_512_hi(xh, xl) { + var c0_hi = rotr64_hi$1(xh, xl, 19); + var c1_hi = rotr64_hi$1(xl, xh, 29); // 61 + var c2_hi = shr64_hi$1(xh, xl, 6); + + var r = c0_hi ^ c1_hi ^ c2_hi; + if (r < 0) + r += 0x100000000; + return r; + } + + function g1_512_lo(xh, xl) { + var c0_lo = rotr64_lo$1(xh, xl, 19); + var c1_lo = rotr64_lo$1(xl, xh, 29); // 61 + var c2_lo = shr64_lo$1(xh, xl, 6); + + var r = c0_lo ^ c1_lo ^ c2_lo; + if (r < 0) + r += 0x100000000; + return r; + } + + function SHA384() { + if (!(this instanceof SHA384)) + return new SHA384(); + + _512.call(this); + this.h = [ + 0xcbbb9d5d, 0xc1059ed8, + 0x629a292a, 0x367cd507, + 0x9159015a, 0x3070dd17, + 0x152fecd8, 0xf70e5939, + 0x67332667, 0xffc00b31, + 0x8eb44a87, 0x68581511, + 0xdb0c2e0d, 0x64f98fa7, + 0x47b5481d, 0xbefa4fa4 ]; + } + utils.inherits(SHA384, _512); + var _384 = SHA384; + + SHA384.blockSize = 1024; + SHA384.outSize = 384; + SHA384.hmacStrength = 192; + SHA384.padLength = 128; + + SHA384.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h.slice(0, 12), 'big'); + else + return utils.split32(this.h.slice(0, 12), 'big'); + }; + + var rotl32$1 = utils.rotl32; + var sum32$2 = utils.sum32; + var sum32_3$1 = utils.sum32_3; + var sum32_4$2 = utils.sum32_4; + var BlockHash$3 = common.BlockHash; + + function RIPEMD160() { + if (!(this instanceof RIPEMD160)) + return new RIPEMD160(); + + BlockHash$3.call(this); + + this.h = [ 0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476, 0xc3d2e1f0 ]; + this.endian = 'little'; + } + utils.inherits(RIPEMD160, BlockHash$3); + var ripemd160 = RIPEMD160; + + RIPEMD160.blockSize = 512; + RIPEMD160.outSize = 160; + RIPEMD160.hmacStrength = 192; + RIPEMD160.padLength = 64; + + RIPEMD160.prototype._update = function update(msg, start) { + var A = this.h[0]; + var B = this.h[1]; + var C = this.h[2]; + var D = this.h[3]; + var E = this.h[4]; + var Ah = A; + var Bh = B; + var Ch = C; + var Dh = D; + var Eh = E; + for (var j = 0; j < 80; j++) { + var T = sum32$2( + rotl32$1( + sum32_4$2(A, f(j, B, C, D), msg[r[j] + start], K(j)), + s[j]), + E); + A = E; + E = D; + D = rotl32$1(C, 10); + C = B; + B = T; + T = sum32$2( + rotl32$1( + sum32_4$2(Ah, f(79 - j, Bh, Ch, Dh), msg[rh[j] + start], Kh(j)), + sh[j]), + Eh); + Ah = Eh; + Eh = Dh; + Dh = rotl32$1(Ch, 10); + Ch = Bh; + Bh = T; + } + T = sum32_3$1(this.h[1], C, Dh); + this.h[1] = sum32_3$1(this.h[2], D, Eh); + this.h[2] = sum32_3$1(this.h[3], E, Ah); + this.h[3] = sum32_3$1(this.h[4], A, Bh); + this.h[4] = sum32_3$1(this.h[0], B, Ch); + this.h[0] = T; + }; + + RIPEMD160.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h, 'little'); + else + return utils.split32(this.h, 'little'); + }; + + function f(j, x, y, z) { + if (j <= 15) + return x ^ y ^ z; + else if (j <= 31) + return (x & y) | ((~x) & z); + else if (j <= 47) + return (x | (~y)) ^ z; + else if (j <= 63) + return (x & z) | (y & (~z)); + else + return x ^ (y | (~z)); + } + + function K(j) { + if (j <= 15) + return 0x00000000; + else if (j <= 31) + return 0x5a827999; + else if (j <= 47) + return 0x6ed9eba1; + else if (j <= 63) + return 0x8f1bbcdc; + else + return 0xa953fd4e; + } + + function Kh(j) { + if (j <= 15) + return 0x50a28be6; + else if (j <= 31) + return 0x5c4dd124; + else if (j <= 47) + return 0x6d703ef3; + else if (j <= 63) + return 0x7a6d76e9; + else + return 0x00000000; + } + + var r = [ + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, + 7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8, + 3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12, + 1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2, + 4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13 + ]; + + var rh = [ + 5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, + 6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2, + 15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13, + 8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14, + 12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11 + ]; + + var s = [ + 11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8, + 7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12, + 11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5, + 11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12, + 9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6 + ]; + + var sh = [ + 8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6, + 9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11, + 9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5, + 15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8, + 8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11 + ]; + + var ripemd = { + ripemd160: ripemd160 + }; + + /** + * A fast MD5 JavaScript implementation + * Copyright (c) 2012 Joseph Myers + * http://www.myersdaily.org/joseph/javascript/md5-text.html + * + * Permission to use, copy, modify, and distribute this software + * and its documentation for any purposes and without + * fee is hereby granted provided that this copyright notice + * appears in all copies. + * + * Of course, this soft is provided "as is" without express or implied + * warranty of any kind. + */ + + // MD5 Digest + async function md5(entree) { + const digest = md51(util.uint8ArrayToString(entree)); + return util.hexToUint8Array(hex(digest)); + } + + function md5cycle(x, k) { + let a = x[0]; + let b = x[1]; + let c = x[2]; + let d = x[3]; + + a = ff(a, b, c, d, k[0], 7, -680876936); + d = ff(d, a, b, c, k[1], 12, -389564586); + c = ff(c, d, a, b, k[2], 17, 606105819); + b = ff(b, c, d, a, k[3], 22, -1044525330); + a = ff(a, b, c, d, k[4], 7, -176418897); + d = ff(d, a, b, c, k[5], 12, 1200080426); + c = ff(c, d, a, b, k[6], 17, -1473231341); + b = ff(b, c, d, a, k[7], 22, -45705983); + a = ff(a, b, c, d, k[8], 7, 1770035416); + d = ff(d, a, b, c, k[9], 12, -1958414417); + c = ff(c, d, a, b, k[10], 17, -42063); + b = ff(b, c, d, a, k[11], 22, -1990404162); + a = ff(a, b, c, d, k[12], 7, 1804603682); + d = ff(d, a, b, c, k[13], 12, -40341101); + c = ff(c, d, a, b, k[14], 17, -1502002290); + b = ff(b, c, d, a, k[15], 22, 1236535329); + + a = gg(a, b, c, d, k[1], 5, -165796510); + d = gg(d, a, b, c, k[6], 9, -1069501632); + c = gg(c, d, a, b, k[11], 14, 643717713); + b = gg(b, c, d, a, k[0], 20, -373897302); + a = gg(a, b, c, d, k[5], 5, -701558691); + d = gg(d, a, b, c, k[10], 9, 38016083); + c = gg(c, d, a, b, k[15], 14, -660478335); + b = gg(b, c, d, a, k[4], 20, -405537848); + a = gg(a, b, c, d, k[9], 5, 568446438); + d = gg(d, a, b, c, k[14], 9, -1019803690); + c = gg(c, d, a, b, k[3], 14, -187363961); + b = gg(b, c, d, a, k[8], 20, 1163531501); + a = gg(a, b, c, d, k[13], 5, -1444681467); + d = gg(d, a, b, c, k[2], 9, -51403784); + c = gg(c, d, a, b, k[7], 14, 1735328473); + b = gg(b, c, d, a, k[12], 20, -1926607734); + + a = hh(a, b, c, d, k[5], 4, -378558); + d = hh(d, a, b, c, k[8], 11, -2022574463); + c = hh(c, d, a, b, k[11], 16, 1839030562); + b = hh(b, c, d, a, k[14], 23, -35309556); + a = hh(a, b, c, d, k[1], 4, -1530992060); + d = hh(d, a, b, c, k[4], 11, 1272893353); + c = hh(c, d, a, b, k[7], 16, -155497632); + b = hh(b, c, d, a, k[10], 23, -1094730640); + a = hh(a, b, c, d, k[13], 4, 681279174); + d = hh(d, a, b, c, k[0], 11, -358537222); + c = hh(c, d, a, b, k[3], 16, -722521979); + b = hh(b, c, d, a, k[6], 23, 76029189); + a = hh(a, b, c, d, k[9], 4, -640364487); + d = hh(d, a, b, c, k[12], 11, -421815835); + c = hh(c, d, a, b, k[15], 16, 530742520); + b = hh(b, c, d, a, k[2], 23, -995338651); + + a = ii(a, b, c, d, k[0], 6, -198630844); + d = ii(d, a, b, c, k[7], 10, 1126891415); + c = ii(c, d, a, b, k[14], 15, -1416354905); + b = ii(b, c, d, a, k[5], 21, -57434055); + a = ii(a, b, c, d, k[12], 6, 1700485571); + d = ii(d, a, b, c, k[3], 10, -1894986606); + c = ii(c, d, a, b, k[10], 15, -1051523); + b = ii(b, c, d, a, k[1], 21, -2054922799); + a = ii(a, b, c, d, k[8], 6, 1873313359); + d = ii(d, a, b, c, k[15], 10, -30611744); + c = ii(c, d, a, b, k[6], 15, -1560198380); + b = ii(b, c, d, a, k[13], 21, 1309151649); + a = ii(a, b, c, d, k[4], 6, -145523070); + d = ii(d, a, b, c, k[11], 10, -1120210379); + c = ii(c, d, a, b, k[2], 15, 718787259); + b = ii(b, c, d, a, k[9], 21, -343485551); + + x[0] = add32(a, x[0]); + x[1] = add32(b, x[1]); + x[2] = add32(c, x[2]); + x[3] = add32(d, x[3]); + } + + function cmn(q, a, b, x, s, t) { + a = add32(add32(a, q), add32(x, t)); + return add32((a << s) | (a >>> (32 - s)), b); + } + + function ff(a, b, c, d, x, s, t) { + return cmn((b & c) | ((~b) & d), a, b, x, s, t); + } + + function gg(a, b, c, d, x, s, t) { + return cmn((b & d) | (c & (~d)), a, b, x, s, t); + } + + function hh(a, b, c, d, x, s, t) { + return cmn(b ^ c ^ d, a, b, x, s, t); + } + + function ii(a, b, c, d, x, s, t) { + return cmn(c ^ (b | (~d)), a, b, x, s, t); + } + + function md51(s) { + const n = s.length; + const state = [1732584193, -271733879, -1732584194, 271733878]; + let i; + for (i = 64; i <= s.length; i += 64) { + md5cycle(state, md5blk(s.substring(i - 64, i))); + } + s = s.substring(i - 64); + const tail = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]; + for (i = 0; i < s.length; i++) { + tail[i >> 2] |= s.charCodeAt(i) << ((i % 4) << 3); + } + tail[i >> 2] |= 0x80 << ((i % 4) << 3); + if (i > 55) { + md5cycle(state, tail); + for (i = 0; i < 16; i++) { + tail[i] = 0; + } + } + tail[14] = n * 8; + md5cycle(state, tail); + return state; + } + + /* there needs to be support for Unicode here, + * unless we pretend that we can redefine the MD-5 + * algorithm for multi-byte characters (perhaps + * by adding every four 16-bit characters and + * shortening the sum to 32 bits). Otherwise + * I suggest performing MD-5 as if every character + * was two bytes--e.g., 0040 0025 = @%--but then + * how will an ordinary MD-5 sum be matched? + * There is no way to standardize text to something + * like UTF-8 before transformation; speed cost is + * utterly prohibitive. The JavaScript standard + * itself needs to look at this: it should start + * providing access to strings as preformed UTF-8 + * 8-bit unsigned value arrays. + */ + function md5blk(s) { /* I figured global was faster. */ + const md5blks = []; + let i; /* Andy King said do it this way. */ + for (i = 0; i < 64; i += 4) { + md5blks[i >> 2] = s.charCodeAt(i) + (s.charCodeAt(i + 1) << 8) + (s.charCodeAt(i + 2) << 16) + (s.charCodeAt(i + 3) << + 24); + } + return md5blks; + } + + const hex_chr = '0123456789abcdef'.split(''); + + function rhex(n) { + let s = ''; + let j = 0; + for (; j < 4; j++) { + s += hex_chr[(n >> (j * 8 + 4)) & 0x0F] + hex_chr[(n >> (j * 8)) & 0x0F]; + } + return s; + } + + function hex(x) { + for (let i = 0; i < x.length; i++) { + x[i] = rhex(x[i]); + } + return x.join(''); + } + + /* this function is much faster, + so if possible we use it. Some IEs + are the only ones I know of that + need the idiotic second function, + generated by an if clause. */ + + function add32(a, b) { + return (a + b) & 0xFFFFFFFF; + } + + /** + * @fileoverview Provides an interface to hashing functions available in Node.js or external libraries. + * @see {@link https://github.com/asmcrypto/asmcrypto.js|asmCrypto} + * @see {@link https://github.com/indutny/hash.js|hash.js} + * @module crypto/hash + * @private + */ + + const webCrypto = util.getWebCrypto(); + const nodeCrypto = util.getNodeCrypto(); + const nodeCryptoHashes = nodeCrypto && nodeCrypto.getHashes(); + + function nodeHash(type) { + if (!nodeCrypto || !nodeCryptoHashes.includes(type)) { + return; + } + return async function (data) { + const shasum = nodeCrypto.createHash(type); + return transform(data, value => { + shasum.update(value); + }, () => new Uint8Array(shasum.digest())); + }; + } + + function hashjsHash(hash, webCryptoHash) { + return async function(data, config$1 = config) { + if (isArrayStream(data)) { + data = await readToEnd(data); + } + if (!util.isStream(data) && webCrypto && webCryptoHash && data.length >= config$1.minBytesForWebCrypto) { + return new Uint8Array(await webCrypto.digest(webCryptoHash, data)); + } + const hashInstance = hash(); + return transform(data, value => { + hashInstance.update(value); + }, () => new Uint8Array(hashInstance.digest())); + }; + } + + function asmcryptoHash(hash, webCryptoHash) { + return async function(data, config$1 = config) { + if (isArrayStream(data)) { + data = await readToEnd(data); + } + if (util.isStream(data)) { + const hashInstance = new hash(); + return transform(data, value => { + hashInstance.process(value); + }, () => hashInstance.finish().result); + } else if (webCrypto && webCryptoHash && data.length >= config$1.minBytesForWebCrypto) { + return new Uint8Array(await webCrypto.digest(webCryptoHash, data)); + } else { + return hash.bytes(data); + } + }; + } + + const hashFunctions = { + md5: nodeHash('md5') || md5, + sha1: nodeHash('sha1') || asmcryptoHash(Sha1, 'SHA-1'), + sha224: nodeHash('sha224') || hashjsHash(_224), + sha256: nodeHash('sha256') || asmcryptoHash(Sha256, 'SHA-256'), + sha384: nodeHash('sha384') || hashjsHash(_384, 'SHA-384'), + sha512: nodeHash('sha512') || hashjsHash(_512, 'SHA-512'), // asmcrypto sha512 is huge. + ripemd: nodeHash('ripemd160') || hashjsHash(ripemd160) + }; + + var hash = { + + /** @see module:md5 */ + md5: hashFunctions.md5, + /** @see asmCrypto */ + sha1: hashFunctions.sha1, + /** @see hash.js */ + sha224: hashFunctions.sha224, + /** @see asmCrypto */ + sha256: hashFunctions.sha256, + /** @see hash.js */ + sha384: hashFunctions.sha384, + /** @see asmCrypto */ + sha512: hashFunctions.sha512, + /** @see hash.js */ + ripemd: hashFunctions.ripemd, + + /** + * Create a hash on the specified data using the specified algorithm + * @param {module:enums.hash} algo - Hash algorithm type (see {@link https://tools.ietf.org/html/rfc4880#section-9.4|RFC 4880 9.4}) + * @param {Uint8Array} data - Data to be hashed + * @returns {Promise} Hash value. + */ + digest: function(algo, data) { + switch (algo) { + case enums.hash.md5: + return this.md5(data); + case enums.hash.sha1: + return this.sha1(data); + case enums.hash.ripemd: + return this.ripemd(data); + case enums.hash.sha256: + return this.sha256(data); + case enums.hash.sha384: + return this.sha384(data); + case enums.hash.sha512: + return this.sha512(data); + case enums.hash.sha224: + return this.sha224(data); + default: + throw new Error('Invalid hash function.'); + } + }, + + /** + * Returns the hash size in bytes of the specified hash algorithm type + * @param {module:enums.hash} algo - Hash algorithm type (See {@link https://tools.ietf.org/html/rfc4880#section-9.4|RFC 4880 9.4}) + * @returns {Integer} Size in bytes of the resulting hash. + */ + getHashByteLength: function(algo) { + switch (algo) { + case enums.hash.md5: + return 16; + case enums.hash.sha1: + case enums.hash.ripemd: + return 20; + case enums.hash.sha256: + return 32; + case enums.hash.sha384: + return 48; + case enums.hash.sha512: + return 64; + case enums.hash.sha224: + return 28; + default: + throw new Error('Invalid hash algorithm.'); + } + } + }; + + class AES_CFB { + static encrypt(data, key, iv) { + return new AES_CFB(key, iv).encrypt(data); + } + static decrypt(data, key, iv) { + return new AES_CFB(key, iv).decrypt(data); + } + constructor(key, iv, aes) { + this.aes = aes ? aes : new AES(key, iv, true, 'CFB'); + delete this.aes.padding; + } + encrypt(data) { + const r1 = this.aes.AES_Encrypt_process(data); + const r2 = this.aes.AES_Encrypt_finish(); + return joinBytes(r1, r2); + } + decrypt(data) { + const r1 = this.aes.AES_Decrypt_process(data); + const r2 = this.aes.AES_Decrypt_finish(); + return joinBytes(r1, r2); + } + } + + /** + * Get implementation of the given cipher + * @param {enums.symmetric} algo + * @returns {Object} + * @throws {Error} on invalid algo + */ + function getCipher(algo) { + const algoName = enums.read(enums.symmetric, algo); + return cipher[algoName]; + } + + // Modified by ProtonTech AG + + const webCrypto$1 = util.getWebCrypto(); + const nodeCrypto$1 = util.getNodeCrypto(); + + const knownAlgos = nodeCrypto$1 ? nodeCrypto$1.getCiphers() : []; + const nodeAlgos = { + idea: knownAlgos.includes('idea-cfb') ? 'idea-cfb' : undefined, /* Unused, not implemented */ + tripledes: knownAlgos.includes('des-ede3-cfb') ? 'des-ede3-cfb' : undefined, + cast5: knownAlgos.includes('cast5-cfb') ? 'cast5-cfb' : undefined, + blowfish: knownAlgos.includes('bf-cfb') ? 'bf-cfb' : undefined, + aes128: knownAlgos.includes('aes-128-cfb') ? 'aes-128-cfb' : undefined, + aes192: knownAlgos.includes('aes-192-cfb') ? 'aes-192-cfb' : undefined, + aes256: knownAlgos.includes('aes-256-cfb') ? 'aes-256-cfb' : undefined + /* twofish is not implemented in OpenSSL */ + }; + + /** + * CFB encryption + * @param {enums.symmetric} algo - block cipher algorithm + * @param {Uint8Array} key + * @param {MaybeStream} plaintext + * @param {Uint8Array} iv + * @param {Object} config - full configuration, defaults to openpgp.config + * @returns MaybeStream + */ + async function encrypt(algo, key, plaintext, iv, config) { + const algoName = enums.read(enums.symmetric, algo); + if (util.getNodeCrypto() && nodeAlgos[algoName]) { // Node crypto library. + return nodeEncrypt(algo, key, plaintext, iv); + } + if (algoName.substr(0, 3) === 'aes') { + return aesEncrypt(algo, key, plaintext, iv, config); + } + + const Cipher = getCipher(algo); + const cipherfn = new Cipher(key); + const block_size = cipherfn.blockSize; + + const blockc = iv.slice(); + let pt = new Uint8Array(); + const process = chunk => { + if (chunk) { + pt = util.concatUint8Array([pt, chunk]); + } + const ciphertext = new Uint8Array(pt.length); + let i; + let j = 0; + while (chunk ? pt.length >= block_size : pt.length) { + const encblock = cipherfn.encrypt(blockc); + for (i = 0; i < block_size; i++) { + blockc[i] = pt[i] ^ encblock[i]; + ciphertext[j++] = blockc[i]; + } + pt = pt.subarray(block_size); + } + return ciphertext.subarray(0, j); + }; + return transform(plaintext, process, process); + } + + /** + * CFB decryption + * @param {enums.symmetric} algo - block cipher algorithm + * @param {Uint8Array} key + * @param {MaybeStream} ciphertext + * @param {Uint8Array} iv + * @returns MaybeStream + */ + async function decrypt(algo, key, ciphertext, iv) { + const algoName = enums.read(enums.symmetric, algo); + if (util.getNodeCrypto() && nodeAlgos[algoName]) { // Node crypto library. + return nodeDecrypt(algo, key, ciphertext, iv); + } + if (algoName.substr(0, 3) === 'aes') { + return aesDecrypt(algo, key, ciphertext, iv); + } + + const Cipher = getCipher(algo); + const cipherfn = new Cipher(key); + const block_size = cipherfn.blockSize; + + let blockp = iv; + let ct = new Uint8Array(); + const process = chunk => { + if (chunk) { + ct = util.concatUint8Array([ct, chunk]); + } + const plaintext = new Uint8Array(ct.length); + let i; + let j = 0; + while (chunk ? ct.length >= block_size : ct.length) { + const decblock = cipherfn.encrypt(blockp); + blockp = ct; + for (i = 0; i < block_size; i++) { + plaintext[j++] = blockp[i] ^ decblock[i]; + } + ct = ct.subarray(block_size); + } + return plaintext.subarray(0, j); + }; + return transform(ciphertext, process, process); + } + + function aesEncrypt(algo, key, pt, iv, config) { + if ( + util.getWebCrypto() && + key.length !== 24 && // Chrome doesn't support 192 bit keys, see https://www.chromium.org/blink/webcrypto#TOC-AES-support + !util.isStream(pt) && + pt.length >= 3000 * config.minBytesForWebCrypto // Default to a 3MB minimum. Chrome is pretty slow for small messages, see: https://bugs.chromium.org/p/chromium/issues/detail?id=701188#c2 + ) { // Web Crypto + return webEncrypt(algo, key, pt, iv); + } + // asm.js fallback + const cfb = new AES_CFB(key, iv); + return transform(pt, value => cfb.aes.AES_Encrypt_process(value), () => cfb.aes.AES_Encrypt_finish()); + } + + function aesDecrypt(algo, key, ct, iv) { + if (util.isStream(ct)) { + const cfb = new AES_CFB(key, iv); + return transform(ct, value => cfb.aes.AES_Decrypt_process(value), () => cfb.aes.AES_Decrypt_finish()); + } + return AES_CFB.decrypt(ct, key, iv); + } + + function xorMut(a, b) { + for (let i = 0; i < a.length; i++) { + a[i] = a[i] ^ b[i]; + } + } + + async function webEncrypt(algo, key, pt, iv) { + const ALGO = 'AES-CBC'; + const _key = await webCrypto$1.importKey('raw', key, { name: ALGO }, false, ['encrypt']); + const { blockSize } = getCipher(algo); + const cbc_pt = util.concatUint8Array([new Uint8Array(blockSize), pt]); + const ct = new Uint8Array(await webCrypto$1.encrypt({ name: ALGO, iv }, _key, cbc_pt)).subarray(0, pt.length); + xorMut(ct, pt); + return ct; + } + + function nodeEncrypt(algo, key, pt, iv) { + const algoName = enums.read(enums.symmetric, algo); + const cipherObj = new nodeCrypto$1.createCipheriv(nodeAlgos[algoName], key, iv); + return transform(pt, value => new Uint8Array(cipherObj.update(value))); + } + + function nodeDecrypt(algo, key, ct, iv) { + const algoName = enums.read(enums.symmetric, algo); + const decipherObj = new nodeCrypto$1.createDecipheriv(nodeAlgos[algoName], key, iv); + return transform(ct, value => new Uint8Array(decipherObj.update(value))); + } + + var cfb = /*#__PURE__*/Object.freeze({ + __proto__: null, + encrypt: encrypt, + decrypt: decrypt + }); + + class AES_CTR { + static encrypt(data, key, nonce) { + return new AES_CTR(key, nonce).encrypt(data); + } + static decrypt(data, key, nonce) { + return new AES_CTR(key, nonce).encrypt(data); + } + constructor(key, nonce, aes) { + this.aes = aes ? aes : new AES(key, undefined, false, 'CTR'); + delete this.aes.padding; + this.AES_CTR_set_options(nonce); + } + encrypt(data) { + const r1 = this.aes.AES_Encrypt_process(data); + const r2 = this.aes.AES_Encrypt_finish(); + return joinBytes(r1, r2); + } + decrypt(data) { + const r1 = this.aes.AES_Encrypt_process(data); + const r2 = this.aes.AES_Encrypt_finish(); + return joinBytes(r1, r2); + } + AES_CTR_set_options(nonce, counter, size) { + let { asm } = this.aes.acquire_asm(); + if (size !== undefined) { + if (size < 8 || size > 48) + throw new IllegalArgumentError('illegal counter size'); + let mask = Math.pow(2, size) - 1; + asm.set_mask(0, 0, (mask / 0x100000000) | 0, mask | 0); + } + else { + size = 48; + asm.set_mask(0, 0, 0xffff, 0xffffffff); + } + if (nonce !== undefined) { + let len = nonce.length; + if (!len || len > 16) + throw new IllegalArgumentError('illegal nonce size'); + let view = new DataView(new ArrayBuffer(16)); + new Uint8Array(view.buffer).set(nonce); + asm.set_nonce(view.getUint32(0), view.getUint32(4), view.getUint32(8), view.getUint32(12)); + } + else { + throw new Error('nonce is required'); + } + if (counter !== undefined) { + if (counter < 0 || counter >= Math.pow(2, size)) + throw new IllegalArgumentError('illegal counter value'); + asm.set_counter(0, 0, (counter / 0x100000000) | 0, counter | 0); + } + } + } + + class AES_CBC { + static encrypt(data, key, padding = true, iv) { + return new AES_CBC(key, iv, padding).encrypt(data); + } + static decrypt(data, key, padding = true, iv) { + return new AES_CBC(key, iv, padding).decrypt(data); + } + constructor(key, iv, padding = true, aes) { + this.aes = aes ? aes : new AES(key, iv, padding, 'CBC'); + } + encrypt(data) { + const r1 = this.aes.AES_Encrypt_process(data); + const r2 = this.aes.AES_Encrypt_finish(); + return joinBytes(r1, r2); + } + decrypt(data) { + const r1 = this.aes.AES_Decrypt_process(data); + const r2 = this.aes.AES_Decrypt_finish(); + return joinBytes(r1, r2); + } + } + + /** + * @fileoverview This module implements AES-CMAC on top of + * native AES-CBC using either the WebCrypto API or Node.js' crypto API. + * @module crypto/cmac + * @private + */ + + const webCrypto$2 = util.getWebCrypto(); + const nodeCrypto$2 = util.getNodeCrypto(); + + + /** + * This implementation of CMAC is based on the description of OMAC in + * http://web.cs.ucdavis.edu/~rogaway/papers/eax.pdf. As per that + * document: + * + * We have made a small modification to the OMAC algorithm as it was + * originally presented, changing one of its two constants. + * Specifically, the constant 4 at line 85 was the constant 1/2 (the + * multiplicative inverse of 2) in the original definition of OMAC [14]. + * The OMAC authors indicate that they will promulgate this modification + * [15], which slightly simplifies implementations. + */ + + const blockLength = 16; + + + /** + * xor `padding` into the end of `data`. This function implements "the + * operation xor→ [which] xors the shorter string into the end of longer + * one". Since data is always as least as long as padding, we can + * simplify the implementation. + * @param {Uint8Array} data + * @param {Uint8Array} padding + */ + function rightXORMut(data, padding) { + const offset = data.length - blockLength; + for (let i = 0; i < blockLength; i++) { + data[i + offset] ^= padding[i]; + } + return data; + } + + function pad(data, padding, padding2) { + // if |M| in {n, 2n, 3n, ...} + if (data.length && data.length % blockLength === 0) { + // then return M xor→ B, + return rightXORMut(data, padding); + } + // else return (M || 10^(n−1−(|M| mod n))) xor→ P + const padded = new Uint8Array(data.length + (blockLength - (data.length % blockLength))); + padded.set(data); + padded[data.length] = 0b10000000; + return rightXORMut(padded, padding2); + } + + const zeroBlock = new Uint8Array(blockLength); + + async function CMAC(key) { + const cbc = await CBC(key); + + // L ← E_K(0^n); B ← 2L; P ← 4L + const padding = util.double(await cbc(zeroBlock)); + const padding2 = util.double(padding); + + return async function(data) { + // return CBC_K(pad(M; B, P)) + return (await cbc(pad(data, padding, padding2))).subarray(-blockLength); + }; + } + + async function CBC(key) { + if (util.getWebCrypto() && key.length !== 24) { // WebCrypto (no 192 bit support) see: https://www.chromium.org/blink/webcrypto#TOC-AES-support + key = await webCrypto$2.importKey('raw', key, { name: 'AES-CBC', length: key.length * 8 }, false, ['encrypt']); + return async function(pt) { + const ct = await webCrypto$2.encrypt({ name: 'AES-CBC', iv: zeroBlock, length: blockLength * 8 }, key, pt); + return new Uint8Array(ct).subarray(0, ct.byteLength - blockLength); + }; + } + if (util.getNodeCrypto()) { // Node crypto library + return async function(pt) { + const en = new nodeCrypto$2.createCipheriv('aes-' + (key.length * 8) + '-cbc', key, zeroBlock); + const ct = en.update(pt); + return new Uint8Array(ct); + }; + } + // asm.js fallback + return async function(pt) { + return AES_CBC.encrypt(pt, key, false, zeroBlock); + }; + } + + // OpenPGP.js - An OpenPGP implementation in javascript + + const webCrypto$3 = util.getWebCrypto(); + const nodeCrypto$3 = util.getNodeCrypto(); + const Buffer$1 = util.getNodeBuffer(); + + + const blockLength$1 = 16; + const ivLength = blockLength$1; + const tagLength = blockLength$1; + + const zero = new Uint8Array(blockLength$1); + const one = new Uint8Array(blockLength$1); one[blockLength$1 - 1] = 1; + const two = new Uint8Array(blockLength$1); two[blockLength$1 - 1] = 2; + + async function OMAC(key) { + const cmac = await CMAC(key); + return function(t, message) { + return cmac(util.concatUint8Array([t, message])); + }; + } + + async function CTR(key) { + if ( + util.getWebCrypto() && + key.length !== 24 // WebCrypto (no 192 bit support) see: https://www.chromium.org/blink/webcrypto#TOC-AES-support + ) { + key = await webCrypto$3.importKey('raw', key, { name: 'AES-CTR', length: key.length * 8 }, false, ['encrypt']); + return async function(pt, iv) { + const ct = await webCrypto$3.encrypt({ name: 'AES-CTR', counter: iv, length: blockLength$1 * 8 }, key, pt); + return new Uint8Array(ct); + }; + } + if (util.getNodeCrypto()) { // Node crypto library + return async function(pt, iv) { + const en = new nodeCrypto$3.createCipheriv('aes-' + (key.length * 8) + '-ctr', key, iv); + const ct = Buffer$1.concat([en.update(pt), en.final()]); + return new Uint8Array(ct); + }; + } + // asm.js fallback + return async function(pt, iv) { + return AES_CTR.encrypt(pt, key, iv); + }; + } + + + /** + * Class to en/decrypt using EAX mode. + * @param {enums.symmetric} cipher - The symmetric cipher algorithm to use + * @param {Uint8Array} key - The encryption key + */ + async function EAX(cipher, key) { + if (cipher !== enums.symmetric.aes128 && + cipher !== enums.symmetric.aes192 && + cipher !== enums.symmetric.aes256) { + throw new Error('EAX mode supports only AES cipher'); + } + + const [ + omac, + ctr + ] = await Promise.all([ + OMAC(key), + CTR(key) + ]); + + return { + /** + * Encrypt plaintext input. + * @param {Uint8Array} plaintext - The cleartext input to be encrypted + * @param {Uint8Array} nonce - The nonce (16 bytes) + * @param {Uint8Array} adata - Associated data to sign + * @returns {Promise} The ciphertext output. + */ + encrypt: async function(plaintext, nonce, adata) { + const [ + omacNonce, + omacAdata + ] = await Promise.all([ + omac(zero, nonce), + omac(one, adata) + ]); + const ciphered = await ctr(plaintext, omacNonce); + const omacCiphered = await omac(two, ciphered); + const tag = omacCiphered; // Assumes that omac(*).length === tagLength. + for (let i = 0; i < tagLength; i++) { + tag[i] ^= omacAdata[i] ^ omacNonce[i]; + } + return util.concatUint8Array([ciphered, tag]); + }, + + /** + * Decrypt ciphertext input. + * @param {Uint8Array} ciphertext - The ciphertext input to be decrypted + * @param {Uint8Array} nonce - The nonce (16 bytes) + * @param {Uint8Array} adata - Associated data to verify + * @returns {Promise} The plaintext output. + */ + decrypt: async function(ciphertext, nonce, adata) { + if (ciphertext.length < tagLength) throw new Error('Invalid EAX ciphertext'); + const ciphered = ciphertext.subarray(0, -tagLength); + const ctTag = ciphertext.subarray(-tagLength); + const [ + omacNonce, + omacAdata, + omacCiphered + ] = await Promise.all([ + omac(zero, nonce), + omac(one, adata), + omac(two, ciphered) + ]); + const tag = omacCiphered; // Assumes that omac(*).length === tagLength. + for (let i = 0; i < tagLength; i++) { + tag[i] ^= omacAdata[i] ^ omacNonce[i]; + } + if (!util.equalsUint8Array(ctTag, tag)) throw new Error('Authentication tag mismatch'); + const plaintext = await ctr(ciphered, omacNonce); + return plaintext; + } + }; + } + + + /** + * Get EAX nonce as defined by {@link https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-04#section-5.16.1|RFC4880bis-04, section 5.16.1}. + * @param {Uint8Array} iv - The initialization vector (16 bytes) + * @param {Uint8Array} chunkIndex - The chunk index (8 bytes) + */ + EAX.getNonce = function(iv, chunkIndex) { + const nonce = iv.slice(); + for (let i = 0; i < chunkIndex.length; i++) { + nonce[8 + i] ^= chunkIndex[i]; + } + return nonce; + }; + + EAX.blockLength = blockLength$1; + EAX.ivLength = ivLength; + EAX.tagLength = tagLength; + + // OpenPGP.js - An OpenPGP implementation in javascript + + const blockLength$2 = 16; + const ivLength$1 = 15; + + // https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-04#section-5.16.2: + // While OCB [RFC7253] allows the authentication tag length to be of any + // number up to 128 bits long, this document requires a fixed + // authentication tag length of 128 bits (16 octets) for simplicity. + const tagLength$1 = 16; + + + function ntz(n) { + let ntz = 0; + for (let i = 1; (n & i) === 0; i <<= 1) { + ntz++; + } + return ntz; + } + + function xorMut$1(S, T) { + for (let i = 0; i < S.length; i++) { + S[i] ^= T[i]; + } + return S; + } + + function xor(S, T) { + return xorMut$1(S.slice(), T); + } + + const zeroBlock$1 = new Uint8Array(blockLength$2); + const one$1 = new Uint8Array([1]); + + /** + * Class to en/decrypt using OCB mode. + * @param {enums.symmetric} cipher - The symmetric cipher algorithm to use + * @param {Uint8Array} key - The encryption key + */ + async function OCB(cipher$1, key) { + + let maxNtz = 0; + let encipher; + let decipher; + let mask; + + constructKeyVariables(cipher$1, key); + + function constructKeyVariables(cipher$1, key) { + const cipherName = enums.read(enums.symmetric, cipher$1); + const aes = new cipher[cipherName](key); + encipher = aes.encrypt.bind(aes); + decipher = aes.decrypt.bind(aes); + + const mask_x = encipher(zeroBlock$1); + const mask_$ = util.double(mask_x); + mask = []; + mask[0] = util.double(mask_$); + + + mask.x = mask_x; + mask.$ = mask_$; + } + + function extendKeyVariables(text, adata) { + const newMaxNtz = util.nbits(Math.max(text.length, adata.length) / blockLength$2 | 0) - 1; + for (let i = maxNtz + 1; i <= newMaxNtz; i++) { + mask[i] = util.double(mask[i - 1]); + } + maxNtz = newMaxNtz; + } + + function hash(adata) { + if (!adata.length) { + // Fast path + return zeroBlock$1; + } + + // + // Consider A as a sequence of 128-bit blocks + // + const m = adata.length / blockLength$2 | 0; + + const offset = new Uint8Array(blockLength$2); + const sum = new Uint8Array(blockLength$2); + for (let i = 0; i < m; i++) { + xorMut$1(offset, mask[ntz(i + 1)]); + xorMut$1(sum, encipher(xor(offset, adata))); + adata = adata.subarray(blockLength$2); + } + + // + // Process any final partial block; compute final hash value + // + if (adata.length) { + xorMut$1(offset, mask.x); + + const cipherInput = new Uint8Array(blockLength$2); + cipherInput.set(adata, 0); + cipherInput[adata.length] = 0b10000000; + xorMut$1(cipherInput, offset); + + xorMut$1(sum, encipher(cipherInput)); + } + + return sum; + } + + /** + * Encrypt/decrypt data. + * @param {encipher|decipher} fn - Encryption/decryption block cipher function + * @param {Uint8Array} text - The cleartext or ciphertext (without tag) input + * @param {Uint8Array} nonce - The nonce (15 bytes) + * @param {Uint8Array} adata - Associated data to sign + * @returns {Promise} The ciphertext or plaintext output, with tag appended in both cases. + */ + function crypt(fn, text, nonce, adata) { + // + // Consider P as a sequence of 128-bit blocks + // + const m = text.length / blockLength$2 | 0; + + // + // Key-dependent variables + // + extendKeyVariables(text, adata); + + // + // Nonce-dependent and per-encryption variables + // + // Nonce = num2str(TAGLEN mod 128,7) || zeros(120-bitlen(N)) || 1 || N + // Note: We assume here that tagLength mod 16 == 0. + const paddedNonce = util.concatUint8Array([zeroBlock$1.subarray(0, ivLength$1 - nonce.length), one$1, nonce]); + // bottom = str2num(Nonce[123..128]) + const bottom = paddedNonce[blockLength$2 - 1] & 0b111111; + // Ktop = ENCIPHER(K, Nonce[1..122] || zeros(6)) + paddedNonce[blockLength$2 - 1] &= 0b11000000; + const kTop = encipher(paddedNonce); + // Stretch = Ktop || (Ktop[1..64] xor Ktop[9..72]) + const stretched = util.concatUint8Array([kTop, xor(kTop.subarray(0, 8), kTop.subarray(1, 9))]); + // Offset_0 = Stretch[1+bottom..128+bottom] + const offset = util.shiftRight(stretched.subarray(0 + (bottom >> 3), 17 + (bottom >> 3)), 8 - (bottom & 7)).subarray(1); + // Checksum_0 = zeros(128) + const checksum = new Uint8Array(blockLength$2); + + const ct = new Uint8Array(text.length + tagLength$1); + + // + // Process any whole blocks + // + let i; + let pos = 0; + for (i = 0; i < m; i++) { + // Offset_i = Offset_{i-1} xor L_{ntz(i)} + xorMut$1(offset, mask[ntz(i + 1)]); + // C_i = Offset_i xor ENCIPHER(K, P_i xor Offset_i) + // P_i = Offset_i xor DECIPHER(K, C_i xor Offset_i) + ct.set(xorMut$1(fn(xor(offset, text)), offset), pos); + // Checksum_i = Checksum_{i-1} xor P_i + xorMut$1(checksum, fn === encipher ? text : ct.subarray(pos)); + + text = text.subarray(blockLength$2); + pos += blockLength$2; + } + + // + // Process any final partial block and compute raw tag + // + if (text.length) { + // Offset_* = Offset_m xor L_* + xorMut$1(offset, mask.x); + // Pad = ENCIPHER(K, Offset_*) + const padding = encipher(offset); + // C_* = P_* xor Pad[1..bitlen(P_*)] + ct.set(xor(text, padding), pos); + + // Checksum_* = Checksum_m xor (P_* || 1 || new Uint8Array(127-bitlen(P_*))) + const xorInput = new Uint8Array(blockLength$2); + xorInput.set(fn === encipher ? text : ct.subarray(pos, -tagLength$1), 0); + xorInput[text.length] = 0b10000000; + xorMut$1(checksum, xorInput); + pos += text.length; + } + // Tag = ENCIPHER(K, Checksum_* xor Offset_* xor L_$) xor HASH(K,A) + const tag = xorMut$1(encipher(xorMut$1(xorMut$1(checksum, offset), mask.$)), hash(adata)); + + // + // Assemble ciphertext + // + // C = C_1 || C_2 || ... || C_m || C_* || Tag[1..TAGLEN] + ct.set(tag, pos); + return ct; + } + + + return { + /** + * Encrypt plaintext input. + * @param {Uint8Array} plaintext - The cleartext input to be encrypted + * @param {Uint8Array} nonce - The nonce (15 bytes) + * @param {Uint8Array} adata - Associated data to sign + * @returns {Promise} The ciphertext output. + */ + encrypt: async function(plaintext, nonce, adata) { + return crypt(encipher, plaintext, nonce, adata); + }, + + /** + * Decrypt ciphertext input. + * @param {Uint8Array} ciphertext - The ciphertext input to be decrypted + * @param {Uint8Array} nonce - The nonce (15 bytes) + * @param {Uint8Array} adata - Associated data to sign + * @returns {Promise} The ciphertext output. + */ + decrypt: async function(ciphertext, nonce, adata) { + if (ciphertext.length < tagLength$1) throw new Error('Invalid OCB ciphertext'); + + const tag = ciphertext.subarray(-tagLength$1); + ciphertext = ciphertext.subarray(0, -tagLength$1); + + const crypted = crypt(decipher, ciphertext, nonce, adata); + // if (Tag[1..TAGLEN] == T) + if (util.equalsUint8Array(tag, crypted.subarray(-tagLength$1))) { + return crypted.subarray(0, -tagLength$1); + } + throw new Error('Authentication tag mismatch'); + } + }; + } + + + /** + * Get OCB nonce as defined by {@link https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-04#section-5.16.2|RFC4880bis-04, section 5.16.2}. + * @param {Uint8Array} iv - The initialization vector (15 bytes) + * @param {Uint8Array} chunkIndex - The chunk index (8 bytes) + */ + OCB.getNonce = function(iv, chunkIndex) { + const nonce = iv.slice(); + for (let i = 0; i < chunkIndex.length; i++) { + nonce[7 + i] ^= chunkIndex[i]; + } + return nonce; + }; + + OCB.blockLength = blockLength$2; + OCB.ivLength = ivLength$1; + OCB.tagLength = tagLength$1; + + const _AES_GCM_data_maxLength = 68719476704; // 2^36 - 2^5 + class AES_GCM { + constructor(key, nonce, adata, tagSize = 16, aes) { + this.tagSize = tagSize; + this.gamma0 = 0; + this.counter = 1; + this.aes = aes ? aes : new AES(key, undefined, false, 'CTR'); + let { asm, heap } = this.aes.acquire_asm(); + // Init GCM + asm.gcm_init(); + // Tag size + if (this.tagSize < 4 || this.tagSize > 16) + throw new IllegalArgumentError('illegal tagSize value'); + // Nonce + const noncelen = nonce.length || 0; + const noncebuf = new Uint8Array(16); + if (noncelen !== 12) { + this._gcm_mac_process(nonce); + heap[0] = 0; + heap[1] = 0; + heap[2] = 0; + heap[3] = 0; + heap[4] = 0; + heap[5] = 0; + heap[6] = 0; + heap[7] = 0; + heap[8] = 0; + heap[9] = 0; + heap[10] = 0; + heap[11] = noncelen >>> 29; + heap[12] = (noncelen >>> 21) & 255; + heap[13] = (noncelen >>> 13) & 255; + heap[14] = (noncelen >>> 5) & 255; + heap[15] = (noncelen << 3) & 255; + asm.mac(AES_asm.MAC.GCM, AES_asm.HEAP_DATA, 16); + asm.get_iv(AES_asm.HEAP_DATA); + asm.set_iv(0, 0, 0, 0); + noncebuf.set(heap.subarray(0, 16)); + } + else { + noncebuf.set(nonce); + noncebuf[15] = 1; + } + const nonceview = new DataView(noncebuf.buffer); + this.gamma0 = nonceview.getUint32(12); + asm.set_nonce(nonceview.getUint32(0), nonceview.getUint32(4), nonceview.getUint32(8), 0); + asm.set_mask(0, 0, 0, 0xffffffff); + // Associated data + if (adata !== undefined) { + if (adata.length > _AES_GCM_data_maxLength) + throw new IllegalArgumentError('illegal adata length'); + if (adata.length) { + this.adata = adata; + this._gcm_mac_process(adata); + } + else { + this.adata = undefined; + } + } + else { + this.adata = undefined; + } + // Counter + if (this.counter < 1 || this.counter > 0xffffffff) + throw new RangeError('counter must be a positive 32-bit integer'); + asm.set_counter(0, 0, 0, (this.gamma0 + this.counter) | 0); + } + static encrypt(cleartext, key, nonce, adata, tagsize) { + return new AES_GCM(key, nonce, adata, tagsize).encrypt(cleartext); + } + static decrypt(ciphertext, key, nonce, adata, tagsize) { + return new AES_GCM(key, nonce, adata, tagsize).decrypt(ciphertext); + } + encrypt(data) { + return this.AES_GCM_encrypt(data); + } + decrypt(data) { + return this.AES_GCM_decrypt(data); + } + AES_GCM_Encrypt_process(data) { + let dpos = 0; + let dlen = data.length || 0; + let { asm, heap } = this.aes.acquire_asm(); + let counter = this.counter; + let pos = this.aes.pos; + let len = this.aes.len; + let rpos = 0; + let rlen = (len + dlen) & -16; + let wlen = 0; + if (((counter - 1) << 4) + len + dlen > _AES_GCM_data_maxLength) + throw new RangeError('counter overflow'); + const result = new Uint8Array(rlen); + while (dlen > 0) { + wlen = _heap_write(heap, pos + len, data, dpos, dlen); + len += wlen; + dpos += wlen; + dlen -= wlen; + wlen = asm.cipher(AES_asm.ENC.CTR, AES_asm.HEAP_DATA + pos, len); + wlen = asm.mac(AES_asm.MAC.GCM, AES_asm.HEAP_DATA + pos, wlen); + if (wlen) + result.set(heap.subarray(pos, pos + wlen), rpos); + counter += wlen >>> 4; + rpos += wlen; + if (wlen < len) { + pos += wlen; + len -= wlen; + } + else { + pos = 0; + len = 0; + } + } + this.counter = counter; + this.aes.pos = pos; + this.aes.len = len; + return result; + } + AES_GCM_Encrypt_finish() { + let { asm, heap } = this.aes.acquire_asm(); + let counter = this.counter; + let tagSize = this.tagSize; + let adata = this.adata; + let pos = this.aes.pos; + let len = this.aes.len; + const result = new Uint8Array(len + tagSize); + asm.cipher(AES_asm.ENC.CTR, AES_asm.HEAP_DATA + pos, (len + 15) & -16); + if (len) + result.set(heap.subarray(pos, pos + len)); + let i = len; + for (; i & 15; i++) + heap[pos + i] = 0; + asm.mac(AES_asm.MAC.GCM, AES_asm.HEAP_DATA + pos, i); + const alen = adata !== undefined ? adata.length : 0; + const clen = ((counter - 1) << 4) + len; + heap[0] = 0; + heap[1] = 0; + heap[2] = 0; + heap[3] = alen >>> 29; + heap[4] = alen >>> 21; + heap[5] = (alen >>> 13) & 255; + heap[6] = (alen >>> 5) & 255; + heap[7] = (alen << 3) & 255; + heap[8] = heap[9] = heap[10] = 0; + heap[11] = clen >>> 29; + heap[12] = (clen >>> 21) & 255; + heap[13] = (clen >>> 13) & 255; + heap[14] = (clen >>> 5) & 255; + heap[15] = (clen << 3) & 255; + asm.mac(AES_asm.MAC.GCM, AES_asm.HEAP_DATA, 16); + asm.get_iv(AES_asm.HEAP_DATA); + asm.set_counter(0, 0, 0, this.gamma0); + asm.cipher(AES_asm.ENC.CTR, AES_asm.HEAP_DATA, 16); + result.set(heap.subarray(0, tagSize), len); + this.counter = 1; + this.aes.pos = 0; + this.aes.len = 0; + return result; + } + AES_GCM_Decrypt_process(data) { + let dpos = 0; + let dlen = data.length || 0; + let { asm, heap } = this.aes.acquire_asm(); + let counter = this.counter; + let tagSize = this.tagSize; + let pos = this.aes.pos; + let len = this.aes.len; + let rpos = 0; + let rlen = len + dlen > tagSize ? (len + dlen - tagSize) & -16 : 0; + let tlen = len + dlen - rlen; + let wlen = 0; + if (((counter - 1) << 4) + len + dlen > _AES_GCM_data_maxLength) + throw new RangeError('counter overflow'); + const result = new Uint8Array(rlen); + while (dlen > tlen) { + wlen = _heap_write(heap, pos + len, data, dpos, dlen - tlen); + len += wlen; + dpos += wlen; + dlen -= wlen; + wlen = asm.mac(AES_asm.MAC.GCM, AES_asm.HEAP_DATA + pos, wlen); + wlen = asm.cipher(AES_asm.DEC.CTR, AES_asm.HEAP_DATA + pos, wlen); + if (wlen) + result.set(heap.subarray(pos, pos + wlen), rpos); + counter += wlen >>> 4; + rpos += wlen; + pos = 0; + len = 0; + } + if (dlen > 0) { + len += _heap_write(heap, 0, data, dpos, dlen); + } + this.counter = counter; + this.aes.pos = pos; + this.aes.len = len; + return result; + } + AES_GCM_Decrypt_finish() { + let { asm, heap } = this.aes.acquire_asm(); + let tagSize = this.tagSize; + let adata = this.adata; + let counter = this.counter; + let pos = this.aes.pos; + let len = this.aes.len; + let rlen = len - tagSize; + if (len < tagSize) + throw new IllegalStateError('authentication tag not found'); + const result = new Uint8Array(rlen); + const atag = new Uint8Array(heap.subarray(pos + rlen, pos + len)); + let i = rlen; + for (; i & 15; i++) + heap[pos + i] = 0; + asm.mac(AES_asm.MAC.GCM, AES_asm.HEAP_DATA + pos, i); + asm.cipher(AES_asm.DEC.CTR, AES_asm.HEAP_DATA + pos, i); + if (rlen) + result.set(heap.subarray(pos, pos + rlen)); + const alen = adata !== undefined ? adata.length : 0; + const clen = ((counter - 1) << 4) + len - tagSize; + heap[0] = 0; + heap[1] = 0; + heap[2] = 0; + heap[3] = alen >>> 29; + heap[4] = alen >>> 21; + heap[5] = (alen >>> 13) & 255; + heap[6] = (alen >>> 5) & 255; + heap[7] = (alen << 3) & 255; + heap[8] = heap[9] = heap[10] = 0; + heap[11] = clen >>> 29; + heap[12] = (clen >>> 21) & 255; + heap[13] = (clen >>> 13) & 255; + heap[14] = (clen >>> 5) & 255; + heap[15] = (clen << 3) & 255; + asm.mac(AES_asm.MAC.GCM, AES_asm.HEAP_DATA, 16); + asm.get_iv(AES_asm.HEAP_DATA); + asm.set_counter(0, 0, 0, this.gamma0); + asm.cipher(AES_asm.ENC.CTR, AES_asm.HEAP_DATA, 16); + let acheck = 0; + for (let i = 0; i < tagSize; ++i) + acheck |= atag[i] ^ heap[i]; + if (acheck) + throw new SecurityError('data integrity check failed'); + this.counter = 1; + this.aes.pos = 0; + this.aes.len = 0; + return result; + } + AES_GCM_decrypt(data) { + const result1 = this.AES_GCM_Decrypt_process(data); + const result2 = this.AES_GCM_Decrypt_finish(); + const result = new Uint8Array(result1.length + result2.length); + if (result1.length) + result.set(result1); + if (result2.length) + result.set(result2, result1.length); + return result; + } + AES_GCM_encrypt(data) { + const result1 = this.AES_GCM_Encrypt_process(data); + const result2 = this.AES_GCM_Encrypt_finish(); + const result = new Uint8Array(result1.length + result2.length); + if (result1.length) + result.set(result1); + if (result2.length) + result.set(result2, result1.length); + return result; + } + _gcm_mac_process(data) { + let { asm, heap } = this.aes.acquire_asm(); + let dpos = 0; + let dlen = data.length || 0; + let wlen = 0; + while (dlen > 0) { + wlen = _heap_write(heap, 0, data, dpos, dlen); + dpos += wlen; + dlen -= wlen; + while (wlen & 15) + heap[wlen++] = 0; + asm.mac(AES_asm.MAC.GCM, AES_asm.HEAP_DATA, wlen); + } + } + } + + // OpenPGP.js - An OpenPGP implementation in javascript + + const webCrypto$4 = util.getWebCrypto(); + const nodeCrypto$4 = util.getNodeCrypto(); + const Buffer$2 = util.getNodeBuffer(); + + const blockLength$3 = 16; + const ivLength$2 = 12; // size of the IV in bytes + const tagLength$2 = 16; // size of the tag in bytes + const ALGO = 'AES-GCM'; + + /** + * Class to en/decrypt using GCM mode. + * @param {enums.symmetric} cipher - The symmetric cipher algorithm to use + * @param {Uint8Array} key - The encryption key + */ + async function GCM(cipher, key) { + if (cipher !== enums.symmetric.aes128 && + cipher !== enums.symmetric.aes192 && + cipher !== enums.symmetric.aes256) { + throw new Error('GCM mode supports only AES cipher'); + } + + if (util.getWebCrypto() && key.length !== 24) { // WebCrypto (no 192 bit support) see: https://www.chromium.org/blink/webcrypto#TOC-AES-support + const _key = await webCrypto$4.importKey('raw', key, { name: ALGO }, false, ['encrypt', 'decrypt']); + + return { + encrypt: async function(pt, iv, adata = new Uint8Array()) { + if (!pt.length) { // iOS does not support GCM-en/decrypting empty messages + return AES_GCM.encrypt(pt, key, iv, adata); + } + const ct = await webCrypto$4.encrypt({ name: ALGO, iv, additionalData: adata, tagLength: tagLength$2 * 8 }, _key, pt); + return new Uint8Array(ct); + }, + + decrypt: async function(ct, iv, adata = new Uint8Array()) { + if (ct.length === tagLength$2) { // iOS does not support GCM-en/decrypting empty messages + return AES_GCM.decrypt(ct, key, iv, adata); + } + const pt = await webCrypto$4.decrypt({ name: ALGO, iv, additionalData: adata, tagLength: tagLength$2 * 8 }, _key, ct); + return new Uint8Array(pt); + } + }; + } + + if (util.getNodeCrypto()) { // Node crypto library + return { + encrypt: async function(pt, iv, adata = new Uint8Array()) { + const en = new nodeCrypto$4.createCipheriv('aes-' + (key.length * 8) + '-gcm', key, iv); + en.setAAD(adata); + const ct = Buffer$2.concat([en.update(pt), en.final(), en.getAuthTag()]); // append auth tag to ciphertext + return new Uint8Array(ct); + }, + + decrypt: async function(ct, iv, adata = new Uint8Array()) { + const de = new nodeCrypto$4.createDecipheriv('aes-' + (key.length * 8) + '-gcm', key, iv); + de.setAAD(adata); + de.setAuthTag(ct.slice(ct.length - tagLength$2, ct.length)); // read auth tag at end of ciphertext + const pt = Buffer$2.concat([de.update(ct.slice(0, ct.length - tagLength$2)), de.final()]); + return new Uint8Array(pt); + } + }; + } + + return { + encrypt: async function(pt, iv, adata) { + return AES_GCM.encrypt(pt, key, iv, adata); + }, + + decrypt: async function(ct, iv, adata) { + return AES_GCM.decrypt(ct, key, iv, adata); + } + }; + } + + + /** + * Get GCM nonce. Note: this operation is not defined by the standard. + * A future version of the standard may define GCM mode differently, + * hopefully under a different ID (we use Private/Experimental algorithm + * ID 100) so that we can maintain backwards compatibility. + * @param {Uint8Array} iv - The initialization vector (12 bytes) + * @param {Uint8Array} chunkIndex - The chunk index (8 bytes) + */ + GCM.getNonce = function(iv, chunkIndex) { + const nonce = iv.slice(); + for (let i = 0; i < chunkIndex.length; i++) { + nonce[4 + i] ^= chunkIndex[i]; + } + return nonce; + }; + + GCM.blockLength = blockLength$3; + GCM.ivLength = ivLength$2; + GCM.tagLength = tagLength$2; + + /** + * @fileoverview Cipher modes + * @module crypto/mode + * @private + */ + + var mode = { + /** @see module:crypto/mode/cfb */ + cfb: cfb, + /** @see module:crypto/mode/gcm */ + gcm: GCM, + experimentalGCM: GCM, + /** @see module:crypto/mode/eax */ + eax: EAX, + /** @see module:crypto/mode/ocb */ + ocb: OCB + }; + + var naclFastLight = createCommonjsModule(function (module) { + /*jshint bitwise: false*/ + + (function(nacl) { + + // Ported in 2014 by Dmitry Chestnykh and Devi Mandiri. + // Public domain. + // + // Implementation derived from TweetNaCl version 20140427. + // See for details: http://tweetnacl.cr.yp.to/ + + var gf = function(init) { + var i, r = new Float64Array(16); + if (init) for (i = 0; i < init.length; i++) r[i] = init[i]; + return r; + }; + + // Pluggable, initialized in high-level API below. + var randombytes = function(/* x, n */) { throw new Error('no PRNG'); }; + + var _9 = new Uint8Array(32); _9[0] = 9; + + var gf0 = gf(), + gf1 = gf([1]), + _121665 = gf([0xdb41, 1]), + D = gf([0x78a3, 0x1359, 0x4dca, 0x75eb, 0xd8ab, 0x4141, 0x0a4d, 0x0070, 0xe898, 0x7779, 0x4079, 0x8cc7, 0xfe73, 0x2b6f, 0x6cee, 0x5203]), + D2 = gf([0xf159, 0x26b2, 0x9b94, 0xebd6, 0xb156, 0x8283, 0x149a, 0x00e0, 0xd130, 0xeef3, 0x80f2, 0x198e, 0xfce7, 0x56df, 0xd9dc, 0x2406]), + X = gf([0xd51a, 0x8f25, 0x2d60, 0xc956, 0xa7b2, 0x9525, 0xc760, 0x692c, 0xdc5c, 0xfdd6, 0xe231, 0xc0a4, 0x53fe, 0xcd6e, 0x36d3, 0x2169]), + Y = gf([0x6658, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666, 0x6666]), + I = gf([0xa0b0, 0x4a0e, 0x1b27, 0xc4ee, 0xe478, 0xad2f, 0x1806, 0x2f43, 0xd7a7, 0x3dfb, 0x0099, 0x2b4d, 0xdf0b, 0x4fc1, 0x2480, 0x2b83]); + + function vn(x, xi, y, yi, n) { + var i,d = 0; + for (i = 0; i < n; i++) d |= x[xi+i]^y[yi+i]; + return (1 & ((d - 1) >>> 8)) - 1; + } + + function crypto_verify_32(x, xi, y, yi) { + return vn(x,xi,y,yi,32); + } + + function set25519(r, a) { + var i; + for (i = 0; i < 16; i++) r[i] = a[i]|0; + } + + function car25519(o) { + var i, v, c = 1; + for (i = 0; i < 16; i++) { + v = o[i] + c + 65535; + c = Math.floor(v / 65536); + o[i] = v - c * 65536; + } + o[0] += c-1 + 37 * (c-1); + } + + function sel25519(p, q, b) { + var t, c = ~(b-1); + for (var i = 0; i < 16; i++) { + t = c & (p[i] ^ q[i]); + p[i] ^= t; + q[i] ^= t; + } + } + + function pack25519(o, n) { + var i, j, b; + var m = gf(), t = gf(); + for (i = 0; i < 16; i++) t[i] = n[i]; + car25519(t); + car25519(t); + car25519(t); + for (j = 0; j < 2; j++) { + m[0] = t[0] - 0xffed; + for (i = 1; i < 15; i++) { + m[i] = t[i] - 0xffff - ((m[i-1]>>16) & 1); + m[i-1] &= 0xffff; + } + m[15] = t[15] - 0x7fff - ((m[14]>>16) & 1); + b = (m[15]>>16) & 1; + m[14] &= 0xffff; + sel25519(t, m, 1-b); + } + for (i = 0; i < 16; i++) { + o[2*i] = t[i] & 0xff; + o[2*i+1] = t[i]>>8; + } + } + + function neq25519(a, b) { + var c = new Uint8Array(32), d = new Uint8Array(32); + pack25519(c, a); + pack25519(d, b); + return crypto_verify_32(c, 0, d, 0); + } + + function par25519(a) { + var d = new Uint8Array(32); + pack25519(d, a); + return d[0] & 1; + } + + function unpack25519(o, n) { + var i; + for (i = 0; i < 16; i++) o[i] = n[2*i] + (n[2*i+1] << 8); + o[15] &= 0x7fff; + } + + function A(o, a, b) { + for (var i = 0; i < 16; i++) o[i] = a[i] + b[i]; + } + + function Z(o, a, b) { + for (var i = 0; i < 16; i++) o[i] = a[i] - b[i]; + } + + function M(o, a, b) { + var v, c, + t0 = 0, t1 = 0, t2 = 0, t3 = 0, t4 = 0, t5 = 0, t6 = 0, t7 = 0, + t8 = 0, t9 = 0, t10 = 0, t11 = 0, t12 = 0, t13 = 0, t14 = 0, t15 = 0, + t16 = 0, t17 = 0, t18 = 0, t19 = 0, t20 = 0, t21 = 0, t22 = 0, t23 = 0, + t24 = 0, t25 = 0, t26 = 0, t27 = 0, t28 = 0, t29 = 0, t30 = 0, + b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3], + b4 = b[4], + b5 = b[5], + b6 = b[6], + b7 = b[7], + b8 = b[8], + b9 = b[9], + b10 = b[10], + b11 = b[11], + b12 = b[12], + b13 = b[13], + b14 = b[14], + b15 = b[15]; + + v = a[0]; + t0 += v * b0; + t1 += v * b1; + t2 += v * b2; + t3 += v * b3; + t4 += v * b4; + t5 += v * b5; + t6 += v * b6; + t7 += v * b7; + t8 += v * b8; + t9 += v * b9; + t10 += v * b10; + t11 += v * b11; + t12 += v * b12; + t13 += v * b13; + t14 += v * b14; + t15 += v * b15; + v = a[1]; + t1 += v * b0; + t2 += v * b1; + t3 += v * b2; + t4 += v * b3; + t5 += v * b4; + t6 += v * b5; + t7 += v * b6; + t8 += v * b7; + t9 += v * b8; + t10 += v * b9; + t11 += v * b10; + t12 += v * b11; + t13 += v * b12; + t14 += v * b13; + t15 += v * b14; + t16 += v * b15; + v = a[2]; + t2 += v * b0; + t3 += v * b1; + t4 += v * b2; + t5 += v * b3; + t6 += v * b4; + t7 += v * b5; + t8 += v * b6; + t9 += v * b7; + t10 += v * b8; + t11 += v * b9; + t12 += v * b10; + t13 += v * b11; + t14 += v * b12; + t15 += v * b13; + t16 += v * b14; + t17 += v * b15; + v = a[3]; + t3 += v * b0; + t4 += v * b1; + t5 += v * b2; + t6 += v * b3; + t7 += v * b4; + t8 += v * b5; + t9 += v * b6; + t10 += v * b7; + t11 += v * b8; + t12 += v * b9; + t13 += v * b10; + t14 += v * b11; + t15 += v * b12; + t16 += v * b13; + t17 += v * b14; + t18 += v * b15; + v = a[4]; + t4 += v * b0; + t5 += v * b1; + t6 += v * b2; + t7 += v * b3; + t8 += v * b4; + t9 += v * b5; + t10 += v * b6; + t11 += v * b7; + t12 += v * b8; + t13 += v * b9; + t14 += v * b10; + t15 += v * b11; + t16 += v * b12; + t17 += v * b13; + t18 += v * b14; + t19 += v * b15; + v = a[5]; + t5 += v * b0; + t6 += v * b1; + t7 += v * b2; + t8 += v * b3; + t9 += v * b4; + t10 += v * b5; + t11 += v * b6; + t12 += v * b7; + t13 += v * b8; + t14 += v * b9; + t15 += v * b10; + t16 += v * b11; + t17 += v * b12; + t18 += v * b13; + t19 += v * b14; + t20 += v * b15; + v = a[6]; + t6 += v * b0; + t7 += v * b1; + t8 += v * b2; + t9 += v * b3; + t10 += v * b4; + t11 += v * b5; + t12 += v * b6; + t13 += v * b7; + t14 += v * b8; + t15 += v * b9; + t16 += v * b10; + t17 += v * b11; + t18 += v * b12; + t19 += v * b13; + t20 += v * b14; + t21 += v * b15; + v = a[7]; + t7 += v * b0; + t8 += v * b1; + t9 += v * b2; + t10 += v * b3; + t11 += v * b4; + t12 += v * b5; + t13 += v * b6; + t14 += v * b7; + t15 += v * b8; + t16 += v * b9; + t17 += v * b10; + t18 += v * b11; + t19 += v * b12; + t20 += v * b13; + t21 += v * b14; + t22 += v * b15; + v = a[8]; + t8 += v * b0; + t9 += v * b1; + t10 += v * b2; + t11 += v * b3; + t12 += v * b4; + t13 += v * b5; + t14 += v * b6; + t15 += v * b7; + t16 += v * b8; + t17 += v * b9; + t18 += v * b10; + t19 += v * b11; + t20 += v * b12; + t21 += v * b13; + t22 += v * b14; + t23 += v * b15; + v = a[9]; + t9 += v * b0; + t10 += v * b1; + t11 += v * b2; + t12 += v * b3; + t13 += v * b4; + t14 += v * b5; + t15 += v * b6; + t16 += v * b7; + t17 += v * b8; + t18 += v * b9; + t19 += v * b10; + t20 += v * b11; + t21 += v * b12; + t22 += v * b13; + t23 += v * b14; + t24 += v * b15; + v = a[10]; + t10 += v * b0; + t11 += v * b1; + t12 += v * b2; + t13 += v * b3; + t14 += v * b4; + t15 += v * b5; + t16 += v * b6; + t17 += v * b7; + t18 += v * b8; + t19 += v * b9; + t20 += v * b10; + t21 += v * b11; + t22 += v * b12; + t23 += v * b13; + t24 += v * b14; + t25 += v * b15; + v = a[11]; + t11 += v * b0; + t12 += v * b1; + t13 += v * b2; + t14 += v * b3; + t15 += v * b4; + t16 += v * b5; + t17 += v * b6; + t18 += v * b7; + t19 += v * b8; + t20 += v * b9; + t21 += v * b10; + t22 += v * b11; + t23 += v * b12; + t24 += v * b13; + t25 += v * b14; + t26 += v * b15; + v = a[12]; + t12 += v * b0; + t13 += v * b1; + t14 += v * b2; + t15 += v * b3; + t16 += v * b4; + t17 += v * b5; + t18 += v * b6; + t19 += v * b7; + t20 += v * b8; + t21 += v * b9; + t22 += v * b10; + t23 += v * b11; + t24 += v * b12; + t25 += v * b13; + t26 += v * b14; + t27 += v * b15; + v = a[13]; + t13 += v * b0; + t14 += v * b1; + t15 += v * b2; + t16 += v * b3; + t17 += v * b4; + t18 += v * b5; + t19 += v * b6; + t20 += v * b7; + t21 += v * b8; + t22 += v * b9; + t23 += v * b10; + t24 += v * b11; + t25 += v * b12; + t26 += v * b13; + t27 += v * b14; + t28 += v * b15; + v = a[14]; + t14 += v * b0; + t15 += v * b1; + t16 += v * b2; + t17 += v * b3; + t18 += v * b4; + t19 += v * b5; + t20 += v * b6; + t21 += v * b7; + t22 += v * b8; + t23 += v * b9; + t24 += v * b10; + t25 += v * b11; + t26 += v * b12; + t27 += v * b13; + t28 += v * b14; + t29 += v * b15; + v = a[15]; + t15 += v * b0; + t16 += v * b1; + t17 += v * b2; + t18 += v * b3; + t19 += v * b4; + t20 += v * b5; + t21 += v * b6; + t22 += v * b7; + t23 += v * b8; + t24 += v * b9; + t25 += v * b10; + t26 += v * b11; + t27 += v * b12; + t28 += v * b13; + t29 += v * b14; + t30 += v * b15; + + t0 += 38 * t16; + t1 += 38 * t17; + t2 += 38 * t18; + t3 += 38 * t19; + t4 += 38 * t20; + t5 += 38 * t21; + t6 += 38 * t22; + t7 += 38 * t23; + t8 += 38 * t24; + t9 += 38 * t25; + t10 += 38 * t26; + t11 += 38 * t27; + t12 += 38 * t28; + t13 += 38 * t29; + t14 += 38 * t30; + // t15 left as is + + // first car + c = 1; + v = t0 + c + 65535; c = Math.floor(v / 65536); t0 = v - c * 65536; + v = t1 + c + 65535; c = Math.floor(v / 65536); t1 = v - c * 65536; + v = t2 + c + 65535; c = Math.floor(v / 65536); t2 = v - c * 65536; + v = t3 + c + 65535; c = Math.floor(v / 65536); t3 = v - c * 65536; + v = t4 + c + 65535; c = Math.floor(v / 65536); t4 = v - c * 65536; + v = t5 + c + 65535; c = Math.floor(v / 65536); t5 = v - c * 65536; + v = t6 + c + 65535; c = Math.floor(v / 65536); t6 = v - c * 65536; + v = t7 + c + 65535; c = Math.floor(v / 65536); t7 = v - c * 65536; + v = t8 + c + 65535; c = Math.floor(v / 65536); t8 = v - c * 65536; + v = t9 + c + 65535; c = Math.floor(v / 65536); t9 = v - c * 65536; + v = t10 + c + 65535; c = Math.floor(v / 65536); t10 = v - c * 65536; + v = t11 + c + 65535; c = Math.floor(v / 65536); t11 = v - c * 65536; + v = t12 + c + 65535; c = Math.floor(v / 65536); t12 = v - c * 65536; + v = t13 + c + 65535; c = Math.floor(v / 65536); t13 = v - c * 65536; + v = t14 + c + 65535; c = Math.floor(v / 65536); t14 = v - c * 65536; + v = t15 + c + 65535; c = Math.floor(v / 65536); t15 = v - c * 65536; + t0 += c-1 + 37 * (c-1); + + // second car + c = 1; + v = t0 + c + 65535; c = Math.floor(v / 65536); t0 = v - c * 65536; + v = t1 + c + 65535; c = Math.floor(v / 65536); t1 = v - c * 65536; + v = t2 + c + 65535; c = Math.floor(v / 65536); t2 = v - c * 65536; + v = t3 + c + 65535; c = Math.floor(v / 65536); t3 = v - c * 65536; + v = t4 + c + 65535; c = Math.floor(v / 65536); t4 = v - c * 65536; + v = t5 + c + 65535; c = Math.floor(v / 65536); t5 = v - c * 65536; + v = t6 + c + 65535; c = Math.floor(v / 65536); t6 = v - c * 65536; + v = t7 + c + 65535; c = Math.floor(v / 65536); t7 = v - c * 65536; + v = t8 + c + 65535; c = Math.floor(v / 65536); t8 = v - c * 65536; + v = t9 + c + 65535; c = Math.floor(v / 65536); t9 = v - c * 65536; + v = t10 + c + 65535; c = Math.floor(v / 65536); t10 = v - c * 65536; + v = t11 + c + 65535; c = Math.floor(v / 65536); t11 = v - c * 65536; + v = t12 + c + 65535; c = Math.floor(v / 65536); t12 = v - c * 65536; + v = t13 + c + 65535; c = Math.floor(v / 65536); t13 = v - c * 65536; + v = t14 + c + 65535; c = Math.floor(v / 65536); t14 = v - c * 65536; + v = t15 + c + 65535; c = Math.floor(v / 65536); t15 = v - c * 65536; + t0 += c-1 + 37 * (c-1); + + o[ 0] = t0; + o[ 1] = t1; + o[ 2] = t2; + o[ 3] = t3; + o[ 4] = t4; + o[ 5] = t5; + o[ 6] = t6; + o[ 7] = t7; + o[ 8] = t8; + o[ 9] = t9; + o[10] = t10; + o[11] = t11; + o[12] = t12; + o[13] = t13; + o[14] = t14; + o[15] = t15; + } + + function S(o, a) { + M(o, a, a); + } + + function inv25519(o, i) { + var c = gf(); + var a; + for (a = 0; a < 16; a++) c[a] = i[a]; + for (a = 253; a >= 0; a--) { + S(c, c); + if(a !== 2 && a !== 4) M(c, c, i); + } + for (a = 0; a < 16; a++) o[a] = c[a]; + } + + function pow2523(o, i) { + var c = gf(); + var a; + for (a = 0; a < 16; a++) c[a] = i[a]; + for (a = 250; a >= 0; a--) { + S(c, c); + if(a !== 1) M(c, c, i); + } + for (a = 0; a < 16; a++) o[a] = c[a]; + } + + function crypto_scalarmult(q, n, p) { + var z = new Uint8Array(32); + var x = new Float64Array(80), r, i; + var a = gf(), b = gf(), c = gf(), + d = gf(), e = gf(), f = gf(); + for (i = 0; i < 31; i++) z[i] = n[i]; + z[31]=(n[31]&127)|64; + z[0]&=248; + unpack25519(x,p); + for (i = 0; i < 16; i++) { + b[i]=x[i]; + d[i]=a[i]=c[i]=0; + } + a[0]=d[0]=1; + for (i=254; i>=0; --i) { + r=(z[i>>>3]>>>(i&7))&1; + sel25519(a,b,r); + sel25519(c,d,r); + A(e,a,c); + Z(a,a,c); + A(c,b,d); + Z(b,b,d); + S(d,e); + S(f,a); + M(a,c,a); + M(c,b,e); + A(e,a,c); + Z(a,a,c); + S(b,a); + Z(c,d,f); + M(a,c,_121665); + A(a,a,d); + M(c,c,a); + M(a,d,f); + M(d,b,x); + S(b,e); + sel25519(a,b,r); + sel25519(c,d,r); + } + for (i = 0; i < 16; i++) { + x[i+16]=a[i]; + x[i+32]=c[i]; + x[i+48]=b[i]; + x[i+64]=d[i]; + } + var x32 = x.subarray(32); + var x16 = x.subarray(16); + inv25519(x32,x32); + M(x16,x16,x32); + pack25519(q,x16); + return 0; + } + + function crypto_scalarmult_base(q, n) { + return crypto_scalarmult(q, n, _9); + } + + function crypto_box_keypair(y, x) { + randombytes(x, 32); + return crypto_scalarmult_base(y, x); + } + + function add(p, q) { + var a = gf(), b = gf(), c = gf(), + d = gf(), e = gf(), f = gf(), + g = gf(), h = gf(), t = gf(); + + Z(a, p[1], p[0]); + Z(t, q[1], q[0]); + M(a, a, t); + A(b, p[0], p[1]); + A(t, q[0], q[1]); + M(b, b, t); + M(c, p[3], q[3]); + M(c, c, D2); + M(d, p[2], q[2]); + A(d, d, d); + Z(e, b, a); + Z(f, d, c); + A(g, d, c); + A(h, b, a); + + M(p[0], e, f); + M(p[1], h, g); + M(p[2], g, f); + M(p[3], e, h); + } + + function cswap(p, q, b) { + var i; + for (i = 0; i < 4; i++) { + sel25519(p[i], q[i], b); + } + } + + function pack(r, p) { + var tx = gf(), ty = gf(), zi = gf(); + inv25519(zi, p[2]); + M(tx, p[0], zi); + M(ty, p[1], zi); + pack25519(r, ty); + r[31] ^= par25519(tx) << 7; + } + + function scalarmult(p, q, s) { + var b, i; + set25519(p[0], gf0); + set25519(p[1], gf1); + set25519(p[2], gf1); + set25519(p[3], gf0); + for (i = 255; i >= 0; --i) { + b = (s[(i/8)|0] >> (i&7)) & 1; + cswap(p, q, b); + add(q, p); + add(p, p); + cswap(p, q, b); + } + } + + function scalarbase(p, s) { + var q = [gf(), gf(), gf(), gf()]; + set25519(q[0], X); + set25519(q[1], Y); + set25519(q[2], gf1); + M(q[3], X, Y); + scalarmult(p, q, s); + } + + function crypto_sign_keypair(pk, sk, seeded) { + var d; + var p = [gf(), gf(), gf(), gf()]; + var i; + + if (!seeded) randombytes(sk, 32); + d = nacl.hash(sk.subarray(0, 32)); + d[0] &= 248; + d[31] &= 127; + d[31] |= 64; + + scalarbase(p, d); + pack(pk, p); + + for (i = 0; i < 32; i++) sk[i+32] = pk[i]; + return 0; + } + + var L = new Float64Array([0xed, 0xd3, 0xf5, 0x5c, 0x1a, 0x63, 0x12, 0x58, 0xd6, 0x9c, 0xf7, 0xa2, 0xde, 0xf9, 0xde, 0x14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x10]); + + function modL(r, x) { + var carry, i, j, k; + for (i = 63; i >= 32; --i) { + carry = 0; + for (j = i - 32, k = i - 12; j < k; ++j) { + x[j] += carry - 16 * x[i] * L[j - (i - 32)]; + carry = Math.floor((x[j] + 128) / 256); + x[j] -= carry * 256; + } + x[j] += carry; + x[i] = 0; + } + carry = 0; + for (j = 0; j < 32; j++) { + x[j] += carry - (x[31] >> 4) * L[j]; + carry = x[j] >> 8; + x[j] &= 255; + } + for (j = 0; j < 32; j++) x[j] -= carry * L[j]; + for (i = 0; i < 32; i++) { + x[i+1] += x[i] >> 8; + r[i] = x[i] & 255; + } + } + + function reduce(r) { + var x = new Float64Array(64), i; + for (i = 0; i < 64; i++) x[i] = r[i]; + for (i = 0; i < 64; i++) r[i] = 0; + modL(r, x); + } + + // Note: difference from C - smlen returned, not passed as argument. + function crypto_sign(sm, m, n, sk) { + var d, h, r; + var i, j, x = new Float64Array(64); + var p = [gf(), gf(), gf(), gf()]; + + d = nacl.hash(sk.subarray(0, 32)); + d[0] &= 248; + d[31] &= 127; + d[31] |= 64; + + var smlen = n + 64; + for (i = 0; i < n; i++) sm[64 + i] = m[i]; + for (i = 0; i < 32; i++) sm[32 + i] = d[32 + i]; + + r = nacl.hash(sm.subarray(32, smlen)); + reduce(r); + scalarbase(p, r); + pack(sm, p); + + for (i = 32; i < 64; i++) sm[i] = sk[i]; + h = nacl.hash(sm.subarray(0, smlen)); + reduce(h); + + for (i = 0; i < 64; i++) x[i] = 0; + for (i = 0; i < 32; i++) x[i] = r[i]; + for (i = 0; i < 32; i++) { + for (j = 0; j < 32; j++) { + x[i+j] += h[i] * d[j]; + } + } + + modL(sm.subarray(32), x); + return smlen; + } + + function unpackneg(r, p) { + var t = gf(), chk = gf(), num = gf(), + den = gf(), den2 = gf(), den4 = gf(), + den6 = gf(); + + set25519(r[2], gf1); + unpack25519(r[1], p); + S(num, r[1]); + M(den, num, D); + Z(num, num, r[2]); + A(den, r[2], den); + + S(den2, den); + S(den4, den2); + M(den6, den4, den2); + M(t, den6, num); + M(t, t, den); + + pow2523(t, t); + M(t, t, num); + M(t, t, den); + M(t, t, den); + M(r[0], t, den); + + S(chk, r[0]); + M(chk, chk, den); + if (neq25519(chk, num)) M(r[0], r[0], I); + + S(chk, r[0]); + M(chk, chk, den); + if (neq25519(chk, num)) return -1; + + if (par25519(r[0]) === (p[31]>>7)) Z(r[0], gf0, r[0]); + + M(r[3], r[0], r[1]); + return 0; + } + + function crypto_sign_open(m, sm, n, pk) { + var i; + var t = new Uint8Array(32), h; + var p = [gf(), gf(), gf(), gf()], + q = [gf(), gf(), gf(), gf()]; + + if (n < 64) return -1; + + if (unpackneg(q, pk)) return -1; + + for (i = 0; i < n; i++) m[i] = sm[i]; + for (i = 0; i < 32; i++) m[i+32] = pk[i]; + h = nacl.hash(m.subarray(0, n)); + reduce(h); + scalarmult(p, q, h); + + scalarbase(q, sm.subarray(32)); + add(p, q); + pack(t, p); + + n -= 64; + if (crypto_verify_32(sm, 0, t, 0)) { + for (i = 0; i < n; i++) m[i] = 0; + return -1; + } + + for (i = 0; i < n; i++) m[i] = sm[i + 64]; + return n; + } + + var crypto_scalarmult_BYTES = 32, + crypto_scalarmult_SCALARBYTES = 32, + crypto_box_PUBLICKEYBYTES = 32, + crypto_box_SECRETKEYBYTES = 32, + crypto_sign_BYTES = 64, + crypto_sign_PUBLICKEYBYTES = 32, + crypto_sign_SECRETKEYBYTES = 64, + crypto_sign_SEEDBYTES = 32; + + function checkArrayTypes() { + for (var i = 0; i < arguments.length; i++) { + if (!(arguments[i] instanceof Uint8Array)) + throw new TypeError('unexpected type, use Uint8Array'); + } + } + + function cleanup(arr) { + for (var i = 0; i < arr.length; i++) arr[i] = 0; + } + + nacl.scalarMult = function(n, p) { + checkArrayTypes(n, p); + if (n.length !== crypto_scalarmult_SCALARBYTES) throw new Error('bad n size'); + if (p.length !== crypto_scalarmult_BYTES) throw new Error('bad p size'); + var q = new Uint8Array(crypto_scalarmult_BYTES); + crypto_scalarmult(q, n, p); + return q; + }; + + nacl.box = {}; + + nacl.box.keyPair = function() { + var pk = new Uint8Array(crypto_box_PUBLICKEYBYTES); + var sk = new Uint8Array(crypto_box_SECRETKEYBYTES); + crypto_box_keypair(pk, sk); + return {publicKey: pk, secretKey: sk}; + }; + + nacl.box.keyPair.fromSecretKey = function(secretKey) { + checkArrayTypes(secretKey); + if (secretKey.length !== crypto_box_SECRETKEYBYTES) + throw new Error('bad secret key size'); + var pk = new Uint8Array(crypto_box_PUBLICKEYBYTES); + crypto_scalarmult_base(pk, secretKey); + return {publicKey: pk, secretKey: new Uint8Array(secretKey)}; + }; + + nacl.sign = function(msg, secretKey) { + checkArrayTypes(msg, secretKey); + if (secretKey.length !== crypto_sign_SECRETKEYBYTES) + throw new Error('bad secret key size'); + var signedMsg = new Uint8Array(crypto_sign_BYTES+msg.length); + crypto_sign(signedMsg, msg, msg.length, secretKey); + return signedMsg; + }; + + nacl.sign.detached = function(msg, secretKey) { + var signedMsg = nacl.sign(msg, secretKey); + var sig = new Uint8Array(crypto_sign_BYTES); + for (var i = 0; i < sig.length; i++) sig[i] = signedMsg[i]; + return sig; + }; + + nacl.sign.detached.verify = function(msg, sig, publicKey) { + checkArrayTypes(msg, sig, publicKey); + if (sig.length !== crypto_sign_BYTES) + throw new Error('bad signature size'); + if (publicKey.length !== crypto_sign_PUBLICKEYBYTES) + throw new Error('bad public key size'); + var sm = new Uint8Array(crypto_sign_BYTES + msg.length); + var m = new Uint8Array(crypto_sign_BYTES + msg.length); + var i; + for (i = 0; i < crypto_sign_BYTES; i++) sm[i] = sig[i]; + for (i = 0; i < msg.length; i++) sm[i+crypto_sign_BYTES] = msg[i]; + return (crypto_sign_open(m, sm, sm.length, publicKey) >= 0); + }; + + nacl.sign.keyPair = function() { + var pk = new Uint8Array(crypto_sign_PUBLICKEYBYTES); + var sk = new Uint8Array(crypto_sign_SECRETKEYBYTES); + crypto_sign_keypair(pk, sk); + return {publicKey: pk, secretKey: sk}; + }; + + nacl.sign.keyPair.fromSecretKey = function(secretKey) { + checkArrayTypes(secretKey); + if (secretKey.length !== crypto_sign_SECRETKEYBYTES) + throw new Error('bad secret key size'); + var pk = new Uint8Array(crypto_sign_PUBLICKEYBYTES); + for (var i = 0; i < pk.length; i++) pk[i] = secretKey[32+i]; + return {publicKey: pk, secretKey: new Uint8Array(secretKey)}; + }; + + nacl.sign.keyPair.fromSeed = function(seed) { + checkArrayTypes(seed); + if (seed.length !== crypto_sign_SEEDBYTES) + throw new Error('bad seed size'); + var pk = new Uint8Array(crypto_sign_PUBLICKEYBYTES); + var sk = new Uint8Array(crypto_sign_SECRETKEYBYTES); + for (var i = 0; i < 32; i++) sk[i] = seed[i]; + crypto_sign_keypair(pk, sk, true); + return {publicKey: pk, secretKey: sk}; + }; + + nacl.setPRNG = function(fn) { + randombytes = fn; + }; + + (function() { + // Initialize PRNG if environment provides CSPRNG. + // If not, methods calling randombytes will throw. + var crypto = typeof self !== 'undefined' ? (self.crypto || self.msCrypto) : null; + if (crypto && crypto.getRandomValues) { + // Browsers. + var QUOTA = 65536; + nacl.setPRNG(function(x, n) { + var i, v = new Uint8Array(n); + for (i = 0; i < n; i += QUOTA) { + crypto.getRandomValues(v.subarray(i, i + Math.min(n - i, QUOTA))); + } + for (i = 0; i < n; i++) x[i] = v[i]; + cleanup(v); + }); + } else if (typeof commonjsRequire !== 'undefined') { + // Node.js. + crypto = void('crypto'); + if (crypto && crypto.randomBytes) { + nacl.setPRNG(function(x, n) { + var i, v = crypto.randomBytes(n); + for (i = 0; i < n; i++) x[i] = v[i]; + cleanup(v); + }); + } + } + })(); + + })(module.exports ? module.exports : (self.nacl = self.nacl || {})); + }); + + // GPG4Browsers - An OpenPGP implementation in javascript + + const nodeCrypto$5 = util.getNodeCrypto(); + + /** + * Retrieve secure random byte array of the specified length + * @param {Integer} length - Length in bytes to generate + * @returns {Uint8Array} Random byte array. + */ + function getRandomBytes(length) { + const buf = new Uint8Array(length); + if (typeof crypto !== 'undefined' && crypto.getRandomValues) { + crypto.getRandomValues(buf); + } else if (nodeCrypto$5) { + const bytes = nodeCrypto$5.randomBytes(buf.length); + buf.set(bytes); + } else { + throw new Error('No secure random number generator available.'); + } + return buf; + } + + /** + * Create a secure random BigInteger that is greater than or equal to min and less than max. + * @param {module:BigInteger} min - Lower bound, included + * @param {module:BigInteger} max - Upper bound, excluded + * @returns {Promise} Random BigInteger. + * @async + */ + async function getRandomBigInteger(min, max) { + const BigInteger = await util.getBigInteger(); + + if (max.lt(min)) { + throw new Error('Illegal parameter value: max <= min'); + } + + const modulus = max.sub(min); + const bytes = modulus.byteLength(); + + // Using a while loop is necessary to avoid bias introduced by the mod operation. + // However, we request 64 extra random bits so that the bias is negligible. + // Section B.1.1 here: https://nvlpubs.nist.gov/nistpubs/FIPS/NIST.FIPS.186-4.pdf + const r = new BigInteger(await getRandomBytes(bytes + 8)); + return r.mod(modulus).add(min); + } + + var random = /*#__PURE__*/Object.freeze({ + __proto__: null, + getRandomBytes: getRandomBytes, + getRandomBigInteger: getRandomBigInteger + }); + + // OpenPGP.js - An OpenPGP implementation in javascript + + /** + * Generate a probably prime random number + * @param {Integer} bits - Bit length of the prime + * @param {BigInteger} e - Optional RSA exponent to check against the prime + * @param {Integer} k - Optional number of iterations of Miller-Rabin test + * @returns BigInteger + * @async + */ + async function randomProbablePrime(bits, e, k) { + const BigInteger = await util.getBigInteger(); + const one = new BigInteger(1); + const min = one.leftShift(new BigInteger(bits - 1)); + const thirty = new BigInteger(30); + /* + * We can avoid any multiples of 3 and 5 by looking at n mod 30 + * n mod 30 = 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 + * the next possible prime is mod 30: + * 1 7 7 7 7 7 7 11 11 11 11 13 13 17 17 17 17 19 19 23 23 23 23 29 29 29 29 29 29 1 + */ + const adds = [1, 6, 5, 4, 3, 2, 1, 4, 3, 2, 1, 2, 1, 4, 3, 2, 1, 2, 1, 4, 3, 2, 1, 6, 5, 4, 3, 2, 1, 2]; + + const n = await getRandomBigInteger(min, min.leftShift(one)); + let i = n.mod(thirty).toNumber(); + + do { + n.iadd(new BigInteger(adds[i])); + i = (i + adds[i]) % adds.length; + // If reached the maximum, go back to the minimum. + if (n.bitLength() > bits) { + n.imod(min.leftShift(one)).iadd(min); + i = n.mod(thirty).toNumber(); + } + } while (!await isProbablePrime(n, e, k)); + return n; + } + + /** + * Probabilistic primality testing + * @param {BigInteger} n - Number to test + * @param {BigInteger} e - Optional RSA exponent to check against the prime + * @param {Integer} k - Optional number of iterations of Miller-Rabin test + * @returns {boolean} + * @async + */ + async function isProbablePrime(n, e, k) { + if (e && !n.dec().gcd(e).isOne()) { + return false; + } + if (!await divisionTest(n)) { + return false; + } + if (!await fermat(n)) { + return false; + } + if (!await millerRabin(n, k)) { + return false; + } + // TODO implement the Lucas test + // See Section C.3.3 here: https://nvlpubs.nist.gov/nistpubs/FIPS/NIST.FIPS.186-4.pdf + return true; + } + + /** + * Tests whether n is probably prime or not using Fermat's test with b = 2. + * Fails if b^(n-1) mod n != 1. + * @param {BigInteger} n - Number to test + * @param {BigInteger} b - Optional Fermat test base + * @returns {boolean} + */ + async function fermat(n, b) { + const BigInteger = await util.getBigInteger(); + b = b || new BigInteger(2); + return b.modExp(n.dec(), n).isOne(); + } + + async function divisionTest(n) { + const BigInteger = await util.getBigInteger(); + return smallPrimes.every(m => { + return n.mod(new BigInteger(m)) !== 0; + }); + } + + // https://github.com/gpg/libgcrypt/blob/master/cipher/primegen.c + const smallPrimes = [ + 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, + 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97, 101, + 103, 107, 109, 113, 127, 131, 137, 139, 149, 151, + 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, + 211, 223, 227, 229, 233, 239, 241, 251, 257, 263, + 269, 271, 277, 281, 283, 293, 307, 311, 313, 317, + 331, 337, 347, 349, 353, 359, 367, 373, 379, 383, + 389, 397, 401, 409, 419, 421, 431, 433, 439, 443, + 449, 457, 461, 463, 467, 479, 487, 491, 499, 503, + 509, 521, 523, 541, 547, 557, 563, 569, 571, 577, + 587, 593, 599, 601, 607, 613, 617, 619, 631, 641, + 643, 647, 653, 659, 661, 673, 677, 683, 691, 701, + 709, 719, 727, 733, 739, 743, 751, 757, 761, 769, + 773, 787, 797, 809, 811, 821, 823, 827, 829, 839, + 853, 857, 859, 863, 877, 881, 883, 887, 907, 911, + 919, 929, 937, 941, 947, 953, 967, 971, 977, 983, + 991, 997, 1009, 1013, 1019, 1021, 1031, 1033, + 1039, 1049, 1051, 1061, 1063, 1069, 1087, 1091, + 1093, 1097, 1103, 1109, 1117, 1123, 1129, 1151, + 1153, 1163, 1171, 1181, 1187, 1193, 1201, 1213, + 1217, 1223, 1229, 1231, 1237, 1249, 1259, 1277, + 1279, 1283, 1289, 1291, 1297, 1301, 1303, 1307, + 1319, 1321, 1327, 1361, 1367, 1373, 1381, 1399, + 1409, 1423, 1427, 1429, 1433, 1439, 1447, 1451, + 1453, 1459, 1471, 1481, 1483, 1487, 1489, 1493, + 1499, 1511, 1523, 1531, 1543, 1549, 1553, 1559, + 1567, 1571, 1579, 1583, 1597, 1601, 1607, 1609, + 1613, 1619, 1621, 1627, 1637, 1657, 1663, 1667, + 1669, 1693, 1697, 1699, 1709, 1721, 1723, 1733, + 1741, 1747, 1753, 1759, 1777, 1783, 1787, 1789, + 1801, 1811, 1823, 1831, 1847, 1861, 1867, 1871, + 1873, 1877, 1879, 1889, 1901, 1907, 1913, 1931, + 1933, 1949, 1951, 1973, 1979, 1987, 1993, 1997, + 1999, 2003, 2011, 2017, 2027, 2029, 2039, 2053, + 2063, 2069, 2081, 2083, 2087, 2089, 2099, 2111, + 2113, 2129, 2131, 2137, 2141, 2143, 2153, 2161, + 2179, 2203, 2207, 2213, 2221, 2237, 2239, 2243, + 2251, 2267, 2269, 2273, 2281, 2287, 2293, 2297, + 2309, 2311, 2333, 2339, 2341, 2347, 2351, 2357, + 2371, 2377, 2381, 2383, 2389, 2393, 2399, 2411, + 2417, 2423, 2437, 2441, 2447, 2459, 2467, 2473, + 2477, 2503, 2521, 2531, 2539, 2543, 2549, 2551, + 2557, 2579, 2591, 2593, 2609, 2617, 2621, 2633, + 2647, 2657, 2659, 2663, 2671, 2677, 2683, 2687, + 2689, 2693, 2699, 2707, 2711, 2713, 2719, 2729, + 2731, 2741, 2749, 2753, 2767, 2777, 2789, 2791, + 2797, 2801, 2803, 2819, 2833, 2837, 2843, 2851, + 2857, 2861, 2879, 2887, 2897, 2903, 2909, 2917, + 2927, 2939, 2953, 2957, 2963, 2969, 2971, 2999, + 3001, 3011, 3019, 3023, 3037, 3041, 3049, 3061, + 3067, 3079, 3083, 3089, 3109, 3119, 3121, 3137, + 3163, 3167, 3169, 3181, 3187, 3191, 3203, 3209, + 3217, 3221, 3229, 3251, 3253, 3257, 3259, 3271, + 3299, 3301, 3307, 3313, 3319, 3323, 3329, 3331, + 3343, 3347, 3359, 3361, 3371, 3373, 3389, 3391, + 3407, 3413, 3433, 3449, 3457, 3461, 3463, 3467, + 3469, 3491, 3499, 3511, 3517, 3527, 3529, 3533, + 3539, 3541, 3547, 3557, 3559, 3571, 3581, 3583, + 3593, 3607, 3613, 3617, 3623, 3631, 3637, 3643, + 3659, 3671, 3673, 3677, 3691, 3697, 3701, 3709, + 3719, 3727, 3733, 3739, 3761, 3767, 3769, 3779, + 3793, 3797, 3803, 3821, 3823, 3833, 3847, 3851, + 3853, 3863, 3877, 3881, 3889, 3907, 3911, 3917, + 3919, 3923, 3929, 3931, 3943, 3947, 3967, 3989, + 4001, 4003, 4007, 4013, 4019, 4021, 4027, 4049, + 4051, 4057, 4073, 4079, 4091, 4093, 4099, 4111, + 4127, 4129, 4133, 4139, 4153, 4157, 4159, 4177, + 4201, 4211, 4217, 4219, 4229, 4231, 4241, 4243, + 4253, 4259, 4261, 4271, 4273, 4283, 4289, 4297, + 4327, 4337, 4339, 4349, 4357, 4363, 4373, 4391, + 4397, 4409, 4421, 4423, 4441, 4447, 4451, 4457, + 4463, 4481, 4483, 4493, 4507, 4513, 4517, 4519, + 4523, 4547, 4549, 4561, 4567, 4583, 4591, 4597, + 4603, 4621, 4637, 4639, 4643, 4649, 4651, 4657, + 4663, 4673, 4679, 4691, 4703, 4721, 4723, 4729, + 4733, 4751, 4759, 4783, 4787, 4789, 4793, 4799, + 4801, 4813, 4817, 4831, 4861, 4871, 4877, 4889, + 4903, 4909, 4919, 4931, 4933, 4937, 4943, 4951, + 4957, 4967, 4969, 4973, 4987, 4993, 4999 + ]; + + + // Miller-Rabin - Miller Rabin algorithm for primality test + // Copyright Fedor Indutny, 2014. + // + // This software is licensed under the MIT License. + // + // Permission is hereby granted, free of charge, to any person obtaining a + // copy of this software and associated documentation files (the + // "Software"), to deal in the Software without restriction, including + // without limitation the rights to use, copy, modify, merge, publish, + // distribute, sublicense, and/or sell copies of the Software, and to permit + // persons to whom the Software is furnished to do so, subject to the + // following conditions: + // + // The above copyright notice and this permission notice shall be included + // in all copies or substantial portions of the Software. + // + // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS + // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN + // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, + // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR + // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE + // USE OR OTHER DEALINGS IN THE SOFTWARE. + + // Adapted on Jan 2018 from version 4.0.1 at https://github.com/indutny/miller-rabin + + // Sample syntax for Fixed-Base Miller-Rabin: + // millerRabin(n, k, () => new BN(small_primes[Math.random() * small_primes.length | 0])) + + /** + * Tests whether n is probably prime or not using the Miller-Rabin test. + * See HAC Remark 4.28. + * @param {BigInteger} n - Number to test + * @param {Integer} k - Optional number of iterations of Miller-Rabin test + * @param {Function} rand - Optional function to generate potential witnesses + * @returns {boolean} + * @async + */ + async function millerRabin(n, k, rand) { + const BigInteger = await util.getBigInteger(); + const len = n.bitLength(); + + if (!k) { + k = Math.max(1, (len / 48) | 0); + } + + const n1 = n.dec(); // n - 1 + + // Find d and s, (n - 1) = (2 ^ s) * d; + let s = 0; + while (!n1.getBit(s)) { s++; } + const d = n.rightShift(new BigInteger(s)); + + for (; k > 0; k--) { + const a = rand ? rand() : await getRandomBigInteger(new BigInteger(2), n1); + + let x = a.modExp(d, n); + if (x.isOne() || x.equal(n1)) { + continue; + } + + let i; + for (i = 1; i < s; i++) { + x = x.mul(x).mod(n); + + if (x.isOne()) { + return false; + } + if (x.equal(n1)) { + break; + } + } + + if (i === s) { + return false; + } + } + + return true; + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * ASN1 object identifiers for hashes + * @see {@link https://tools.ietf.org/html/rfc4880#section-5.2.2} + */ + const hash_headers = []; + hash_headers[1] = [0x30, 0x20, 0x30, 0x0c, 0x06, 0x08, 0x2a, 0x86, 0x48, 0x86, 0xf7, 0x0d, 0x02, 0x05, 0x05, 0x00, 0x04, + 0x10]; + hash_headers[2] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2b, 0x0e, 0x03, 0x02, 0x1a, 0x05, 0x00, 0x04, 0x14]; + hash_headers[3] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2B, 0x24, 0x03, 0x02, 0x01, 0x05, 0x00, 0x04, 0x14]; + hash_headers[8] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x01, 0x05, 0x00, + 0x04, 0x20]; + hash_headers[9] = [0x30, 0x41, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x02, 0x05, 0x00, + 0x04, 0x30]; + hash_headers[10] = [0x30, 0x51, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x03, 0x05, + 0x00, 0x04, 0x40]; + hash_headers[11] = [0x30, 0x2d, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x04, 0x05, + 0x00, 0x04, 0x1C]; + + /** + * Create padding with secure random data + * @private + * @param {Integer} length - Length of the padding in bytes + * @returns {Uint8Array} Random padding. + */ + function getPKCS1Padding(length) { + const result = new Uint8Array(length); + let count = 0; + while (count < length) { + const randomBytes = getRandomBytes(length - count); + for (let i = 0; i < randomBytes.length; i++) { + if (randomBytes[i] !== 0) { + result[count++] = randomBytes[i]; + } + } + } + return result; + } + + /** + * Create a EME-PKCS1-v1_5 padded message + * @see {@link https://tools.ietf.org/html/rfc4880#section-13.1.1|RFC 4880 13.1.1} + * @param {Uint8Array} message - Message to be encoded + * @param {Integer} keyLength - The length in octets of the key modulus + * @returns {Uint8Array} EME-PKCS1 padded message. + */ + function emeEncode(message, keyLength) { + const mLength = message.length; + // length checking + if (mLength > keyLength - 11) { + throw new Error('Message too long'); + } + // Generate an octet string PS of length k - mLen - 3 consisting of + // pseudo-randomly generated nonzero octets + const PS = getPKCS1Padding(keyLength - mLength - 3); + // Concatenate PS, the message M, and other padding to form an + // encoded message EM of length k octets as EM = 0x00 || 0x02 || PS || 0x00 || M. + const encoded = new Uint8Array(keyLength); + // 0x00 byte + encoded[1] = 2; + encoded.set(PS, 2); + // 0x00 bytes + encoded.set(message, keyLength - mLength); + return encoded; + } + + /** + * Decode a EME-PKCS1-v1_5 padded message + * @see {@link https://tools.ietf.org/html/rfc4880#section-13.1.2|RFC 4880 13.1.2} + * @param {Uint8Array} encoded - Encoded message bytes + * @param {Uint8Array} randomPayload - Data to return in case of decoding error (needed for constant-time processing) + * @returns {Uint8Array} decoded data or `randomPayload` (on error, if given) + * @throws {Error} on decoding failure, unless `randomPayload` is provided + */ + function emeDecode(encoded, randomPayload) { + // encoded format: 0x00 0x02 0x00 + let offset = 2; + let separatorNotFound = 1; + for (let j = offset; j < encoded.length; j++) { + separatorNotFound &= encoded[j] !== 0; + offset += separatorNotFound; + } + + const psLen = offset - 2; + const payload = encoded.subarray(offset + 1); // discard the 0x00 separator + const isValidPadding = encoded[0] === 0 & encoded[1] === 2 & psLen >= 8 & !separatorNotFound; + + if (randomPayload) { + return util.selectUint8Array(isValidPadding, payload, randomPayload); + } + + if (isValidPadding) { + return payload; + } + + throw new Error('Decryption error'); + } + + /** + * Create a EMSA-PKCS1-v1_5 padded message + * @see {@link https://tools.ietf.org/html/rfc4880#section-13.1.3|RFC 4880 13.1.3} + * @param {Integer} algo - Hash algorithm type used + * @param {Uint8Array} hashed - Message to be encoded + * @param {Integer} emLen - Intended length in octets of the encoded message + * @returns {Uint8Array} Encoded message. + */ + async function emsaEncode(algo, hashed, emLen) { + let i; + if (hashed.length !== hash.getHashByteLength(algo)) { + throw new Error('Invalid hash length'); + } + // produce an ASN.1 DER value for the hash function used. + // Let T be the full hash prefix + const hashPrefix = new Uint8Array(hash_headers[algo].length); + for (i = 0; i < hash_headers[algo].length; i++) { + hashPrefix[i] = hash_headers[algo][i]; + } + // and let tLen be the length in octets prefix and hashed data + const tLen = hashPrefix.length + hashed.length; + if (emLen < tLen + 11) { + throw new Error('Intended encoded message length too short'); + } + // an octet string PS consisting of emLen - tLen - 3 octets with hexadecimal value 0xFF + // The length of PS will be at least 8 octets + const PS = new Uint8Array(emLen - tLen - 3).fill(0xff); + + // Concatenate PS, the hash prefix, hashed data, and other padding to form the + // encoded message EM as EM = 0x00 || 0x01 || PS || 0x00 || prefix || hashed + const EM = new Uint8Array(emLen); + EM[1] = 0x01; + EM.set(PS, 2); + EM.set(hashPrefix, emLen - tLen); + EM.set(hashed, emLen - hashed.length); + return EM; + } + + var pkcs1 = /*#__PURE__*/Object.freeze({ + __proto__: null, + emeEncode: emeEncode, + emeDecode: emeDecode, + emsaEncode: emsaEncode + }); + + // GPG4Browsers - An OpenPGP implementation in javascript + + const webCrypto$5 = util.getWebCrypto(); + const nodeCrypto$6 = util.getNodeCrypto(); + const asn1 = nodeCrypto$6 ? void('asn1.js') : undefined; + + /* eslint-disable no-invalid-this */ + const RSAPrivateKey = nodeCrypto$6 ? asn1.define('RSAPrivateKey', function () { + this.seq().obj( // used for native NodeJS crypto + this.key('version').int(), // 0 + this.key('modulus').int(), // n + this.key('publicExponent').int(), // e + this.key('privateExponent').int(), // d + this.key('prime1').int(), // p + this.key('prime2').int(), // q + this.key('exponent1').int(), // dp + this.key('exponent2').int(), // dq + this.key('coefficient').int() // u + ); + }) : undefined; + + const RSAPublicKey = nodeCrypto$6 ? asn1.define('RSAPubliceKey', function () { + this.seq().obj( // used for native NodeJS crypto + this.key('modulus').int(), // n + this.key('publicExponent').int() // e + ); + }) : undefined; + /* eslint-enable no-invalid-this */ + + /** Create signature + * @param {module:enums.hash} hashAlgo - Hash algorithm + * @param {Uint8Array} data - Message + * @param {Uint8Array} n - RSA public modulus + * @param {Uint8Array} e - RSA public exponent + * @param {Uint8Array} d - RSA private exponent + * @param {Uint8Array} p - RSA private prime p + * @param {Uint8Array} q - RSA private prime q + * @param {Uint8Array} u - RSA private coefficient + * @param {Uint8Array} hashed - Hashed message + * @returns {Promise} RSA Signature. + * @async + */ + async function sign(hashAlgo, data, n, e, d, p, q, u, hashed) { + if (data && !util.isStream(data)) { + if (util.getWebCrypto()) { + try { + return await webSign(enums.read(enums.webHash, hashAlgo), data, n, e, d, p, q, u); + } catch (err) { + util.printDebugError(err); + } + } else if (util.getNodeCrypto()) { + return nodeSign(hashAlgo, data, n, e, d, p, q, u); + } + } + return bnSign(hashAlgo, n, d, hashed); + } + + /** + * Verify signature + * @param {module:enums.hash} hashAlgo - Hash algorithm + * @param {Uint8Array} data - Message + * @param {Uint8Array} s - Signature + * @param {Uint8Array} n - RSA public modulus + * @param {Uint8Array} e - RSA public exponent + * @param {Uint8Array} hashed - Hashed message + * @returns {Boolean} + * @async + */ + async function verify(hashAlgo, data, s, n, e, hashed) { + if (data && !util.isStream(data)) { + if (util.getWebCrypto()) { + try { + return await webVerify(enums.read(enums.webHash, hashAlgo), data, s, n, e); + } catch (err) { + util.printDebugError(err); + } + } else if (util.getNodeCrypto()) { + return nodeVerify(hashAlgo, data, s, n, e); + } + } + return bnVerify(hashAlgo, s, n, e, hashed); + } + + /** + * Encrypt message + * @param {Uint8Array} data - Message + * @param {Uint8Array} n - RSA public modulus + * @param {Uint8Array} e - RSA public exponent + * @returns {Promise} RSA Ciphertext. + * @async + */ + async function encrypt$1(data, n, e) { + if (util.getNodeCrypto()) { + return nodeEncrypt$1(data, n, e); + } + return bnEncrypt(data, n, e); + } + + /** + * Decrypt RSA message + * @param {Uint8Array} m - Message + * @param {Uint8Array} n - RSA public modulus + * @param {Uint8Array} e - RSA public exponent + * @param {Uint8Array} d - RSA private exponent + * @param {Uint8Array} p - RSA private prime p + * @param {Uint8Array} q - RSA private prime q + * @param {Uint8Array} u - RSA private coefficient + * @param {Uint8Array} randomPayload - Data to return on decryption error, instead of throwing + * (needed for constant-time processing) + * @returns {Promise} RSA Plaintext. + * @throws {Error} on decryption error, unless `randomPayload` is given + * @async + */ + async function decrypt$1(data, n, e, d, p, q, u, randomPayload) { + if (util.getNodeCrypto()) { + return nodeDecrypt$1(data, n, e, d, p, q, u, randomPayload); + } + return bnDecrypt(data, n, e, d, p, q, u, randomPayload); + } + + /** + * Generate a new random private key B bits long with public exponent E. + * + * When possible, webCrypto or nodeCrypto is used. Otherwise, primes are generated using + * 40 rounds of the Miller-Rabin probabilistic random prime generation algorithm. + * @see module:crypto/public_key/prime + * @param {Integer} bits - RSA bit length + * @param {Integer} e - RSA public exponent + * @returns {{n, e, d, + * p, q ,u: Uint8Array}} RSA public modulus, RSA public exponent, RSA private exponent, + * RSA private prime p, RSA private prime q, u = p ** -1 mod q + * @async + */ + async function generate(bits, e) { + const BigInteger = await util.getBigInteger(); + + e = new BigInteger(e); + + // Native RSA keygen using Web Crypto + if (util.getWebCrypto()) { + const keyGenOpt = { + name: 'RSASSA-PKCS1-v1_5', + modulusLength: bits, // the specified keysize in bits + publicExponent: e.toUint8Array(), // take three bytes (max 65537) for exponent + hash: { + name: 'SHA-1' // not required for actual RSA keys, but for crypto api 'sign' and 'verify' + } + }; + const keyPair = await webCrypto$5.generateKey(keyGenOpt, true, ['sign', 'verify']); + + // export the generated keys as JsonWebKey (JWK) + // https://tools.ietf.org/html/draft-ietf-jose-json-web-key-33 + const jwk = await webCrypto$5.exportKey('jwk', keyPair.privateKey); + // map JWK parameters to corresponding OpenPGP names + return { + n: b64ToUint8Array(jwk.n), + e: e.toUint8Array(), + d: b64ToUint8Array(jwk.d), + // switch p and q + p: b64ToUint8Array(jwk.q), + q: b64ToUint8Array(jwk.p), + // Since p and q are switched in places, u is the inverse of jwk.q + u: b64ToUint8Array(jwk.qi) + }; + } else if (util.getNodeCrypto() && nodeCrypto$6.generateKeyPair && RSAPrivateKey) { + const opts = { + modulusLength: bits, + publicExponent: e.toNumber(), + publicKeyEncoding: { type: 'pkcs1', format: 'der' }, + privateKeyEncoding: { type: 'pkcs1', format: 'der' } + }; + const prv = await new Promise((resolve, reject) => { + nodeCrypto$6.generateKeyPair('rsa', opts, (err, _, der) => { + if (err) { + reject(err); + } else { + resolve(RSAPrivateKey.decode(der, 'der')); + } + }); + }); + /** + * OpenPGP spec differs from DER spec, DER: `u = (inverse of q) mod p`, OpenPGP: `u = (inverse of p) mod q`. + * @link https://tools.ietf.org/html/rfc3447#section-3.2 + * @link https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-08#section-5.6.1 + */ + return { + n: prv.modulus.toArrayLike(Uint8Array), + e: prv.publicExponent.toArrayLike(Uint8Array), + d: prv.privateExponent.toArrayLike(Uint8Array), + // switch p and q + p: prv.prime2.toArrayLike(Uint8Array), + q: prv.prime1.toArrayLike(Uint8Array), + // Since p and q are switched in places, we can keep u as defined by DER + u: prv.coefficient.toArrayLike(Uint8Array) + }; + } + + // RSA keygen fallback using 40 iterations of the Miller-Rabin test + // See https://stackoverflow.com/a/6330138 for justification + // Also see section C.3 here: https://nvlpubs.nist.gov/nistpubs/FIPS/NIST + let p; + let q; + let n; + do { + q = await randomProbablePrime(bits - (bits >> 1), e, 40); + p = await randomProbablePrime(bits >> 1, e, 40); + n = p.mul(q); + } while (n.bitLength() !== bits); + + const phi = p.dec().imul(q.dec()); + + if (q.lt(p)) { + [p, q] = [q, p]; + } + + return { + n: n.toUint8Array(), + e: e.toUint8Array(), + d: e.modInv(phi).toUint8Array(), + p: p.toUint8Array(), + q: q.toUint8Array(), + // dp: d.mod(p.subn(1)), + // dq: d.mod(q.subn(1)), + u: p.modInv(q).toUint8Array() + }; + } + + /** + * Validate RSA parameters + * @param {Uint8Array} n - RSA public modulus + * @param {Uint8Array} e - RSA public exponent + * @param {Uint8Array} d - RSA private exponent + * @param {Uint8Array} p - RSA private prime p + * @param {Uint8Array} q - RSA private prime q + * @param {Uint8Array} u - RSA inverse of p w.r.t. q + * @returns {Promise} Whether params are valid. + * @async + */ + async function validateParams(n, e, d, p, q, u) { + const BigInteger = await util.getBigInteger(); + n = new BigInteger(n); + p = new BigInteger(p); + q = new BigInteger(q); + + // expect pq = n + if (!p.mul(q).equal(n)) { + return false; + } + + const two = new BigInteger(2); + // expect p*u = 1 mod q + u = new BigInteger(u); + if (!p.mul(u).mod(q).isOne()) { + return false; + } + + e = new BigInteger(e); + d = new BigInteger(d); + /** + * In RSA pkcs#1 the exponents (d, e) are inverses modulo lcm(p-1, q-1) + * We check that [de = 1 mod (p-1)] and [de = 1 mod (q-1)] + * By CRT on coprime factors of (p-1, q-1) it follows that [de = 1 mod lcm(p-1, q-1)] + * + * We blind the multiplication with r, and check that rde = r mod lcm(p-1, q-1) + */ + const nSizeOver3 = new BigInteger(Math.floor(n.bitLength() / 3)); + const r = await getRandomBigInteger(two, two.leftShift(nSizeOver3)); // r in [ 2, 2^{|n|/3} ) < p and q + const rde = r.mul(d).mul(e); + + const areInverses = rde.mod(p.dec()).equal(r) && rde.mod(q.dec()).equal(r); + if (!areInverses) { + return false; + } + + return true; + } + + async function bnSign(hashAlgo, n, d, hashed) { + const BigInteger = await util.getBigInteger(); + n = new BigInteger(n); + const m = new BigInteger(await emsaEncode(hashAlgo, hashed, n.byteLength())); + d = new BigInteger(d); + if (m.gte(n)) { + throw new Error('Message size cannot exceed modulus size'); + } + return m.modExp(d, n).toUint8Array('be', n.byteLength()); + } + + async function webSign(hashName, data, n, e, d, p, q, u) { + /** OpenPGP keys require that p < q, and Safari Web Crypto requires that p > q. + * We swap them in privateToJWK, so it usually works out, but nevertheless, + * not all OpenPGP keys are compatible with this requirement. + * OpenPGP.js used to generate RSA keys the wrong way around (p > q), and still + * does if the underlying Web Crypto does so (though the tested implementations + * don't do so). + */ + const jwk = await privateToJWK(n, e, d, p, q, u); + const algo = { + name: 'RSASSA-PKCS1-v1_5', + hash: { name: hashName } + }; + const key = await webCrypto$5.importKey('jwk', jwk, algo, false, ['sign']); + return new Uint8Array(await webCrypto$5.sign('RSASSA-PKCS1-v1_5', key, data)); + } + + async function nodeSign(hashAlgo, data, n, e, d, p, q, u) { + const { default: BN } = await Promise.resolve().then(function () { return bn$1; }); + const pBNum = new BN(p); + const qBNum = new BN(q); + const dBNum = new BN(d); + const dq = dBNum.mod(qBNum.subn(1)); // d mod (q-1) + const dp = dBNum.mod(pBNum.subn(1)); // d mod (p-1) + const sign = nodeCrypto$6.createSign(enums.read(enums.hash, hashAlgo)); + sign.write(data); + sign.end(); + const keyObject = { + version: 0, + modulus: new BN(n), + publicExponent: new BN(e), + privateExponent: new BN(d), + // switch p and q + prime1: new BN(q), + prime2: new BN(p), + // switch dp and dq + exponent1: dq, + exponent2: dp, + coefficient: new BN(u) + }; + if (typeof nodeCrypto$6.createPrivateKey !== 'undefined') { //from version 11.6.0 Node supports der encoded key objects + const der = RSAPrivateKey.encode(keyObject, 'der'); + return new Uint8Array(sign.sign({ key: der, format: 'der', type: 'pkcs1' })); + } + const pem = RSAPrivateKey.encode(keyObject, 'pem', { + label: 'RSA PRIVATE KEY' + }); + return new Uint8Array(sign.sign(pem)); + } + + async function bnVerify(hashAlgo, s, n, e, hashed) { + const BigInteger = await util.getBigInteger(); + n = new BigInteger(n); + s = new BigInteger(s); + e = new BigInteger(e); + if (s.gte(n)) { + throw new Error('Signature size cannot exceed modulus size'); + } + const EM1 = s.modExp(e, n).toUint8Array('be', n.byteLength()); + const EM2 = await emsaEncode(hashAlgo, hashed, n.byteLength()); + return util.equalsUint8Array(EM1, EM2); + } + + async function webVerify(hashName, data, s, n, e) { + const jwk = publicToJWK(n, e); + const key = await webCrypto$5.importKey('jwk', jwk, { + name: 'RSASSA-PKCS1-v1_5', + hash: { name: hashName } + }, false, ['verify']); + return webCrypto$5.verify('RSASSA-PKCS1-v1_5', key, s, data); + } + + async function nodeVerify(hashAlgo, data, s, n, e) { + const { default: BN } = await Promise.resolve().then(function () { return bn$1; }); + + const verify = nodeCrypto$6.createVerify(enums.read(enums.hash, hashAlgo)); + verify.write(data); + verify.end(); + const keyObject = { + modulus: new BN(n), + publicExponent: new BN(e) + }; + let key; + if (typeof nodeCrypto$6.createPrivateKey !== 'undefined') { //from version 11.6.0 Node supports der encoded key objects + const der = RSAPublicKey.encode(keyObject, 'der'); + key = { key: der, format: 'der', type: 'pkcs1' }; + } else { + key = RSAPublicKey.encode(keyObject, 'pem', { + label: 'RSA PUBLIC KEY' + }); + } + try { + return await verify.verify(key, s); + } catch (err) { + return false; + } + } + + async function nodeEncrypt$1(data, n, e) { + const { default: BN } = await Promise.resolve().then(function () { return bn$1; }); + + const keyObject = { + modulus: new BN(n), + publicExponent: new BN(e) + }; + let key; + if (typeof nodeCrypto$6.createPrivateKey !== 'undefined') { + const der = RSAPublicKey.encode(keyObject, 'der'); + key = { key: der, format: 'der', type: 'pkcs1', padding: nodeCrypto$6.constants.RSA_PKCS1_PADDING }; + } else { + const pem = RSAPublicKey.encode(keyObject, 'pem', { + label: 'RSA PUBLIC KEY' + }); + key = { key: pem, padding: nodeCrypto$6.constants.RSA_PKCS1_PADDING }; + } + return new Uint8Array(nodeCrypto$6.publicEncrypt(key, data)); + } + + async function bnEncrypt(data, n, e) { + const BigInteger = await util.getBigInteger(); + n = new BigInteger(n); + data = new BigInteger(emeEncode(data, n.byteLength())); + e = new BigInteger(e); + if (data.gte(n)) { + throw new Error('Message size cannot exceed modulus size'); + } + return data.modExp(e, n).toUint8Array('be', n.byteLength()); + } + + async function nodeDecrypt$1(data, n, e, d, p, q, u, randomPayload) { + const { default: BN } = await Promise.resolve().then(function () { return bn$1; }); + + const pBNum = new BN(p); + const qBNum = new BN(q); + const dBNum = new BN(d); + const dq = dBNum.mod(qBNum.subn(1)); // d mod (q-1) + const dp = dBNum.mod(pBNum.subn(1)); // d mod (p-1) + const keyObject = { + version: 0, + modulus: new BN(n), + publicExponent: new BN(e), + privateExponent: new BN(d), + // switch p and q + prime1: new BN(q), + prime2: new BN(p), + // switch dp and dq + exponent1: dq, + exponent2: dp, + coefficient: new BN(u) + }; + let key; + if (typeof nodeCrypto$6.createPrivateKey !== 'undefined') { + const der = RSAPrivateKey.encode(keyObject, 'der'); + key = { key: der, format: 'der' , type: 'pkcs1', padding: nodeCrypto$6.constants.RSA_PKCS1_PADDING }; + } else { + const pem = RSAPrivateKey.encode(keyObject, 'pem', { + label: 'RSA PRIVATE KEY' + }); + key = { key: pem, padding: nodeCrypto$6.constants.RSA_PKCS1_PADDING }; + } + try { + return new Uint8Array(nodeCrypto$6.privateDecrypt(key, data)); + } catch (err) { + if (randomPayload) { + return randomPayload; + } + throw new Error('Decryption error'); + } + } + + async function bnDecrypt(data, n, e, d, p, q, u, randomPayload) { + const BigInteger = await util.getBigInteger(); + data = new BigInteger(data); + n = new BigInteger(n); + e = new BigInteger(e); + d = new BigInteger(d); + p = new BigInteger(p); + q = new BigInteger(q); + u = new BigInteger(u); + if (data.gte(n)) { + throw new Error('Data too large.'); + } + const dq = d.mod(q.dec()); // d mod (q-1) + const dp = d.mod(p.dec()); // d mod (p-1) + + const unblinder = (await getRandomBigInteger(new BigInteger(2), n)).mod(n); + const blinder = unblinder.modInv(n).modExp(e, n); + data = data.mul(blinder).mod(n); + + + const mp = data.modExp(dp, p); // data**{d mod (q-1)} mod p + const mq = data.modExp(dq, q); // data**{d mod (p-1)} mod q + const h = u.mul(mq.sub(mp)).mod(q); // u * (mq-mp) mod q (operands already < q) + + let result = h.mul(p).add(mp); // result < n due to relations above + + result = result.mul(unblinder).mod(n); + + + return emeDecode(result.toUint8Array('be', n.byteLength()), randomPayload); + } + + /** Convert Openpgp private key params to jwk key according to + * @link https://tools.ietf.org/html/rfc7517 + * @param {String} hashAlgo + * @param {Uint8Array} n + * @param {Uint8Array} e + * @param {Uint8Array} d + * @param {Uint8Array} p + * @param {Uint8Array} q + * @param {Uint8Array} u + */ + async function privateToJWK(n, e, d, p, q, u) { + const BigInteger = await util.getBigInteger(); + const pNum = new BigInteger(p); + const qNum = new BigInteger(q); + const dNum = new BigInteger(d); + + let dq = dNum.mod(qNum.dec()); // d mod (q-1) + let dp = dNum.mod(pNum.dec()); // d mod (p-1) + dp = dp.toUint8Array(); + dq = dq.toUint8Array(); + return { + kty: 'RSA', + n: uint8ArrayToB64(n, true), + e: uint8ArrayToB64(e, true), + d: uint8ArrayToB64(d, true), + // switch p and q + p: uint8ArrayToB64(q, true), + q: uint8ArrayToB64(p, true), + // switch dp and dq + dp: uint8ArrayToB64(dq, true), + dq: uint8ArrayToB64(dp, true), + qi: uint8ArrayToB64(u, true), + ext: true + }; + } + + /** Convert Openpgp key public params to jwk key according to + * @link https://tools.ietf.org/html/rfc7517 + * @param {String} hashAlgo + * @param {Uint8Array} n + * @param {Uint8Array} e + */ + function publicToJWK(n, e) { + return { + kty: 'RSA', + n: uint8ArrayToB64(n, true), + e: uint8ArrayToB64(e, true), + ext: true + }; + } + + var rsa = /*#__PURE__*/Object.freeze({ + __proto__: null, + sign: sign, + verify: verify, + encrypt: encrypt$1, + decrypt: decrypt$1, + generate: generate, + validateParams: validateParams + }); + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * ElGamal Encryption function + * Note that in OpenPGP, the message needs to be padded with PKCS#1 (same as RSA) + * @param {Uint8Array} data - To be padded and encrypted + * @param {Uint8Array} p + * @param {Uint8Array} g + * @param {Uint8Array} y + * @returns {Promise<{ c1: Uint8Array, c2: Uint8Array }>} + * @async + */ + async function encrypt$2(data, p, g, y) { + const BigInteger = await util.getBigInteger(); + p = new BigInteger(p); + g = new BigInteger(g); + y = new BigInteger(y); + + const padded = emeEncode(data, p.byteLength()); + const m = new BigInteger(padded); + + // OpenPGP uses a "special" version of ElGamal where g is generator of the full group Z/pZ* + // hence g has order p-1, and to avoid that k = 0 mod p-1, we need to pick k in [1, p-2] + const k = await getRandomBigInteger(new BigInteger(1), p.dec()); + return { + c1: g.modExp(k, p).toUint8Array(), + c2: y.modExp(k, p).imul(m).imod(p).toUint8Array() + }; + } + + /** + * ElGamal Encryption function + * @param {Uint8Array} c1 + * @param {Uint8Array} c2 + * @param {Uint8Array} p + * @param {Uint8Array} x + * @param {Uint8Array} randomPayload - Data to return on unpadding error, instead of throwing + * (needed for constant-time processing) + * @returns {Promise} Unpadded message. + * @throws {Error} on decryption error, unless `randomPayload` is given + * @async + */ + async function decrypt$2(c1, c2, p, x, randomPayload) { + const BigInteger = await util.getBigInteger(); + c1 = new BigInteger(c1); + c2 = new BigInteger(c2); + p = new BigInteger(p); + x = new BigInteger(x); + + const padded = c1.modExp(x, p).modInv(p).imul(c2).imod(p); + return emeDecode(padded.toUint8Array('be', p.byteLength()), randomPayload); + } + + /** + * Validate ElGamal parameters + * @param {Uint8Array} p - ElGamal prime + * @param {Uint8Array} g - ElGamal group generator + * @param {Uint8Array} y - ElGamal public key + * @param {Uint8Array} x - ElGamal private exponent + * @returns {Promise} Whether params are valid. + * @async + */ + async function validateParams$1(p, g, y, x) { + const BigInteger = await util.getBigInteger(); + p = new BigInteger(p); + g = new BigInteger(g); + y = new BigInteger(y); + + const one = new BigInteger(1); + // Check that 1 < g < p + if (g.lte(one) || g.gte(p)) { + return false; + } + + // Expect p-1 to be large + const pSize = new BigInteger(p.bitLength()); + const n1023 = new BigInteger(1023); + if (pSize.lt(n1023)) { + return false; + } + + /** + * g should have order p-1 + * Check that g ** (p-1) = 1 mod p + */ + if (!g.modExp(p.dec(), p).isOne()) { + return false; + } + + /** + * Since p-1 is not prime, g might have a smaller order that divides p-1 + * We want to make sure that the order is large enough to hinder a small subgroup attack + * + * We just check g**i != 1 for all i up to a threshold + */ + let res = g; + const i = new BigInteger(1); + const threshold = new BigInteger(2).leftShift(new BigInteger(17)); // we want order > threshold + while (i.lt(threshold)) { + res = res.mul(g).imod(p); + if (res.isOne()) { + return false; + } + i.iinc(); + } + + /** + * Re-derive public key y' = g ** x mod p + * Expect y == y' + * + * Blinded exponentiation computes g**{r(p-1) + x} to compare to y + */ + x = new BigInteger(x); + const two = new BigInteger(2); + const r = await getRandomBigInteger(two.leftShift(pSize.dec()), two.leftShift(pSize)); // draw r of same size as p-1 + const rqx = p.dec().imul(r).iadd(x); + if (!y.equal(g.modExp(rqx, p))) { + return false; + } + + return true; + } + + var elgamal = /*#__PURE__*/Object.freeze({ + __proto__: null, + encrypt: encrypt$2, + decrypt: decrypt$2, + validateParams: validateParams$1 + }); + + // OpenPGP.js - An OpenPGP implementation in javascript + + class OID { + constructor(oid) { + if (oid instanceof OID) { + this.oid = oid.oid; + } else if (util.isArray(oid) || + util.isUint8Array(oid)) { + oid = new Uint8Array(oid); + if (oid[0] === 0x06) { // DER encoded oid byte array + if (oid[1] !== oid.length - 2) { + throw new Error('Length mismatch in DER encoded oid'); + } + oid = oid.subarray(2); + } + this.oid = oid; + } else { + this.oid = ''; + } + } + + /** + * Method to read an OID object + * @param {Uint8Array} input - Where to read the OID from + * @returns {Number} Number of read bytes. + */ + read(input) { + if (input.length >= 1) { + const length = input[0]; + if (input.length >= 1 + length) { + this.oid = input.subarray(1, 1 + length); + return 1 + this.oid.length; + } + } + throw new Error('Invalid oid'); + } + + /** + * Serialize an OID object + * @returns {Uint8Array} Array with the serialized value the OID. + */ + write() { + return util.concatUint8Array([new Uint8Array([this.oid.length]), this.oid]); + } + + /** + * Serialize an OID object as a hex string + * @returns {string} String with the hex value of the OID. + */ + toHex() { + return util.uint8ArrayToHex(this.oid); + } + + /** + * If a known curve object identifier, return the canonical name of the curve + * @returns {string} String with the canonical name of the curve. + */ + getName() { + const hex = this.toHex(); + if (enums.curve[hex]) { + return enums.write(enums.curve, hex); + } else { + throw new Error('Unknown curve object identifier.'); + } + } + } + + // OpenPGP.js - An OpenPGP implementation in javascript + + function keyFromPrivate(indutnyCurve, priv) { + const keyPair = indutnyCurve.keyPair({ priv: priv }); + return keyPair; + } + + function keyFromPublic(indutnyCurve, pub) { + const keyPair = indutnyCurve.keyPair({ pub: pub }); + if (keyPair.validate().result !== true) { + throw new Error('Invalid elliptic public key'); + } + return keyPair; + } + + async function getIndutnyCurve(name) { + if (!config.useIndutnyElliptic) { + throw new Error('This curve is only supported in the full build of OpenPGP.js'); + } + const { default: elliptic } = await Promise.resolve().then(function () { return elliptic$1; }); + return new elliptic.ec(name); + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + function readSimpleLength(bytes) { + let len = 0; + let offset; + const type = bytes[0]; + + + if (type < 192) { + [len] = bytes; + offset = 1; + } else if (type < 255) { + len = ((bytes[0] - 192) << 8) + (bytes[1]) + 192; + offset = 2; + } else if (type === 255) { + len = util.readNumber(bytes.subarray(1, 1 + 4)); + offset = 5; + } + + return { + len: len, + offset: offset + }; + } + + /** + * Encodes a given integer of length to the openpgp length specifier to a + * string + * + * @param {Integer} length - The length to encode + * @returns {Uint8Array} String with openpgp length representation. + */ + function writeSimpleLength(length) { + if (length < 192) { + return new Uint8Array([length]); + } else if (length > 191 && length < 8384) { + /* + * let a = (total data packet length) - 192 let bc = two octet + * representation of a let d = b + 192 + */ + return new Uint8Array([((length - 192) >> 8) + 192, (length - 192) & 0xFF]); + } + return util.concatUint8Array([new Uint8Array([255]), util.writeNumber(length, 4)]); + } + + function writePartialLength(power) { + if (power < 0 || power > 30) { + throw new Error('Partial Length power must be between 1 and 30'); + } + return new Uint8Array([224 + power]); + } + + function writeTag(tag_type) { + /* we're only generating v4 packet headers here */ + return new Uint8Array([0xC0 | tag_type]); + } + + /** + * Writes a packet header version 4 with the given tag_type and length to a + * string + * + * @param {Integer} tag_type - Tag type + * @param {Integer} length - Length of the payload + * @returns {String} String of the header. + */ + function writeHeader(tag_type, length) { + /* we're only generating v4 packet headers here */ + return util.concatUint8Array([writeTag(tag_type), writeSimpleLength(length)]); + } + + /** + * Whether the packet type supports partial lengths per RFC4880 + * @param {Integer} tag - Tag type + * @returns {Boolean} String of the header. + */ + function supportsStreaming(tag) { + return [ + enums.packet.literalData, + enums.packet.compressedData, + enums.packet.symmetricallyEncryptedData, + enums.packet.symEncryptedIntegrityProtectedData, + enums.packet.aeadEncryptedData + ].includes(tag); + } + + /** + * Generic static Packet Parser function + * + * @param {Uint8Array | ReadableStream} input - Input stream as string + * @param {Function} callback - Function to call with the parsed packet + * @returns {Boolean} Returns false if the stream was empty and parsing is done, and true otherwise. + */ + async function readPackets(input, callback) { + const reader = getReader(input); + let writer; + let callbackReturned; + try { + const peekedBytes = await reader.peekBytes(2); + // some sanity checks + if (!peekedBytes || peekedBytes.length < 2 || (peekedBytes[0] & 0x80) === 0) { + throw new Error('Error during parsing. This message / key probably does not conform to a valid OpenPGP format.'); + } + const headerByte = await reader.readByte(); + let tag = -1; + let format = -1; + let packetLength; + + format = 0; // 0 = old format; 1 = new format + if ((headerByte & 0x40) !== 0) { + format = 1; + } + + let packetLengthType; + if (format) { + // new format header + tag = headerByte & 0x3F; // bit 5-0 + } else { + // old format header + tag = (headerByte & 0x3F) >> 2; // bit 5-2 + packetLengthType = headerByte & 0x03; // bit 1-0 + } + + const packetSupportsStreaming = supportsStreaming(tag); + let packet = null; + if (packetSupportsStreaming) { + if (util.isStream(input) === 'array') { + const arrayStream = new ArrayStream(); + writer = getWriter(arrayStream); + packet = arrayStream; + } else { + const transform = new TransformStream(); + writer = getWriter(transform.writable); + packet = transform.readable; + } + // eslint-disable-next-line callback-return + callbackReturned = callback({ tag, packet }); + } else { + packet = []; + } + + let wasPartialLength; + do { + if (!format) { + // 4.2.1. Old Format Packet Lengths + switch (packetLengthType) { + case 0: + // The packet has a one-octet length. The header is 2 octets + // long. + packetLength = await reader.readByte(); + break; + case 1: + // The packet has a two-octet length. The header is 3 octets + // long. + packetLength = (await reader.readByte() << 8) | await reader.readByte(); + break; + case 2: + // The packet has a four-octet length. The header is 5 + // octets long. + packetLength = (await reader.readByte() << 24) | (await reader.readByte() << 16) | (await reader.readByte() << + 8) | await reader.readByte(); + break; + default: + // 3 - The packet is of indeterminate length. The header is 1 + // octet long, and the implementation must determine how long + // the packet is. If the packet is in a file, this means that + // the packet extends until the end of the file. In general, + // an implementation SHOULD NOT use indeterminate-length + // packets except where the end of the data will be clear + // from the context, and even then it is better to use a + // definite length, or a new format header. The new format + // headers described below have a mechanism for precisely + // encoding data of indeterminate length. + packetLength = Infinity; + break; + } + } else { // 4.2.2. New Format Packet Lengths + // 4.2.2.1. One-Octet Lengths + const lengthByte = await reader.readByte(); + wasPartialLength = false; + if (lengthByte < 192) { + packetLength = lengthByte; + // 4.2.2.2. Two-Octet Lengths + } else if (lengthByte >= 192 && lengthByte < 224) { + packetLength = ((lengthByte - 192) << 8) + (await reader.readByte()) + 192; + // 4.2.2.4. Partial Body Lengths + } else if (lengthByte > 223 && lengthByte < 255) { + packetLength = 1 << (lengthByte & 0x1F); + wasPartialLength = true; + if (!packetSupportsStreaming) { + throw new TypeError('This packet type does not support partial lengths.'); + } + // 4.2.2.3. Five-Octet Lengths + } else { + packetLength = (await reader.readByte() << 24) | (await reader.readByte() << 16) | (await reader.readByte() << + 8) | await reader.readByte(); + } + } + if (packetLength > 0) { + let bytesRead = 0; + while (true) { + if (writer) await writer.ready; + const { done, value } = await reader.read(); + if (done) { + if (packetLength === Infinity) break; + throw new Error('Unexpected end of packet'); + } + const chunk = packetLength === Infinity ? value : value.subarray(0, packetLength - bytesRead); + if (writer) await writer.write(chunk); + else packet.push(chunk); + bytesRead += value.length; + if (bytesRead >= packetLength) { + reader.unshift(value.subarray(packetLength - bytesRead + value.length)); + break; + } + } + } + } while (wasPartialLength); + + // If this was not a packet that "supports streaming", we peek to check + // whether it is the last packet in the message. We peek 2 bytes instead + // of 1 because the beginning of this function also peeks 2 bytes, and we + // want to cut a `subarray` of the correct length into `web-stream-tools`' + // `externalBuffer` as a tiny optimization here. + // + // If it *was* a streaming packet (i.e. the data packets), we peek at the + // entire remainder of the stream, in order to forward errors in the + // remainder of the stream to the packet data. (Note that this means we + // read/peek at all signature packets before closing the literal data + // packet, for example.) This forwards MDC errors to the literal data + // stream, for example, so that they don't get lost / forgotten on + // decryptedMessage.packets.stream, which we never look at. + // + // An example of what we do when stream-parsing a message containing + // [ one-pass signature packet, literal data packet, signature packet ]: + // 1. Read the one-pass signature packet + // 2. Peek 2 bytes of the literal data packet + // 3. Parse the one-pass signature packet + // + // 4. Read the literal data packet, simultaneously stream-parsing it + // 5. Peek until the end of the message + // 6. Finish parsing the literal data packet + // + // 7. Read the signature packet again (we already peeked at it in step 5) + // 8. Peek at the end of the stream again (`peekBytes` returns undefined) + // 9. Parse the signature packet + // + // Note that this means that if there's an error in the very end of the + // stream, such as an MDC error, we throw in step 5 instead of in step 8 + // (or never), which is the point of this exercise. + const nextPacket = await reader.peekBytes(packetSupportsStreaming ? Infinity : 2); + if (writer) { + await writer.ready; + await writer.close(); + } else { + packet = util.concatUint8Array(packet); + // eslint-disable-next-line callback-return + await callback({ tag, packet }); + } + return !nextPacket || !nextPacket.length; + } catch (e) { + if (writer) { + await writer.abort(e); + return true; + } else { + throw e; + } + } finally { + if (writer) { + await callbackReturned; + } + reader.releaseLock(); + } + } + + class UnsupportedError extends Error { + constructor(...params) { + super(...params); + + if (Error.captureStackTrace) { + Error.captureStackTrace(this, UnsupportedError); + } + + this.name = 'UnsupportedError'; + } + } + + class UnparseablePacket { + constructor(tag, rawContent) { + this.tag = tag; + this.rawContent = rawContent; + } + + write() { + return this.rawContent; + } + } + + // OpenPGP.js - An OpenPGP implementation in javascript + + const webCrypto$6 = util.getWebCrypto(); + const nodeCrypto$7 = util.getNodeCrypto(); + + const webCurves = { + 'p256': 'P-256', + 'p384': 'P-384', + 'p521': 'P-521' + }; + const knownCurves = nodeCrypto$7 ? nodeCrypto$7.getCurves() : []; + const nodeCurves = nodeCrypto$7 ? { + secp256k1: knownCurves.includes('secp256k1') ? 'secp256k1' : undefined, + p256: knownCurves.includes('prime256v1') ? 'prime256v1' : undefined, + p384: knownCurves.includes('secp384r1') ? 'secp384r1' : undefined, + p521: knownCurves.includes('secp521r1') ? 'secp521r1' : undefined, + ed25519: knownCurves.includes('ED25519') ? 'ED25519' : undefined, + curve25519: knownCurves.includes('X25519') ? 'X25519' : undefined, + brainpoolP256r1: knownCurves.includes('brainpoolP256r1') ? 'brainpoolP256r1' : undefined, + brainpoolP384r1: knownCurves.includes('brainpoolP384r1') ? 'brainpoolP384r1' : undefined, + brainpoolP512r1: knownCurves.includes('brainpoolP512r1') ? 'brainpoolP512r1' : undefined + } : {}; + + const curves = { + p256: { + oid: [0x06, 0x08, 0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x07], + keyType: enums.publicKey.ecdsa, + hash: enums.hash.sha256, + cipher: enums.symmetric.aes128, + node: nodeCurves.p256, + web: webCurves.p256, + payloadSize: 32, + sharedSize: 256 + }, + p384: { + oid: [0x06, 0x05, 0x2B, 0x81, 0x04, 0x00, 0x22], + keyType: enums.publicKey.ecdsa, + hash: enums.hash.sha384, + cipher: enums.symmetric.aes192, + node: nodeCurves.p384, + web: webCurves.p384, + payloadSize: 48, + sharedSize: 384 + }, + p521: { + oid: [0x06, 0x05, 0x2B, 0x81, 0x04, 0x00, 0x23], + keyType: enums.publicKey.ecdsa, + hash: enums.hash.sha512, + cipher: enums.symmetric.aes256, + node: nodeCurves.p521, + web: webCurves.p521, + payloadSize: 66, + sharedSize: 528 + }, + secp256k1: { + oid: [0x06, 0x05, 0x2B, 0x81, 0x04, 0x00, 0x0A], + keyType: enums.publicKey.ecdsa, + hash: enums.hash.sha256, + cipher: enums.symmetric.aes128, + node: nodeCurves.secp256k1, + payloadSize: 32 + }, + ed25519: { + oid: [0x06, 0x09, 0x2B, 0x06, 0x01, 0x04, 0x01, 0xDA, 0x47, 0x0F, 0x01], + keyType: enums.publicKey.eddsa, + hash: enums.hash.sha512, + node: false, // nodeCurves.ed25519 TODO + payloadSize: 32 + }, + curve25519: { + oid: [0x06, 0x0A, 0x2B, 0x06, 0x01, 0x04, 0x01, 0x97, 0x55, 0x01, 0x05, 0x01], + keyType: enums.publicKey.ecdh, + hash: enums.hash.sha256, + cipher: enums.symmetric.aes128, + node: false, // nodeCurves.curve25519 TODO + payloadSize: 32 + }, + brainpoolP256r1: { + oid: [0x06, 0x09, 0x2B, 0x24, 0x03, 0x03, 0x02, 0x08, 0x01, 0x01, 0x07], + keyType: enums.publicKey.ecdsa, + hash: enums.hash.sha256, + cipher: enums.symmetric.aes128, + node: nodeCurves.brainpoolP256r1, + payloadSize: 32 + }, + brainpoolP384r1: { + oid: [0x06, 0x09, 0x2B, 0x24, 0x03, 0x03, 0x02, 0x08, 0x01, 0x01, 0x0B], + keyType: enums.publicKey.ecdsa, + hash: enums.hash.sha384, + cipher: enums.symmetric.aes192, + node: nodeCurves.brainpoolP384r1, + payloadSize: 48 + }, + brainpoolP512r1: { + oid: [0x06, 0x09, 0x2B, 0x24, 0x03, 0x03, 0x02, 0x08, 0x01, 0x01, 0x0D], + keyType: enums.publicKey.ecdsa, + hash: enums.hash.sha512, + cipher: enums.symmetric.aes256, + node: nodeCurves.brainpoolP512r1, + payloadSize: 64 + } + }; + + class Curve { + constructor(oidOrName, params) { + try { + if (util.isArray(oidOrName) || + util.isUint8Array(oidOrName)) { + // by oid byte array + oidOrName = new OID(oidOrName); + } + if (oidOrName instanceof OID) { + // by curve OID + oidOrName = oidOrName.getName(); + } + // by curve name or oid string + this.name = enums.write(enums.curve, oidOrName); + } catch (err) { + throw new UnsupportedError('Unknown curve'); + } + params = params || curves[this.name]; + + this.keyType = params.keyType; + + this.oid = params.oid; + this.hash = params.hash; + this.cipher = params.cipher; + this.node = params.node && curves[this.name]; + this.web = params.web && curves[this.name]; + this.payloadSize = params.payloadSize; + if (this.web && util.getWebCrypto()) { + this.type = 'web'; + } else if (this.node && util.getNodeCrypto()) { + this.type = 'node'; + } else if (this.name === 'curve25519') { + this.type = 'curve25519'; + } else if (this.name === 'ed25519') { + this.type = 'ed25519'; + } + } + + async genKeyPair() { + let keyPair; + switch (this.type) { + case 'web': + try { + return await webGenKeyPair(this.name); + } catch (err) { + util.printDebugError('Browser did not support generating ec key ' + err.message); + break; + } + case 'node': + return nodeGenKeyPair(this.name); + case 'curve25519': { + const privateKey = getRandomBytes(32); + privateKey[0] = (privateKey[0] & 127) | 64; + privateKey[31] &= 248; + const secretKey = privateKey.slice().reverse(); + keyPair = naclFastLight.box.keyPair.fromSecretKey(secretKey); + const publicKey = util.concatUint8Array([new Uint8Array([0x40]), keyPair.publicKey]); + return { publicKey, privateKey }; + } + case 'ed25519': { + const privateKey = getRandomBytes(32); + const keyPair = naclFastLight.sign.keyPair.fromSeed(privateKey); + const publicKey = util.concatUint8Array([new Uint8Array([0x40]), keyPair.publicKey]); + return { publicKey, privateKey }; + } + } + const indutnyCurve = await getIndutnyCurve(this.name); + keyPair = await indutnyCurve.genKeyPair({ + entropy: util.uint8ArrayToString(getRandomBytes(32)) + }); + return { publicKey: new Uint8Array(keyPair.getPublic('array', false)), privateKey: keyPair.getPrivate().toArrayLike(Uint8Array) }; + } + } + + async function generate$1(curve) { + const BigInteger = await util.getBigInteger(); + + curve = new Curve(curve); + const keyPair = await curve.genKeyPair(); + const Q = new BigInteger(keyPair.publicKey).toUint8Array(); + const secret = new BigInteger(keyPair.privateKey).toUint8Array('be', curve.payloadSize); + return { + oid: curve.oid, + Q, + secret, + hash: curve.hash, + cipher: curve.cipher + }; + } + + /** + * Get preferred hash algo to use with the given curve + * @param {module:type/oid} oid - curve oid + * @returns {enums.hash} hash algorithm + */ + function getPreferredHashAlgo(oid) { + return curves[enums.write(enums.curve, oid.toHex())].hash; + } + + /** + * Validate ECDH and ECDSA parameters + * Not suitable for EdDSA (different secret key format) + * @param {module:enums.publicKey} algo - EC algorithm, to filter supported curves + * @param {module:type/oid} oid - EC object identifier + * @param {Uint8Array} Q - EC public point + * @param {Uint8Array} d - EC secret scalar + * @returns {Promise} Whether params are valid. + * @async + */ + async function validateStandardParams(algo, oid, Q, d) { + const supportedCurves = { + p256: true, + p384: true, + p521: true, + secp256k1: true, + curve25519: algo === enums.publicKey.ecdh, + brainpoolP256r1: true, + brainpoolP384r1: true, + brainpoolP512r1: true + }; + + // Check whether the given curve is supported + const curveName = oid.getName(); + if (!supportedCurves[curveName]) { + return false; + } + + if (curveName === 'curve25519') { + d = d.slice().reverse(); + // Re-derive public point Q' + const { publicKey } = naclFastLight.box.keyPair.fromSecretKey(d); + + Q = new Uint8Array(Q); + const dG = new Uint8Array([0x40, ...publicKey]); // Add public key prefix + if (!util.equalsUint8Array(dG, Q)) { + return false; + } + + return true; + } + + const curve = await getIndutnyCurve(curveName); + try { + // Parse Q and check that it is on the curve but not at infinity + Q = keyFromPublic(curve, Q).getPublic(); + } catch (validationErrors) { + return false; + } + + /** + * Re-derive public point Q' = dG from private key + * Expect Q == Q' + */ + const dG = keyFromPrivate(curve, d).getPublic(); + if (!dG.eq(Q)) { + return false; + } + + return true; + } + + ////////////////////////// + // // + // Helper functions // + // // + ////////////////////////// + + + async function webGenKeyPair(name) { + // Note: keys generated with ECDSA and ECDH are structurally equivalent + const webCryptoKey = await webCrypto$6.generateKey({ name: 'ECDSA', namedCurve: webCurves[name] }, true, ['sign', 'verify']); + + const privateKey = await webCrypto$6.exportKey('jwk', webCryptoKey.privateKey); + const publicKey = await webCrypto$6.exportKey('jwk', webCryptoKey.publicKey); + + return { + publicKey: jwkToRawPublic(publicKey), + privateKey: b64ToUint8Array(privateKey.d) + }; + } + + async function nodeGenKeyPair(name) { + // Note: ECDSA and ECDH key generation is structurally equivalent + const ecdh = nodeCrypto$7.createECDH(nodeCurves[name]); + await ecdh.generateKeys(); + return { + publicKey: new Uint8Array(ecdh.getPublicKey()), + privateKey: new Uint8Array(ecdh.getPrivateKey()) + }; + } + + ////////////////////////// + // // + // Helper functions // + // // + ////////////////////////// + + /** + * @param {JsonWebKey} jwk - key for conversion + * + * @returns {Uint8Array} Raw public key. + */ + function jwkToRawPublic(jwk) { + const bufX = b64ToUint8Array(jwk.x); + const bufY = b64ToUint8Array(jwk.y); + const publicKey = new Uint8Array(bufX.length + bufY.length + 1); + publicKey[0] = 0x04; + publicKey.set(bufX, 1); + publicKey.set(bufY, bufX.length + 1); + return publicKey; + } + + /** + * @param {Integer} payloadSize - ec payload size + * @param {String} name - curve name + * @param {Uint8Array} publicKey - public key + * + * @returns {JsonWebKey} Public key in jwk format. + */ + function rawPublicToJWK(payloadSize, name, publicKey) { + const len = payloadSize; + const bufX = publicKey.slice(1, len + 1); + const bufY = publicKey.slice(len + 1, len * 2 + 1); + // https://www.rfc-editor.org/rfc/rfc7518.txt + const jwk = { + kty: 'EC', + crv: name, + x: uint8ArrayToB64(bufX, true), + y: uint8ArrayToB64(bufY, true), + ext: true + }; + return jwk; + } + + /** + * @param {Integer} payloadSize - ec payload size + * @param {String} name - curve name + * @param {Uint8Array} publicKey - public key + * @param {Uint8Array} privateKey - private key + * + * @returns {JsonWebKey} Private key in jwk format. + */ + function privateToJWK$1(payloadSize, name, publicKey, privateKey) { + const jwk = rawPublicToJWK(payloadSize, name, publicKey); + jwk.d = uint8ArrayToB64(privateKey, true); + return jwk; + } + + // OpenPGP.js - An OpenPGP implementation in javascript + + const webCrypto$7 = util.getWebCrypto(); + const nodeCrypto$8 = util.getNodeCrypto(); + + /** + * Sign a message using the provided key + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {module:enums.hash} hashAlgo - Hash algorithm used to sign + * @param {Uint8Array} message - Message to sign + * @param {Uint8Array} publicKey - Public key + * @param {Uint8Array} privateKey - Private key used to sign the message + * @param {Uint8Array} hashed - The hashed message + * @returns {Promise<{ + * r: Uint8Array, + * s: Uint8Array + * }>} Signature of the message + * @async + */ + async function sign$1(oid, hashAlgo, message, publicKey, privateKey, hashed) { + const curve = new Curve(oid); + if (message && !util.isStream(message)) { + const keyPair = { publicKey, privateKey }; + switch (curve.type) { + case 'web': { + // If browser doesn't support a curve, we'll catch it + try { + // Need to await to make sure browser succeeds + return await webSign$1(curve, hashAlgo, message, keyPair); + } catch (err) { + // We do not fallback if the error is related to key integrity + // Unfortunaley Safari does not support p521 and throws a DataError when using it + // So we need to always fallback for that curve + if (curve.name !== 'p521' && (err.name === 'DataError' || err.name === 'OperationError')) { + throw err; + } + util.printDebugError('Browser did not support signing: ' + err.message); + } + break; + } + case 'node': { + const signature = await nodeSign$1(curve, hashAlgo, message, keyPair); + return { + r: signature.r.toArrayLike(Uint8Array), + s: signature.s.toArrayLike(Uint8Array) + }; + } + } + } + return ellipticSign(curve, hashed, privateKey); + } + + /** + * Verifies if a signature is valid for a message + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {module:enums.hash} hashAlgo - Hash algorithm used in the signature + * @param {{r: Uint8Array, + s: Uint8Array}} signature Signature to verify + * @param {Uint8Array} message - Message to verify + * @param {Uint8Array} publicKey - Public key used to verify the message + * @param {Uint8Array} hashed - The hashed message + * @returns {Boolean} + * @async + */ + async function verify$1(oid, hashAlgo, signature, message, publicKey, hashed) { + const curve = new Curve(oid); + if (message && !util.isStream(message)) { + switch (curve.type) { + case 'web': + try { + // Need to await to make sure browser succeeds + return await webVerify$1(curve, hashAlgo, signature, message, publicKey); + } catch (err) { + // We do not fallback if the error is related to key integrity + // Unfortunately Safari does not support p521 and throws a DataError when using it + // So we need to always fallback for that curve + if (curve.name !== 'p521' && (err.name === 'DataError' || err.name === 'OperationError')) { + throw err; + } + util.printDebugError('Browser did not support verifying: ' + err.message); + } + break; + case 'node': + return nodeVerify$1(curve, hashAlgo, signature, message, publicKey); + } + } + const digest = (typeof hashAlgo === 'undefined') ? message : hashed; + return ellipticVerify(curve, signature, digest, publicKey); + } + + /** + * Validate ECDSA parameters + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {Uint8Array} Q - ECDSA public point + * @param {Uint8Array} d - ECDSA secret scalar + * @returns {Promise} Whether params are valid. + * @async + */ + async function validateParams$2(oid, Q, d) { + const curve = new Curve(oid); + // Reject curves x25519 and ed25519 + if (curve.keyType !== enums.publicKey.ecdsa) { + return false; + } + + // To speed up the validation, we try to use node- or webcrypto when available + // and sign + verify a random message + switch (curve.type) { + case 'web': + case 'node': { + const message = getRandomBytes(8); + const hashAlgo = enums.hash.sha256; + const hashed = await hash.digest(hashAlgo, message); + try { + const signature = await sign$1(oid, hashAlgo, message, Q, d, hashed); + return await verify$1(oid, hashAlgo, signature, message, Q, hashed); + } catch (err) { + return false; + } + } + default: + return validateStandardParams(enums.publicKey.ecdsa, oid, Q, d); + } + } + + + ////////////////////////// + // // + // Helper functions // + // // + ////////////////////////// + + async function ellipticSign(curve, hashed, privateKey) { + const indutnyCurve = await getIndutnyCurve(curve.name); + const key = keyFromPrivate(indutnyCurve, privateKey); + const signature = key.sign(hashed); + return { + r: signature.r.toArrayLike(Uint8Array), + s: signature.s.toArrayLike(Uint8Array) + }; + } + + async function ellipticVerify(curve, signature, digest, publicKey) { + const indutnyCurve = await getIndutnyCurve(curve.name); + const key = keyFromPublic(indutnyCurve, publicKey); + return key.verify(digest, signature); + } + + async function webSign$1(curve, hashAlgo, message, keyPair) { + const len = curve.payloadSize; + const jwk = privateToJWK$1(curve.payloadSize, webCurves[curve.name], keyPair.publicKey, keyPair.privateKey); + const key = await webCrypto$7.importKey( + 'jwk', + jwk, + { + 'name': 'ECDSA', + 'namedCurve': webCurves[curve.name], + 'hash': { name: enums.read(enums.webHash, curve.hash) } + }, + false, + ['sign'] + ); + + const signature = new Uint8Array(await webCrypto$7.sign( + { + 'name': 'ECDSA', + 'namedCurve': webCurves[curve.name], + 'hash': { name: enums.read(enums.webHash, hashAlgo) } + }, + key, + message + )); + + return { + r: signature.slice(0, len), + s: signature.slice(len, len << 1) + }; + } + + async function webVerify$1(curve, hashAlgo, { r, s }, message, publicKey) { + const jwk = rawPublicToJWK(curve.payloadSize, webCurves[curve.name], publicKey); + const key = await webCrypto$7.importKey( + 'jwk', + jwk, + { + 'name': 'ECDSA', + 'namedCurve': webCurves[curve.name], + 'hash': { name: enums.read(enums.webHash, curve.hash) } + }, + false, + ['verify'] + ); + + const signature = util.concatUint8Array([r, s]).buffer; + + return webCrypto$7.verify( + { + 'name': 'ECDSA', + 'namedCurve': webCurves[curve.name], + 'hash': { name: enums.read(enums.webHash, hashAlgo) } + }, + key, + signature, + message + ); + } + + async function nodeSign$1(curve, hashAlgo, message, keyPair) { + const sign = nodeCrypto$8.createSign(enums.read(enums.hash, hashAlgo)); + sign.write(message); + sign.end(); + const key = ECPrivateKey.encode({ + version: 1, + parameters: curve.oid, + privateKey: Array.from(keyPair.privateKey), + publicKey: { unused: 0, data: Array.from(keyPair.publicKey) } + }, 'pem', { + label: 'EC PRIVATE KEY' + }); + + return ECDSASignature.decode(sign.sign(key), 'der'); + } + + async function nodeVerify$1(curve, hashAlgo, { r, s }, message, publicKey) { + const { default: BN } = await Promise.resolve().then(function () { return bn$1; }); + + const verify = nodeCrypto$8.createVerify(enums.read(enums.hash, hashAlgo)); + verify.write(message); + verify.end(); + const key = SubjectPublicKeyInfo.encode({ + algorithm: { + algorithm: [1, 2, 840, 10045, 2, 1], + parameters: curve.oid + }, + subjectPublicKey: { unused: 0, data: Array.from(publicKey) } + }, 'pem', { + label: 'PUBLIC KEY' + }); + const signature = ECDSASignature.encode({ + r: new BN(r), s: new BN(s) + }, 'der'); + + try { + return verify.verify(key, signature); + } catch (err) { + return false; + } + } + + // Originally written by Owen Smith https://github.com/omsmith + // Adapted on Feb 2018 from https://github.com/Brightspace/node-jwk-to-pem/ + + /* eslint-disable no-invalid-this */ + + const asn1$1 = nodeCrypto$8 ? void('asn1.js') : undefined; + + const ECDSASignature = nodeCrypto$8 ? + asn1$1.define('ECDSASignature', function() { + this.seq().obj( + this.key('r').int(), + this.key('s').int() + ); + }) : undefined; + + const ECPrivateKey = nodeCrypto$8 ? + asn1$1.define('ECPrivateKey', function() { + this.seq().obj( + this.key('version').int(), + this.key('privateKey').octstr(), + this.key('parameters').explicit(0).optional().any(), + this.key('publicKey').explicit(1).optional().bitstr() + ); + }) : undefined; + + const AlgorithmIdentifier = nodeCrypto$8 ? + asn1$1.define('AlgorithmIdentifier', function() { + this.seq().obj( + this.key('algorithm').objid(), + this.key('parameters').optional().any() + ); + }) : undefined; + + const SubjectPublicKeyInfo = nodeCrypto$8 ? + asn1$1.define('SubjectPublicKeyInfo', function() { + this.seq().obj( + this.key('algorithm').use(AlgorithmIdentifier), + this.key('subjectPublicKey').bitstr() + ); + }) : undefined; + + var ecdsa = /*#__PURE__*/Object.freeze({ + __proto__: null, + sign: sign$1, + verify: verify$1, + validateParams: validateParams$2 + }); + + // OpenPGP.js - An OpenPGP implementation in javascript + + naclFastLight.hash = bytes => new Uint8Array(_512().update(bytes).digest()); + + /** + * Sign a message using the provided key + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {module:enums.hash} hashAlgo - Hash algorithm used to sign (must be sha256 or stronger) + * @param {Uint8Array} message - Message to sign + * @param {Uint8Array} publicKey - Public key + * @param {Uint8Array} privateKey - Private key used to sign the message + * @param {Uint8Array} hashed - The hashed message + * @returns {Promise<{ + * r: Uint8Array, + * s: Uint8Array + * }>} Signature of the message + * @async + */ + async function sign$2(oid, hashAlgo, message, publicKey, privateKey, hashed) { + if (hash.getHashByteLength(hashAlgo) < hash.getHashByteLength(enums.hash.sha256)) { + // see https://tools.ietf.org/id/draft-ietf-openpgp-rfc4880bis-10.html#section-15-7.2 + throw new Error('Hash algorithm too weak: sha256 or stronger is required for EdDSA.'); + } + const secretKey = util.concatUint8Array([privateKey, publicKey.subarray(1)]); + const signature = naclFastLight.sign.detached(hashed, secretKey); + // EdDSA signature params are returned in little-endian format + return { + r: signature.subarray(0, 32), + s: signature.subarray(32) + }; + } + + /** + * Verifies if a signature is valid for a message + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {module:enums.hash} hashAlgo - Hash algorithm used in the signature + * @param {{r: Uint8Array, + s: Uint8Array}} signature Signature to verify the message + * @param {Uint8Array} m - Message to verify + * @param {Uint8Array} publicKey - Public key used to verify the message + * @param {Uint8Array} hashed - The hashed message + * @returns {Boolean} + * @async + */ + async function verify$2(oid, hashAlgo, { r, s }, m, publicKey, hashed) { + const signature = util.concatUint8Array([r, s]); + return naclFastLight.sign.detached.verify(hashed, signature, publicKey.subarray(1)); + } + /** + * Validate EdDSA parameters + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {Uint8Array} Q - EdDSA public point + * @param {Uint8Array} k - EdDSA secret seed + * @returns {Promise} Whether params are valid. + * @async + */ + async function validateParams$3(oid, Q, k) { + // Check whether the given curve is supported + if (oid.getName() !== 'ed25519') { + return false; + } + + /** + * Derive public point Q' = dG from private key + * and expect Q == Q' + */ + const { publicKey } = naclFastLight.sign.keyPair.fromSeed(k); + const dG = new Uint8Array([0x40, ...publicKey]); // Add public key prefix + return util.equalsUint8Array(Q, dG); + } + + var eddsa = /*#__PURE__*/Object.freeze({ + __proto__: null, + sign: sign$2, + verify: verify$2, + validateParams: validateParams$3 + }); + + // OpenPGP.js - An OpenPGP implementation in javascript + + /** + * AES key wrap + * @function + * @param {Uint8Array} key + * @param {Uint8Array} data + * @returns {Uint8Array} + */ + function wrap(key, data) { + const aes = new cipher['aes' + (key.length * 8)](key); + const IV = new Uint32Array([0xA6A6A6A6, 0xA6A6A6A6]); + const P = unpack(data); + let A = IV; + const R = P; + const n = P.length / 2; + const t = new Uint32Array([0, 0]); + let B = new Uint32Array(4); + for (let j = 0; j <= 5; ++j) { + for (let i = 0; i < n; ++i) { + t[1] = n * j + (1 + i); + // B = A + B[0] = A[0]; + B[1] = A[1]; + // B = A || R[i] + B[2] = R[2 * i]; + B[3] = R[2 * i + 1]; + // B = AES(K, B) + B = unpack(aes.encrypt(pack(B))); + // A = MSB(64, B) ^ t + A = B.subarray(0, 2); + A[0] ^= t[0]; + A[1] ^= t[1]; + // R[i] = LSB(64, B) + R[2 * i] = B[2]; + R[2 * i + 1] = B[3]; + } + } + return pack(A, R); + } + + /** + * AES key unwrap + * @function + * @param {String} key + * @param {String} data + * @returns {Uint8Array} + * @throws {Error} + */ + function unwrap(key, data) { + const aes = new cipher['aes' + (key.length * 8)](key); + const IV = new Uint32Array([0xA6A6A6A6, 0xA6A6A6A6]); + const C = unpack(data); + let A = C.subarray(0, 2); + const R = C.subarray(2); + const n = C.length / 2 - 1; + const t = new Uint32Array([0, 0]); + let B = new Uint32Array(4); + for (let j = 5; j >= 0; --j) { + for (let i = n - 1; i >= 0; --i) { + t[1] = n * j + (i + 1); + // B = A ^ t + B[0] = A[0] ^ t[0]; + B[1] = A[1] ^ t[1]; + // B = (A ^ t) || R[i] + B[2] = R[2 * i]; + B[3] = R[2 * i + 1]; + // B = AES-1(B) + B = unpack(aes.decrypt(pack(B))); + // A = MSB(64, B) + A = B.subarray(0, 2); + // R[i] = LSB(64, B) + R[2 * i] = B[2]; + R[2 * i + 1] = B[3]; + } + } + if (A[0] === IV[0] && A[1] === IV[1]) { + return pack(R); + } + throw new Error('Key Data Integrity failed'); + } + + function createArrayBuffer(data) { + if (util.isString(data)) { + const { length } = data; + const buffer = new ArrayBuffer(length); + const view = new Uint8Array(buffer); + for (let j = 0; j < length; ++j) { + view[j] = data.charCodeAt(j); + } + return buffer; + } + return new Uint8Array(data).buffer; + } + + function unpack(data) { + const { length } = data; + const buffer = createArrayBuffer(data); + const view = new DataView(buffer); + const arr = new Uint32Array(length / 4); + for (let i = 0; i < length / 4; ++i) { + arr[i] = view.getUint32(4 * i); + } + return arr; + } + + function pack() { + let length = 0; + for (let k = 0; k < arguments.length; ++k) { + length += 4 * arguments[k].length; + } + const buffer = new ArrayBuffer(length); + const view = new DataView(buffer); + let offset = 0; + for (let i = 0; i < arguments.length; ++i) { + for (let j = 0; j < arguments[i].length; ++j) { + view.setUint32(offset + 4 * j, arguments[i][j]); + } + offset += 4 * arguments[i].length; + } + return new Uint8Array(buffer); + } + + var aesKW = /*#__PURE__*/Object.freeze({ + __proto__: null, + wrap: wrap, + unwrap: unwrap + }); + + // OpenPGP.js - An OpenPGP implementation in javascript + + /** + * @fileoverview Functions to add and remove PKCS5 padding + * @see PublicKeyEncryptedSessionKeyPacket + * @module crypto/pkcs5 + * @private + */ + + /** + * Add pkcs5 padding to a message + * @param {Uint8Array} message - message to pad + * @returns {Uint8Array} Padded message. + */ + function encode$1(message) { + const c = 8 - (message.length % 8); + const padded = new Uint8Array(message.length + c).fill(c); + padded.set(message); + return padded; + } + + /** + * Remove pkcs5 padding from a message + * @param {Uint8Array} message - message to remove padding from + * @returns {Uint8Array} Message without padding. + */ + function decode$1(message) { + const len = message.length; + if (len > 0) { + const c = message[len - 1]; + if (c >= 1) { + const provided = message.subarray(len - c); + const computed = new Uint8Array(c).fill(c); + if (util.equalsUint8Array(provided, computed)) { + return message.subarray(0, len - c); + } + } + } + throw new Error('Invalid padding'); + } + + var pkcs5 = /*#__PURE__*/Object.freeze({ + __proto__: null, + encode: encode$1, + decode: decode$1 + }); + + // OpenPGP.js - An OpenPGP implementation in javascript + + const webCrypto$8 = util.getWebCrypto(); + const nodeCrypto$9 = util.getNodeCrypto(); + + /** + * Validate ECDH parameters + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {Uint8Array} Q - ECDH public point + * @param {Uint8Array} d - ECDH secret scalar + * @returns {Promise} Whether params are valid. + * @async + */ + async function validateParams$4(oid, Q, d) { + return validateStandardParams(enums.publicKey.ecdh, oid, Q, d); + } + + // Build Param for ECDH algorithm (RFC 6637) + function buildEcdhParam(public_algo, oid, kdfParams, fingerprint) { + return util.concatUint8Array([ + oid.write(), + new Uint8Array([public_algo]), + kdfParams.write(), + util.stringToUint8Array('Anonymous Sender '), + fingerprint.subarray(0, 20) + ]); + } + + // Key Derivation Function (RFC 6637) + async function kdf(hashAlgo, X, length, param, stripLeading = false, stripTrailing = false) { + // Note: X is little endian for Curve25519, big-endian for all others. + // This is not ideal, but the RFC's are unclear + // https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-02#appendix-B + let i; + if (stripLeading) { + // Work around old go crypto bug + for (i = 0; i < X.length && X[i] === 0; i++); + X = X.subarray(i); + } + if (stripTrailing) { + // Work around old OpenPGP.js bug + for (i = X.length - 1; i >= 0 && X[i] === 0; i--); + X = X.subarray(0, i + 1); + } + const digest = await hash.digest(hashAlgo, util.concatUint8Array([ + new Uint8Array([0, 0, 0, 1]), + X, + param + ])); + return digest.subarray(0, length); + } + + /** + * Generate ECDHE ephemeral key and secret from public key + * + * @param {Curve} curve - Elliptic curve object + * @param {Uint8Array} Q - Recipient public key + * @returns {Promise<{publicKey: Uint8Array, sharedKey: Uint8Array}>} + * @async + */ + async function genPublicEphemeralKey(curve, Q) { + switch (curve.type) { + case 'curve25519': { + const d = getRandomBytes(32); + const { secretKey, sharedKey } = await genPrivateEphemeralKey(curve, Q, null, d); + let { publicKey } = naclFastLight.box.keyPair.fromSecretKey(secretKey); + publicKey = util.concatUint8Array([new Uint8Array([0x40]), publicKey]); + return { publicKey, sharedKey }; // Note: sharedKey is little-endian here, unlike below + } + case 'web': + if (curve.web && util.getWebCrypto()) { + try { + return await webPublicEphemeralKey(curve, Q); + } catch (err) { + util.printDebugError(err); + } + } + break; + case 'node': + return nodePublicEphemeralKey(curve, Q); + } + return ellipticPublicEphemeralKey(curve, Q); + } + + /** + * Encrypt and wrap a session key + * + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {module:type/kdf_params} kdfParams - KDF params including cipher and algorithm to use + * @param {Uint8Array} data - Unpadded session key data + * @param {Uint8Array} Q - Recipient public key + * @param {Uint8Array} fingerprint - Recipient fingerprint + * @returns {Promise<{publicKey: Uint8Array, wrappedKey: Uint8Array}>} + * @async + */ + async function encrypt$3(oid, kdfParams, data, Q, fingerprint) { + const m = encode$1(data); + + const curve = new Curve(oid); + const { publicKey, sharedKey } = await genPublicEphemeralKey(curve, Q); + const param = buildEcdhParam(enums.publicKey.ecdh, oid, kdfParams, fingerprint); + const { keySize } = getCipher(kdfParams.cipher); + const Z = await kdf(kdfParams.hash, sharedKey, keySize, param); + const wrappedKey = wrap(Z, m); + return { publicKey, wrappedKey }; + } + + /** + * Generate ECDHE secret from private key and public part of ephemeral key + * + * @param {Curve} curve - Elliptic curve object + * @param {Uint8Array} V - Public part of ephemeral key + * @param {Uint8Array} Q - Recipient public key + * @param {Uint8Array} d - Recipient private key + * @returns {Promise<{secretKey: Uint8Array, sharedKey: Uint8Array}>} + * @async + */ + async function genPrivateEphemeralKey(curve, V, Q, d) { + if (d.length !== curve.payloadSize) { + const privateKey = new Uint8Array(curve.payloadSize); + privateKey.set(d, curve.payloadSize - d.length); + d = privateKey; + } + switch (curve.type) { + case 'curve25519': { + const secretKey = d.slice().reverse(); + const sharedKey = naclFastLight.scalarMult(secretKey, V.subarray(1)); + return { secretKey, sharedKey }; // Note: sharedKey is little-endian here, unlike below + } + case 'web': + if (curve.web && util.getWebCrypto()) { + try { + return await webPrivateEphemeralKey(curve, V, Q, d); + } catch (err) { + util.printDebugError(err); + } + } + break; + case 'node': + return nodePrivateEphemeralKey(curve, V, d); + } + return ellipticPrivateEphemeralKey(curve, V, d); + } + + /** + * Decrypt and unwrap the value derived from session key + * + * @param {module:type/oid} oid - Elliptic curve object identifier + * @param {module:type/kdf_params} kdfParams - KDF params including cipher and algorithm to use + * @param {Uint8Array} V - Public part of ephemeral key + * @param {Uint8Array} C - Encrypted and wrapped value derived from session key + * @param {Uint8Array} Q - Recipient public key + * @param {Uint8Array} d - Recipient private key + * @param {Uint8Array} fingerprint - Recipient fingerprint + * @returns {Promise} Value derived from session key. + * @async + */ + async function decrypt$3(oid, kdfParams, V, C, Q, d, fingerprint) { + const curve = new Curve(oid); + const { sharedKey } = await genPrivateEphemeralKey(curve, V, Q, d); + const param = buildEcdhParam(enums.publicKey.ecdh, oid, kdfParams, fingerprint); + const { keySize } = getCipher(kdfParams.cipher); + let err; + for (let i = 0; i < 3; i++) { + try { + // Work around old go crypto bug and old OpenPGP.js bug, respectively. + const Z = await kdf(kdfParams.hash, sharedKey, keySize, param, i === 1, i === 2); + return decode$1(unwrap(Z, C)); + } catch (e) { + err = e; + } + } + throw err; + } + + /** + * Generate ECDHE secret from private key and public part of ephemeral key using webCrypto + * + * @param {Curve} curve - Elliptic curve object + * @param {Uint8Array} V - Public part of ephemeral key + * @param {Uint8Array} Q - Recipient public key + * @param {Uint8Array} d - Recipient private key + * @returns {Promise<{secretKey: Uint8Array, sharedKey: Uint8Array}>} + * @async + */ + async function webPrivateEphemeralKey(curve, V, Q, d) { + const recipient = privateToJWK$1(curve.payloadSize, curve.web.web, Q, d); + let privateKey = webCrypto$8.importKey( + 'jwk', + recipient, + { + name: 'ECDH', + namedCurve: curve.web.web + }, + true, + ['deriveKey', 'deriveBits'] + ); + const jwk = rawPublicToJWK(curve.payloadSize, curve.web.web, V); + let sender = webCrypto$8.importKey( + 'jwk', + jwk, + { + name: 'ECDH', + namedCurve: curve.web.web + }, + true, + [] + ); + [privateKey, sender] = await Promise.all([privateKey, sender]); + let S = webCrypto$8.deriveBits( + { + name: 'ECDH', + namedCurve: curve.web.web, + public: sender + }, + privateKey, + curve.web.sharedSize + ); + let secret = webCrypto$8.exportKey( + 'jwk', + privateKey + ); + [S, secret] = await Promise.all([S, secret]); + const sharedKey = new Uint8Array(S); + const secretKey = b64ToUint8Array(secret.d); + return { secretKey, sharedKey }; + } + + /** + * Generate ECDHE ephemeral key and secret from public key using webCrypto + * + * @param {Curve} curve - Elliptic curve object + * @param {Uint8Array} Q - Recipient public key + * @returns {Promise<{publicKey: Uint8Array, sharedKey: Uint8Array}>} + * @async + */ + async function webPublicEphemeralKey(curve, Q) { + const jwk = rawPublicToJWK(curve.payloadSize, curve.web.web, Q); + let keyPair = webCrypto$8.generateKey( + { + name: 'ECDH', + namedCurve: curve.web.web + }, + true, + ['deriveKey', 'deriveBits'] + ); + let recipient = webCrypto$8.importKey( + 'jwk', + jwk, + { + name: 'ECDH', + namedCurve: curve.web.web + }, + false, + [] + ); + [keyPair, recipient] = await Promise.all([keyPair, recipient]); + let s = webCrypto$8.deriveBits( + { + name: 'ECDH', + namedCurve: curve.web.web, + public: recipient + }, + keyPair.privateKey, + curve.web.sharedSize + ); + let p = webCrypto$8.exportKey( + 'jwk', + keyPair.publicKey + ); + [s, p] = await Promise.all([s, p]); + const sharedKey = new Uint8Array(s); + const publicKey = new Uint8Array(jwkToRawPublic(p)); + return { publicKey, sharedKey }; + } + + /** + * Generate ECDHE secret from private key and public part of ephemeral key using indutny/elliptic + * + * @param {Curve} curve - Elliptic curve object + * @param {Uint8Array} V - Public part of ephemeral key + * @param {Uint8Array} d - Recipient private key + * @returns {Promise<{secretKey: Uint8Array, sharedKey: Uint8Array}>} + * @async + */ + async function ellipticPrivateEphemeralKey(curve, V, d) { + const indutnyCurve = await getIndutnyCurve(curve.name); + V = keyFromPublic(indutnyCurve, V); + d = keyFromPrivate(indutnyCurve, d); + const secretKey = new Uint8Array(d.getPrivate()); + const S = d.derive(V.getPublic()); + const len = indutnyCurve.curve.p.byteLength(); + const sharedKey = S.toArrayLike(Uint8Array, 'be', len); + return { secretKey, sharedKey }; + } + + /** + * Generate ECDHE ephemeral key and secret from public key using indutny/elliptic + * + * @param {Curve} curve - Elliptic curve object + * @param {Uint8Array} Q - Recipient public key + * @returns {Promise<{publicKey: Uint8Array, sharedKey: Uint8Array}>} + * @async + */ + async function ellipticPublicEphemeralKey(curve, Q) { + const indutnyCurve = await getIndutnyCurve(curve.name); + const v = await curve.genKeyPair(); + Q = keyFromPublic(indutnyCurve, Q); + const V = keyFromPrivate(indutnyCurve, v.privateKey); + const publicKey = v.publicKey; + const S = V.derive(Q.getPublic()); + const len = indutnyCurve.curve.p.byteLength(); + const sharedKey = S.toArrayLike(Uint8Array, 'be', len); + return { publicKey, sharedKey }; + } + + /** + * Generate ECDHE secret from private key and public part of ephemeral key using nodeCrypto + * + * @param {Curve} curve - Elliptic curve object + * @param {Uint8Array} V - Public part of ephemeral key + * @param {Uint8Array} d - Recipient private key + * @returns {Promise<{secretKey: Uint8Array, sharedKey: Uint8Array}>} + * @async + */ + async function nodePrivateEphemeralKey(curve, V, d) { + const recipient = nodeCrypto$9.createECDH(curve.node.node); + recipient.setPrivateKey(d); + const sharedKey = new Uint8Array(recipient.computeSecret(V)); + const secretKey = new Uint8Array(recipient.getPrivateKey()); + return { secretKey, sharedKey }; + } + + /** + * Generate ECDHE ephemeral key and secret from public key using nodeCrypto + * + * @param {Curve} curve - Elliptic curve object + * @param {Uint8Array} Q - Recipient public key + * @returns {Promise<{publicKey: Uint8Array, sharedKey: Uint8Array}>} + * @async + */ + async function nodePublicEphemeralKey(curve, Q) { + const sender = nodeCrypto$9.createECDH(curve.node.node); + sender.generateKeys(); + const sharedKey = new Uint8Array(sender.computeSecret(Q)); + const publicKey = new Uint8Array(sender.getPublicKey()); + return { publicKey, sharedKey }; + } + + var ecdh = /*#__PURE__*/Object.freeze({ + __proto__: null, + validateParams: validateParams$4, + encrypt: encrypt$3, + decrypt: decrypt$3 + }); + + // OpenPGP.js - An OpenPGP implementation in javascript + + var elliptic = /*#__PURE__*/Object.freeze({ + __proto__: null, + Curve: Curve, + ecdh: ecdh, + ecdsa: ecdsa, + eddsa: eddsa, + generate: generate$1, + getPreferredHashAlgo: getPreferredHashAlgo + }); + + // GPG4Browsers - An OpenPGP implementation in javascript + + /* + TODO regarding the hash function, read: + https://tools.ietf.org/html/rfc4880#section-13.6 + https://tools.ietf.org/html/rfc4880#section-14 + */ + + /** + * DSA Sign function + * @param {Integer} hashAlgo + * @param {Uint8Array} hashed + * @param {Uint8Array} g + * @param {Uint8Array} p + * @param {Uint8Array} q + * @param {Uint8Array} x + * @returns {Promise<{ r: Uint8Array, s: Uint8Array }>} + * @async + */ + async function sign$3(hashAlgo, hashed, g, p, q, x) { + const BigInteger = await util.getBigInteger(); + const one = new BigInteger(1); + p = new BigInteger(p); + q = new BigInteger(q); + g = new BigInteger(g); + x = new BigInteger(x); + + let k; + let r; + let s; + let t; + g = g.mod(p); + x = x.mod(q); + // If the output size of the chosen hash is larger than the number of + // bits of q, the hash result is truncated to fit by taking the number + // of leftmost bits equal to the number of bits of q. This (possibly + // truncated) hash function result is treated as a number and used + // directly in the DSA signature algorithm. + const h = new BigInteger(hashed.subarray(0, q.byteLength())).mod(q); + // FIPS-186-4, section 4.6: + // The values of r and s shall be checked to determine if r = 0 or s = 0. + // If either r = 0 or s = 0, a new value of k shall be generated, and the + // signature shall be recalculated. It is extremely unlikely that r = 0 + // or s = 0 if signatures are generated properly. + while (true) { + // See Appendix B here: https://nvlpubs.nist.gov/nistpubs/FIPS/NIST.FIPS.186-4.pdf + k = await getRandomBigInteger(one, q); // returns in [1, q-1] + r = g.modExp(k, p).imod(q); // (g**k mod p) mod q + if (r.isZero()) { + continue; + } + const xr = x.mul(r).imod(q); + t = h.add(xr).imod(q); // H(m) + x*r mod q + s = k.modInv(q).imul(t).imod(q); // k**-1 * (H(m) + x*r) mod q + if (s.isZero()) { + continue; + } + break; + } + return { + r: r.toUint8Array('be', q.byteLength()), + s: s.toUint8Array('be', q.byteLength()) + }; + } + + /** + * DSA Verify function + * @param {Integer} hashAlgo + * @param {Uint8Array} r + * @param {Uint8Array} s + * @param {Uint8Array} hashed + * @param {Uint8Array} g + * @param {Uint8Array} p + * @param {Uint8Array} q + * @param {Uint8Array} y + * @returns {boolean} + * @async + */ + async function verify$3(hashAlgo, r, s, hashed, g, p, q, y) { + const BigInteger = await util.getBigInteger(); + const zero = new BigInteger(0); + r = new BigInteger(r); + s = new BigInteger(s); + + p = new BigInteger(p); + q = new BigInteger(q); + g = new BigInteger(g); + y = new BigInteger(y); + + if (r.lte(zero) || r.gte(q) || + s.lte(zero) || s.gte(q)) { + util.printDebug('invalid DSA Signature'); + return false; + } + const h = new BigInteger(hashed.subarray(0, q.byteLength())).imod(q); + const w = s.modInv(q); // s**-1 mod q + if (w.isZero()) { + util.printDebug('invalid DSA Signature'); + return false; + } + + g = g.mod(p); + y = y.mod(p); + const u1 = h.mul(w).imod(q); // H(m) * w mod q + const u2 = r.mul(w).imod(q); // r * w mod q + const t1 = g.modExp(u1, p); // g**u1 mod p + const t2 = y.modExp(u2, p); // y**u2 mod p + const v = t1.mul(t2).imod(p).imod(q); // (g**u1 * y**u2 mod p) mod q + return v.equal(r); + } + + /** + * Validate DSA parameters + * @param {Uint8Array} p - DSA prime + * @param {Uint8Array} q - DSA group order + * @param {Uint8Array} g - DSA sub-group generator + * @param {Uint8Array} y - DSA public key + * @param {Uint8Array} x - DSA private key + * @returns {Promise} Whether params are valid. + * @async + */ + async function validateParams$5(p, q, g, y, x) { + const BigInteger = await util.getBigInteger(); + p = new BigInteger(p); + q = new BigInteger(q); + g = new BigInteger(g); + y = new BigInteger(y); + const one = new BigInteger(1); + // Check that 1 < g < p + if (g.lte(one) || g.gte(p)) { + return false; + } + + /** + * Check that subgroup order q divides p-1 + */ + if (!p.dec().mod(q).isZero()) { + return false; + } + + /** + * g has order q + * Check that g ** q = 1 mod p + */ + if (!g.modExp(q, p).isOne()) { + return false; + } + + /** + * Check q is large and probably prime (we mainly want to avoid small factors) + */ + const qSize = new BigInteger(q.bitLength()); + const n150 = new BigInteger(150); + if (qSize.lt(n150) || !(await isProbablePrime(q, null, 32))) { + return false; + } + + /** + * Re-derive public key y' = g ** x mod p + * Expect y == y' + * + * Blinded exponentiation computes g**{rq + x} to compare to y + */ + x = new BigInteger(x); + const two = new BigInteger(2); + const r = await getRandomBigInteger(two.leftShift(qSize.dec()), two.leftShift(qSize)); // draw r of same size as q + const rqx = q.mul(r).add(x); + if (!y.equal(g.modExp(rqx, p))) { + return false; + } + + return true; + } + + var dsa = /*#__PURE__*/Object.freeze({ + __proto__: null, + sign: sign$3, + verify: verify$3, + validateParams: validateParams$5 + }); + + /** + * @fileoverview Asymmetric cryptography functions + * @module crypto/public_key + * @private + */ + + var publicKey = { + /** @see module:crypto/public_key/rsa */ + rsa: rsa, + /** @see module:crypto/public_key/elgamal */ + elgamal: elgamal, + /** @see module:crypto/public_key/elliptic */ + elliptic: elliptic, + /** @see module:crypto/public_key/dsa */ + dsa: dsa, + /** @see tweetnacl */ + nacl: naclFastLight + }; + + /** + * @fileoverview Provides functions for asymmetric signing and signature verification + * @module crypto/signature + * @private + */ + + /** + * Parse signature in binary form to get the parameters. + * The returned values are only padded for EdDSA, since in the other cases their expected length + * depends on the key params, hence we delegate the padding to the signature verification function. + * See {@link https://tools.ietf.org/html/rfc4880#section-9.1|RFC 4880 9.1} + * See {@link https://tools.ietf.org/html/rfc4880#section-5.2.2|RFC 4880 5.2.2.} + * @param {module:enums.publicKey} algo - Public key algorithm + * @param {Uint8Array} signature - Data for which the signature was created + * @returns {Promise} True if signature is valid. + * @async + */ + function parseSignatureParams(algo, signature) { + let read = 0; + switch (algo) { + // Algorithm-Specific Fields for RSA signatures: + // - MPI of RSA signature value m**d mod n. + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaSign: { + const s = util.readMPI(signature.subarray(read)); + // The signature needs to be the same length as the public key modulo n. + // We pad s on signature verification, where we have access to n. + return { s }; + } + // Algorithm-Specific Fields for DSA or ECDSA signatures: + // - MPI of DSA or ECDSA value r. + // - MPI of DSA or ECDSA value s. + case enums.publicKey.dsa: + case enums.publicKey.ecdsa: + { + const r = util.readMPI(signature.subarray(read)); read += r.length + 2; + const s = util.readMPI(signature.subarray(read)); + return { r, s }; + } + // Algorithm-Specific Fields for EdDSA signatures: + // - MPI of an EC point r. + // - EdDSA value s, in MPI, in the little endian representation + case enums.publicKey.eddsa: { + // When parsing little-endian MPI data, we always need to left-pad it, as done with big-endian values: + // https://www.ietf.org/archive/id/draft-ietf-openpgp-rfc4880bis-10.html#section-3.2-9 + let r = util.readMPI(signature.subarray(read)); read += r.length + 2; + r = util.leftPad(r, 32); + let s = util.readMPI(signature.subarray(read)); + s = util.leftPad(s, 32); + return { r, s }; + } + default: + throw new UnsupportedError('Unknown signature algorithm.'); + } + } + + /** + * Verifies the signature provided for data using specified algorithms and public key parameters. + * See {@link https://tools.ietf.org/html/rfc4880#section-9.1|RFC 4880 9.1} + * and {@link https://tools.ietf.org/html/rfc4880#section-9.4|RFC 4880 9.4} + * for public key and hash algorithms. + * @param {module:enums.publicKey} algo - Public key algorithm + * @param {module:enums.hash} hashAlgo - Hash algorithm + * @param {Object} signature - Named algorithm-specific signature parameters + * @param {Object} publicParams - Algorithm-specific public key parameters + * @param {Uint8Array} data - Data for which the signature was created + * @param {Uint8Array} hashed - The hashed data + * @returns {Promise} True if signature is valid. + * @async + */ + async function verify$4(algo, hashAlgo, signature, publicParams, data, hashed) { + switch (algo) { + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaSign: { + const { n, e } = publicParams; + const s = util.leftPad(signature.s, n.length); // padding needed for webcrypto and node crypto + return publicKey.rsa.verify(hashAlgo, data, s, n, e, hashed); + } + case enums.publicKey.dsa: { + const { g, p, q, y } = publicParams; + const { r, s } = signature; // no need to pad, since we always handle them as BigIntegers + return publicKey.dsa.verify(hashAlgo, r, s, hashed, g, p, q, y); + } + case enums.publicKey.ecdsa: { + const { oid, Q } = publicParams; + const curveSize = new publicKey.elliptic.Curve(oid).payloadSize; + // padding needed for webcrypto + const r = util.leftPad(signature.r, curveSize); + const s = util.leftPad(signature.s, curveSize); + return publicKey.elliptic.ecdsa.verify(oid, hashAlgo, { r, s }, data, Q, hashed); + } + case enums.publicKey.eddsa: { + const { oid, Q } = publicParams; + // signature already padded on parsing + return publicKey.elliptic.eddsa.verify(oid, hashAlgo, signature, data, Q, hashed); + } + default: + throw new Error('Unknown signature algorithm.'); + } + } + + /** + * Creates a signature on data using specified algorithms and private key parameters. + * See {@link https://tools.ietf.org/html/rfc4880#section-9.1|RFC 4880 9.1} + * and {@link https://tools.ietf.org/html/rfc4880#section-9.4|RFC 4880 9.4} + * for public key and hash algorithms. + * @param {module:enums.publicKey} algo - Public key algorithm + * @param {module:enums.hash} hashAlgo - Hash algorithm + * @param {Object} publicKeyParams - Algorithm-specific public and private key parameters + * @param {Object} privateKeyParams - Algorithm-specific public and private key parameters + * @param {Uint8Array} data - Data to be signed + * @param {Uint8Array} hashed - The hashed data + * @returns {Promise} Signature Object containing named signature parameters. + * @async + */ + async function sign$4(algo, hashAlgo, publicKeyParams, privateKeyParams, data, hashed) { + if (!publicKeyParams || !privateKeyParams) { + throw new Error('Missing key parameters'); + } + switch (algo) { + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaSign: { + const { n, e } = publicKeyParams; + const { d, p, q, u } = privateKeyParams; + const s = await publicKey.rsa.sign(hashAlgo, data, n, e, d, p, q, u, hashed); + return { s }; + } + case enums.publicKey.dsa: { + const { g, p, q } = publicKeyParams; + const { x } = privateKeyParams; + return publicKey.dsa.sign(hashAlgo, hashed, g, p, q, x); + } + case enums.publicKey.elgamal: { + throw new Error('Signing with Elgamal is not defined in the OpenPGP standard.'); + } + case enums.publicKey.ecdsa: { + const { oid, Q } = publicKeyParams; + const { d } = privateKeyParams; + return publicKey.elliptic.ecdsa.sign(oid, hashAlgo, data, Q, d, hashed); + } + case enums.publicKey.eddsa: { + const { oid, Q } = publicKeyParams; + const { seed } = privateKeyParams; + return publicKey.elliptic.eddsa.sign(oid, hashAlgo, data, Q, seed, hashed); + } + default: + throw new Error('Unknown signature algorithm.'); + } + } + + var signature = /*#__PURE__*/Object.freeze({ + __proto__: null, + parseSignatureParams: parseSignatureParams, + verify: verify$4, + sign: sign$4 + }); + + // OpenPGP.js - An OpenPGP implementation in javascript + + class ECDHSymmetricKey { + constructor(data) { + if (typeof data === 'undefined') { + data = new Uint8Array([]); + } else if (util.isString(data)) { + data = util.stringToUint8Array(data); + } else { + data = new Uint8Array(data); + } + this.data = data; + } + + /** + * Read an ECDHSymmetricKey from an Uint8Array + * @param {Uint8Array} input - Where to read the encoded symmetric key from + * @returns {Number} Number of read bytes. + */ + read(input) { + if (input.length >= 1) { + const length = input[0]; + if (input.length >= 1 + length) { + this.data = input.subarray(1, 1 + length); + return 1 + this.data.length; + } + } + throw new Error('Invalid symmetric key'); + } + + /** + * Write an ECDHSymmetricKey as an Uint8Array + * @returns {Uint8Array} An array containing the value + */ + write() { + return util.concatUint8Array([new Uint8Array([this.data.length]), this.data]); + } + } + + // OpenPGP.js - An OpenPGP implementation in javascript + // Copyright (C) 2015-2016 Decentral + // + // This library is free software; you can redistribute it and/or + // modify it under the terms of the GNU Lesser General Public + // License as published by the Free Software Foundation; either + // version 3.0 of the License, or (at your option) any later version. + // + // This library is distributed in the hope that it will be useful, + // but WITHOUT ANY WARRANTY; without even the implied warranty of + // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + // Lesser General Public License for more details. + // + // You should have received a copy of the GNU Lesser General Public + // License along with this library; if not, write to the Free Software + // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + + /** + * Implementation of type KDF parameters + * + * {@link https://tools.ietf.org/html/rfc6637#section-7|RFC 6637 7}: + * A key derivation function (KDF) is necessary to implement the EC + * encryption. The Concatenation Key Derivation Function (Approved + * Alternative 1) [NIST-SP800-56A] with the KDF hash function that is + * SHA2-256 [FIPS-180-3] or stronger is REQUIRED. + * @module type/kdf_params + * @private + */ + + class KDFParams { + /** + * @param {enums.hash} hash - Hash algorithm + * @param {enums.symmetric} cipher - Symmetric algorithm + */ + constructor(data) { + if (data) { + const { hash, cipher } = data; + this.hash = hash; + this.cipher = cipher; + } else { + this.hash = null; + this.cipher = null; + } + } + + /** + * Read KDFParams from an Uint8Array + * @param {Uint8Array} input - Where to read the KDFParams from + * @returns {Number} Number of read bytes. + */ + read(input) { + if (input.length < 4 || input[0] !== 3 || input[1] !== 1) { + throw new Error('Cannot read KDFParams'); + } + this.hash = input[2]; + this.cipher = input[3]; + return 4; + } + + /** + * Write KDFParams to an Uint8Array + * @returns {Uint8Array} Array with the KDFParams value + */ + write() { + return new Uint8Array([3, 1, this.hash, this.cipher]); + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Encrypts data using specified algorithm and public key parameters. + * See {@link https://tools.ietf.org/html/rfc4880#section-9.1|RFC 4880 9.1} for public key algorithms. + * @param {module:enums.publicKey} algo - Public key algorithm + * @param {Object} publicParams - Algorithm-specific public key parameters + * @param {Uint8Array} data - Data to be encrypted + * @param {Uint8Array} fingerprint - Recipient fingerprint + * @returns {Promise} Encrypted session key parameters. + * @async + */ + async function publicKeyEncrypt(algo, publicParams, data, fingerprint) { + switch (algo) { + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaEncryptSign: { + const { n, e } = publicParams; + const c = await publicKey.rsa.encrypt(data, n, e); + return { c }; + } + case enums.publicKey.elgamal: { + const { p, g, y } = publicParams; + return publicKey.elgamal.encrypt(data, p, g, y); + } + case enums.publicKey.ecdh: { + const { oid, Q, kdfParams } = publicParams; + const { publicKey: V, wrappedKey: C } = await publicKey.elliptic.ecdh.encrypt( + oid, kdfParams, data, Q, fingerprint); + return { V, C: new ECDHSymmetricKey(C) }; + } + default: + return []; + } + } + + /** + * Decrypts data using specified algorithm and private key parameters. + * See {@link https://tools.ietf.org/html/rfc4880#section-5.5.3|RFC 4880 5.5.3} + * @param {module:enums.publicKey} algo - Public key algorithm + * @param {Object} publicKeyParams - Algorithm-specific public key parameters + * @param {Object} privateKeyParams - Algorithm-specific private key parameters + * @param {Object} sessionKeyParams - Encrypted session key parameters + * @param {Uint8Array} fingerprint - Recipient fingerprint + * @param {Uint8Array} [randomPayload] - Data to return on decryption error, instead of throwing + * (needed for constant-time processing in RSA and ElGamal) + * @returns {Promise} Decrypted data. + * @throws {Error} on sensitive decryption error, unless `randomPayload` is given + * @async + */ + async function publicKeyDecrypt(algo, publicKeyParams, privateKeyParams, sessionKeyParams, fingerprint, randomPayload) { + switch (algo) { + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaEncrypt: { + const { c } = sessionKeyParams; + const { n, e } = publicKeyParams; + const { d, p, q, u } = privateKeyParams; + return publicKey.rsa.decrypt(c, n, e, d, p, q, u, randomPayload); + } + case enums.publicKey.elgamal: { + const { c1, c2 } = sessionKeyParams; + const p = publicKeyParams.p; + const x = privateKeyParams.x; + return publicKey.elgamal.decrypt(c1, c2, p, x, randomPayload); + } + case enums.publicKey.ecdh: { + const { oid, Q, kdfParams } = publicKeyParams; + const { d } = privateKeyParams; + const { V, C } = sessionKeyParams; + return publicKey.elliptic.ecdh.decrypt( + oid, kdfParams, V, C.data, Q, d, fingerprint); + } + default: + throw new Error('Unknown public key encryption algorithm.'); + } + } + + /** + * Parse public key material in binary form to get the key parameters + * @param {module:enums.publicKey} algo - The key algorithm + * @param {Uint8Array} bytes - The key material to parse + * @returns {{ read: Number, publicParams: Object }} Number of read bytes plus key parameters referenced by name. + */ + function parsePublicKeyParams(algo, bytes) { + let read = 0; + switch (algo) { + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaSign: { + const n = util.readMPI(bytes.subarray(read)); read += n.length + 2; + const e = util.readMPI(bytes.subarray(read)); read += e.length + 2; + return { read, publicParams: { n, e } }; + } + case enums.publicKey.dsa: { + const p = util.readMPI(bytes.subarray(read)); read += p.length + 2; + const q = util.readMPI(bytes.subarray(read)); read += q.length + 2; + const g = util.readMPI(bytes.subarray(read)); read += g.length + 2; + const y = util.readMPI(bytes.subarray(read)); read += y.length + 2; + return { read, publicParams: { p, q, g, y } }; + } + case enums.publicKey.elgamal: { + const p = util.readMPI(bytes.subarray(read)); read += p.length + 2; + const g = util.readMPI(bytes.subarray(read)); read += g.length + 2; + const y = util.readMPI(bytes.subarray(read)); read += y.length + 2; + return { read, publicParams: { p, g, y } }; + } + case enums.publicKey.ecdsa: { + const oid = new OID(); read += oid.read(bytes); + checkSupportedCurve(oid); + const Q = util.readMPI(bytes.subarray(read)); read += Q.length + 2; + return { read: read, publicParams: { oid, Q } }; + } + case enums.publicKey.eddsa: { + const oid = new OID(); read += oid.read(bytes); + checkSupportedCurve(oid); + let Q = util.readMPI(bytes.subarray(read)); read += Q.length + 2; + Q = util.leftPad(Q, 33); + return { read: read, publicParams: { oid, Q } }; + } + case enums.publicKey.ecdh: { + const oid = new OID(); read += oid.read(bytes); + checkSupportedCurve(oid); + const Q = util.readMPI(bytes.subarray(read)); read += Q.length + 2; + const kdfParams = new KDFParams(); read += kdfParams.read(bytes.subarray(read)); + return { read: read, publicParams: { oid, Q, kdfParams } }; + } + default: + throw new UnsupportedError('Unknown public key encryption algorithm.'); + } + } + + /** + * Parse private key material in binary form to get the key parameters + * @param {module:enums.publicKey} algo - The key algorithm + * @param {Uint8Array} bytes - The key material to parse + * @param {Object} publicParams - (ECC only) public params, needed to format some private params + * @returns {{ read: Number, privateParams: Object }} Number of read bytes plus the key parameters referenced by name. + */ + function parsePrivateKeyParams(algo, bytes, publicParams) { + let read = 0; + switch (algo) { + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaSign: { + const d = util.readMPI(bytes.subarray(read)); read += d.length + 2; + const p = util.readMPI(bytes.subarray(read)); read += p.length + 2; + const q = util.readMPI(bytes.subarray(read)); read += q.length + 2; + const u = util.readMPI(bytes.subarray(read)); read += u.length + 2; + return { read, privateParams: { d, p, q, u } }; + } + case enums.publicKey.dsa: + case enums.publicKey.elgamal: { + const x = util.readMPI(bytes.subarray(read)); read += x.length + 2; + return { read, privateParams: { x } }; + } + case enums.publicKey.ecdsa: + case enums.publicKey.ecdh: { + const curve = new Curve(publicParams.oid); + let d = util.readMPI(bytes.subarray(read)); read += d.length + 2; + d = util.leftPad(d, curve.payloadSize); + return { read, privateParams: { d } }; + } + case enums.publicKey.eddsa: { + const curve = new Curve(publicParams.oid); + let seed = util.readMPI(bytes.subarray(read)); read += seed.length + 2; + seed = util.leftPad(seed, curve.payloadSize); + return { read, privateParams: { seed } }; + } + default: + throw new UnsupportedError('Unknown public key encryption algorithm.'); + } + } + + /** Returns the types comprising the encrypted session key of an algorithm + * @param {module:enums.publicKey} algo - The key algorithm + * @param {Uint8Array} bytes - The key material to parse + * @returns {Object} The session key parameters referenced by name. + */ + function parseEncSessionKeyParams(algo, bytes) { + let read = 0; + switch (algo) { + // Algorithm-Specific Fields for RSA encrypted session keys: + // - MPI of RSA encrypted value m**e mod n. + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaEncryptSign: { + const c = util.readMPI(bytes.subarray(read)); + return { c }; + } + + // Algorithm-Specific Fields for Elgamal encrypted session keys: + // - MPI of Elgamal value g**k mod p + // - MPI of Elgamal value m * y**k mod p + case enums.publicKey.elgamal: { + const c1 = util.readMPI(bytes.subarray(read)); read += c1.length + 2; + const c2 = util.readMPI(bytes.subarray(read)); + return { c1, c2 }; + } + // Algorithm-Specific Fields for ECDH encrypted session keys: + // - MPI containing the ephemeral key used to establish the shared secret + // - ECDH Symmetric Key + case enums.publicKey.ecdh: { + const V = util.readMPI(bytes.subarray(read)); read += V.length + 2; + const C = new ECDHSymmetricKey(); C.read(bytes.subarray(read)); + return { V, C }; + } + default: + throw new UnsupportedError('Unknown public key encryption algorithm.'); + } + } + + /** + * Convert params to MPI and serializes them in the proper order + * @param {module:enums.publicKey} algo - The public key algorithm + * @param {Object} params - The key parameters indexed by name + * @returns {Uint8Array} The array containing the MPIs. + */ + function serializeParams(algo, params) { + const orderedParams = Object.keys(params).map(name => { + const param = params[name]; + return util.isUint8Array(param) ? util.uint8ArrayToMPI(param) : param.write(); + }); + return util.concatUint8Array(orderedParams); + } + + /** + * Generate algorithm-specific key parameters + * @param {module:enums.publicKey} algo - The public key algorithm + * @param {Integer} bits - Bit length for RSA keys + * @param {module:type/oid} oid - Object identifier for ECC keys + * @returns {Promise<{ publicParams: {Object}, privateParams: {Object} }>} The parameters referenced by name. + * @async + */ + function generateParams(algo, bits, oid) { + switch (algo) { + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaSign: { + return publicKey.rsa.generate(bits, 65537).then(({ n, e, d, p, q, u }) => ({ + privateParams: { d, p, q, u }, + publicParams: { n, e } + })); + } + case enums.publicKey.ecdsa: + return publicKey.elliptic.generate(oid).then(({ oid, Q, secret }) => ({ + privateParams: { d: secret }, + publicParams: { oid: new OID(oid), Q } + })); + case enums.publicKey.eddsa: + return publicKey.elliptic.generate(oid).then(({ oid, Q, secret }) => ({ + privateParams: { seed: secret }, + publicParams: { oid: new OID(oid), Q } + })); + case enums.publicKey.ecdh: + return publicKey.elliptic.generate(oid).then(({ oid, Q, secret, hash, cipher }) => ({ + privateParams: { d: secret }, + publicParams: { + oid: new OID(oid), + Q, + kdfParams: new KDFParams({ hash, cipher }) + } + })); + case enums.publicKey.dsa: + case enums.publicKey.elgamal: + throw new Error('Unsupported algorithm for key generation.'); + default: + throw new Error('Unknown public key algorithm.'); + } + } + + /** + * Validate algorithm-specific key parameters + * @param {module:enums.publicKey} algo - The public key algorithm + * @param {Object} publicParams - Algorithm-specific public key parameters + * @param {Object} privateParams - Algorithm-specific private key parameters + * @returns {Promise} Whether the parameters are valid. + * @async + */ + async function validateParams$6(algo, publicParams, privateParams) { + if (!publicParams || !privateParams) { + throw new Error('Missing key parameters'); + } + switch (algo) { + case enums.publicKey.rsaEncrypt: + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaSign: { + const { n, e } = publicParams; + const { d, p, q, u } = privateParams; + return publicKey.rsa.validateParams(n, e, d, p, q, u); + } + case enums.publicKey.dsa: { + const { p, q, g, y } = publicParams; + const { x } = privateParams; + return publicKey.dsa.validateParams(p, q, g, y, x); + } + case enums.publicKey.elgamal: { + const { p, g, y } = publicParams; + const { x } = privateParams; + return publicKey.elgamal.validateParams(p, g, y, x); + } + case enums.publicKey.ecdsa: + case enums.publicKey.ecdh: { + const algoModule = publicKey.elliptic[enums.read(enums.publicKey, algo)]; + const { oid, Q } = publicParams; + const { d } = privateParams; + return algoModule.validateParams(oid, Q, d); + } + case enums.publicKey.eddsa: { + const { oid, Q } = publicParams; + const { seed } = privateParams; + return publicKey.elliptic.eddsa.validateParams(oid, Q, seed); + } + default: + throw new Error('Unknown public key algorithm.'); + } + } + + /** + * Generates a random byte prefix for the specified algorithm + * See {@link https://tools.ietf.org/html/rfc4880#section-9.2|RFC 4880 9.2} for algorithms. + * @param {module:enums.symmetric} algo - Symmetric encryption algorithm + * @returns {Promise} Random bytes with length equal to the block size of the cipher, plus the last two bytes repeated. + * @async + */ + async function getPrefixRandom(algo) { + const { blockSize } = getCipher(algo); + const prefixrandom = await getRandomBytes(blockSize); + const repeat = new Uint8Array([prefixrandom[prefixrandom.length - 2], prefixrandom[prefixrandom.length - 1]]); + return util.concat([prefixrandom, repeat]); + } + + /** + * Generating a session key for the specified symmetric algorithm + * See {@link https://tools.ietf.org/html/rfc4880#section-9.2|RFC 4880 9.2} for algorithms. + * @param {module:enums.symmetric} algo - Symmetric encryption algorithm + * @returns {Uint8Array} Random bytes as a string to be used as a key. + */ + function generateSessionKey(algo) { + const { keySize } = getCipher(algo); + return getRandomBytes(keySize); + } + + /** + * Get implementation of the given AEAD mode + * @param {enums.aead} algo + * @returns {Object} + * @throws {Error} on invalid algo + */ + function getAEADMode(algo) { + const algoName = enums.read(enums.aead, algo); + return mode[algoName]; + } + + /** + * Check whether the given curve OID is supported + * @param {module:type/oid} oid - EC object identifier + * @throws {UnsupportedError} if curve is not supported + */ + function checkSupportedCurve(oid) { + try { + oid.getName(); + } catch (e) { + throw new UnsupportedError('Unknown curve OID'); + } + } + + var crypto$1 = /*#__PURE__*/Object.freeze({ + __proto__: null, + publicKeyEncrypt: publicKeyEncrypt, + publicKeyDecrypt: publicKeyDecrypt, + parsePublicKeyParams: parsePublicKeyParams, + parsePrivateKeyParams: parsePrivateKeyParams, + parseEncSessionKeyParams: parseEncSessionKeyParams, + serializeParams: serializeParams, + generateParams: generateParams, + validateParams: validateParams$6, + getPrefixRandom: getPrefixRandom, + generateSessionKey: generateSessionKey, + getAEADMode: getAEADMode, + getCipher: getCipher + }); + + /** + * @fileoverview Provides access to all cryptographic primitives used in OpenPGP.js + * @see module:crypto/crypto + * @see module:crypto/signature + * @see module:crypto/public_key + * @see module:crypto/cipher + * @see module:crypto/random + * @see module:crypto/hash + * @module crypto + * @private + */ + + // TODO move cfb and gcm to cipher + const mod = { + /** @see module:crypto/cipher */ + cipher: cipher, + /** @see module:crypto/hash */ + hash: hash, + /** @see module:crypto/mode */ + mode: mode, + /** @see module:crypto/public_key */ + publicKey: publicKey, + /** @see module:crypto/signature */ + signature: signature, + /** @see module:crypto/random */ + random: random, + /** @see module:crypto/pkcs1 */ + pkcs1: pkcs1, + /** @see module:crypto/pkcs5 */ + pkcs5: pkcs5, + /** @see module:crypto/aes_kw */ + aesKW: aesKW + }; + + Object.assign(mod, crypto$1); + + var TYPED_OK = typeof Uint8Array !== "undefined" && + typeof Uint16Array !== "undefined" && + typeof Int32Array !== "undefined"; + + + // reduce buffer size, avoiding mem copy + function shrinkBuf(buf, size) { + if (buf.length === size) { + return buf; + } + if (buf.subarray) { + return buf.subarray(0, size); + } + buf.length = size; + return buf; + } + + + const fnTyped = { + arraySet: function (dest, src, src_offs, len, dest_offs) { + if (src.subarray && dest.subarray) { + dest.set(src.subarray(src_offs, src_offs + len), dest_offs); + return; + } + // Fallback to ordinary array + for (let i = 0; i < len; i++) { + dest[dest_offs + i] = src[src_offs + i]; + } + }, + // Join array of chunks to single array. + flattenChunks: function (chunks) { + let i, l, len, pos, chunk; + + // calculate data length + len = 0; + for (i = 0, l = chunks.length; i < l; i++) { + len += chunks[i].length; + } + + // join chunks + const result = new Uint8Array(len); + pos = 0; + for (i = 0, l = chunks.length; i < l; i++) { + chunk = chunks[i]; + result.set(chunk, pos); + pos += chunk.length; + } + + return result; + } + }; + + const fnUntyped = { + arraySet: function (dest, src, src_offs, len, dest_offs) { + for (let i = 0; i < len; i++) { + dest[dest_offs + i] = src[src_offs + i]; + } + }, + // Join array of chunks to single array. + flattenChunks: function (chunks) { + return [].concat.apply([], chunks); + } + }; + + + // Enable/Disable typed arrays use, for testing + // + + let Buf8 = TYPED_OK ? Uint8Array : Array; + let Buf16 = TYPED_OK ? Uint16Array : Array; + let Buf32 = TYPED_OK ? Int32Array : Array; + let flattenChunks = TYPED_OK ? fnTyped.flattenChunks : fnUntyped.flattenChunks; + let arraySet = TYPED_OK ? fnTyped.arraySet : fnUntyped.arraySet; + + // (C) 1995-2013 Jean-loup Gailly and Mark Adler + // (C) 2014-2017 Vitaly Puzrin and Andrey Tupitsin + // + // This software is provided 'as-is', without any express or implied + // warranty. In no event will the authors be held liable for any damages + // arising from the use of this software. + // + // Permission is granted to anyone to use this software for any purpose, + // including commercial applications, and to alter it and redistribute it + // freely, subject to the following restrictions: + // + // 1. The origin of this software must not be misrepresented; you must not + // claim that you wrote the original software. If you use this software + // in a product, an acknowledgment in the product documentation would be + // appreciated but is not required. + // 2. Altered source versions must be plainly marked as such, and must not be + // misrepresented as being the original software. + // 3. This notice may not be removed or altered from any source distribution. + + /* Allowed flush values; see deflate() and inflate() below for details */ + const Z_NO_FLUSH = 0; + const Z_PARTIAL_FLUSH = 1; + const Z_SYNC_FLUSH = 2; + const Z_FULL_FLUSH = 3; + const Z_FINISH = 4; + const Z_BLOCK = 5; + const Z_TREES = 6; + + /* Return codes for the compression/decompression functions. Negative values + * are errors, positive values are used for special but normal events. + */ + const Z_OK = 0; + const Z_STREAM_END = 1; + const Z_NEED_DICT = 2; + const Z_STREAM_ERROR = -2; + const Z_DATA_ERROR = -3; + //export const Z_MEM_ERROR = -4; + const Z_BUF_ERROR = -5; + const Z_DEFAULT_COMPRESSION = -1; + + + const Z_FILTERED = 1; + const Z_HUFFMAN_ONLY = 2; + const Z_RLE = 3; + const Z_FIXED = 4; + const Z_DEFAULT_STRATEGY = 0; + + /* Possible values of the data_type field (though see inflate()) */ + const Z_BINARY = 0; + const Z_TEXT = 1; + //export const Z_ASCII = 1; // = Z_TEXT (deprecated) + const Z_UNKNOWN = 2; + + /* The deflate compression method */ + const Z_DEFLATED = 8; + //export const Z_NULL = null // Use -1 or null inline, depending on var type + + /*============================================================================*/ + + + function zero$1(buf) { + let len = buf.length; while (--len >= 0) { + buf[len] = 0; + } + } + + // From zutil.h + + const STORED_BLOCK = 0; + const STATIC_TREES = 1; + const DYN_TREES = 2; + /* The three kinds of block type */ + + const MIN_MATCH = 3; + const MAX_MATCH = 258; + /* The minimum and maximum match lengths */ + + // From deflate.h + /* =========================================================================== + * Internal compression state. + */ + + const LENGTH_CODES = 29; + /* number of length codes, not counting the special END_BLOCK code */ + + const LITERALS = 256; + /* number of literal bytes 0..255 */ + + const L_CODES = LITERALS + 1 + LENGTH_CODES; + /* number of Literal or Length codes, including the END_BLOCK code */ + + const D_CODES = 30; + /* number of distance codes */ + + const BL_CODES = 19; + /* number of codes used to transfer the bit lengths */ + + const HEAP_SIZE = 2 * L_CODES + 1; + /* maximum heap size */ + + const MAX_BITS = 15; + /* All codes must not exceed MAX_BITS bits */ + + const Buf_size = 16; + /* size of bit buffer in bi_buf */ + + + /* =========================================================================== + * Constants + */ + + const MAX_BL_BITS = 7; + /* Bit length codes must not exceed MAX_BL_BITS bits */ + + const END_BLOCK = 256; + /* end of block literal code */ + + const REP_3_6 = 16; + /* repeat previous bit length 3-6 times (2 bits of repeat count) */ + + const REPZ_3_10 = 17; + /* repeat a zero length 3-10 times (3 bits of repeat count) */ + + const REPZ_11_138 = 18; + /* repeat a zero length 11-138 times (7 bits of repeat count) */ + + /* eslint-disable comma-spacing,array-bracket-spacing */ + const extra_lbits = /* extra bits for each length code */ + [0,0,0,0,0,0,0,0,1,1,1,1,2,2,2,2,3,3,3,3,4,4,4,4,5,5,5,5,0]; + + const extra_dbits = /* extra bits for each distance code */ + [0,0,0,0,1,1,2,2,3,3,4,4,5,5,6,6,7,7,8,8,9,9,10,10,11,11,12,12,13,13]; + + const extra_blbits = /* extra bits for each bit length code */ + [0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,2,3,7]; + + const bl_order = + [16,17,18,0,8,7,9,6,10,5,11,4,12,3,13,2,14,1,15]; + /* eslint-enable comma-spacing,array-bracket-spacing */ + + /* The lengths of the bit length codes are sent in order of decreasing + * probability, to avoid transmitting the lengths for unused bit length codes. + */ + + /* =========================================================================== + * Local data. These are initialized only once. + */ + + // We pre-fill arrays with 0 to avoid uninitialized gaps + + const DIST_CODE_LEN = 512; /* see definition of array dist_code below */ + + // !!!! Use flat array instead of structure, Freq = i*2, Len = i*2+1 + const static_ltree = new Array((L_CODES + 2) * 2); + zero$1(static_ltree); + /* The static literal tree. Since the bit lengths are imposed, there is no + * need for the L_CODES extra codes used during heap construction. However + * The codes 286 and 287 are needed to build a canonical tree (see _tr_init + * below). + */ + + const static_dtree = new Array(D_CODES * 2); + zero$1(static_dtree); + /* The static distance tree. (Actually a trivial tree since all codes use + * 5 bits.) + */ + + const _dist_code = new Array(DIST_CODE_LEN); + zero$1(_dist_code); + /* Distance codes. The first 256 values correspond to the distances + * 3 .. 258, the last 256 values correspond to the top 8 bits of + * the 15 bit distances. + */ + + const _length_code = new Array(MAX_MATCH - MIN_MATCH + 1); + zero$1(_length_code); + /* length code for each normalized match length (0 == MIN_MATCH) */ + + const base_length = new Array(LENGTH_CODES); + zero$1(base_length); + /* First normalized length for each code (0 = MIN_MATCH) */ + + const base_dist = new Array(D_CODES); + zero$1(base_dist); + /* First normalized distance for each code (0 = distance of 1) */ + + + function StaticTreeDesc(static_tree, extra_bits, extra_base, elems, max_length) { + + this.static_tree = static_tree; /* static tree or NULL */ + this.extra_bits = extra_bits; /* extra bits for each code or NULL */ + this.extra_base = extra_base; /* base index for extra_bits */ + this.elems = elems; /* max number of elements in the tree */ + this.max_length = max_length; /* max bit length for the codes */ + + // show if `static_tree` has data or dummy - needed for monomorphic objects + this.has_stree = static_tree && static_tree.length; + } + + + let static_l_desc; + let static_d_desc; + let static_bl_desc; + + + function TreeDesc(dyn_tree, stat_desc) { + this.dyn_tree = dyn_tree; /* the dynamic tree */ + this.max_code = 0; /* largest code with non zero frequency */ + this.stat_desc = stat_desc; /* the corresponding static tree */ + } + + + + function d_code(dist) { + return dist < 256 ? _dist_code[dist] : _dist_code[256 + (dist >>> 7)]; + } + + + /* =========================================================================== + * Output a short LSB first on the stream. + * IN assertion: there is enough room in pendingBuf. + */ + function put_short(s, w) { + // put_byte(s, (uch)((w) & 0xff)); + // put_byte(s, (uch)((ush)(w) >> 8)); + s.pending_buf[s.pending++] = w & 0xff; + s.pending_buf[s.pending++] = w >>> 8 & 0xff; + } + + + /* =========================================================================== + * Send a value on a given number of bits. + * IN assertion: length <= 16 and value fits in length bits. + */ + function send_bits(s, value, length) { + if (s.bi_valid > Buf_size - length) { + s.bi_buf |= value << s.bi_valid & 0xffff; + put_short(s, s.bi_buf); + s.bi_buf = value >> Buf_size - s.bi_valid; + s.bi_valid += length - Buf_size; + } else { + s.bi_buf |= value << s.bi_valid & 0xffff; + s.bi_valid += length; + } + } + + + function send_code(s, c, tree) { + send_bits(s, tree[c * 2]/*.Code*/, tree[c * 2 + 1]/*.Len*/); + } + + + /* =========================================================================== + * Reverse the first len bits of a code, using straightforward code (a faster + * method would use a table) + * IN assertion: 1 <= len <= 15 + */ + function bi_reverse(code, len) { + let res = 0; + do { + res |= code & 1; + code >>>= 1; + res <<= 1; + } while (--len > 0); + return res >>> 1; + } + + + /* =========================================================================== + * Flush the bit buffer, keeping at most 7 bits in it. + */ + function bi_flush(s) { + if (s.bi_valid === 16) { + put_short(s, s.bi_buf); + s.bi_buf = 0; + s.bi_valid = 0; + + } else if (s.bi_valid >= 8) { + s.pending_buf[s.pending++] = s.bi_buf & 0xff; + s.bi_buf >>= 8; + s.bi_valid -= 8; + } + } + + + /* =========================================================================== + * Compute the optimal bit lengths for a tree and update the total bit length + * for the current block. + * IN assertion: the fields freq and dad are set, heap[heap_max] and + * above are the tree nodes sorted by increasing frequency. + * OUT assertions: the field len is set to the optimal bit length, the + * array bl_count contains the frequencies for each bit length. + * The length opt_len is updated; static_len is also updated if stree is + * not null. + */ + function gen_bitlen(s, desc) + // deflate_state *s; + // tree_desc *desc; /* the tree descriptor */ + { + const tree = desc.dyn_tree; + const max_code = desc.max_code; + const stree = desc.stat_desc.static_tree; + const has_stree = desc.stat_desc.has_stree; + const extra = desc.stat_desc.extra_bits; + const base = desc.stat_desc.extra_base; + const max_length = desc.stat_desc.max_length; + let h; /* heap index */ + let n, m; /* iterate over the tree elements */ + let bits; /* bit length */ + let xbits; /* extra bits */ + let f; /* frequency */ + let overflow = 0; /* number of elements with bit length too large */ + + for (bits = 0; bits <= MAX_BITS; bits++) { + s.bl_count[bits] = 0; + } + + /* In a first pass, compute the optimal bit lengths (which may + * overflow in the case of the bit length tree). + */ + tree[s.heap[s.heap_max] * 2 + 1]/*.Len*/ = 0; /* root of the heap */ + + for (h = s.heap_max + 1; h < HEAP_SIZE; h++) { + n = s.heap[h]; + bits = tree[tree[n * 2 + 1]/*.Dad*/ * 2 + 1]/*.Len*/ + 1; + if (bits > max_length) { + bits = max_length; + overflow++; + } + tree[n * 2 + 1]/*.Len*/ = bits; + /* We overwrite tree[n].Dad which is no longer needed */ + + if (n > max_code) { + continue; + } /* not a leaf node */ + + s.bl_count[bits]++; + xbits = 0; + if (n >= base) { + xbits = extra[n - base]; + } + f = tree[n * 2]/*.Freq*/; + s.opt_len += f * (bits + xbits); + if (has_stree) { + s.static_len += f * (stree[n * 2 + 1]/*.Len*/ + xbits); + } + } + if (overflow === 0) { + return; + } + + // Trace((stderr,"\nbit length overflow\n")); + /* This happens for example on obj2 and pic of the Calgary corpus */ + + /* Find the first bit length which could increase: */ + do { + bits = max_length - 1; + while (s.bl_count[bits] === 0) { + bits--; + } + s.bl_count[bits]--; /* move one leaf down the tree */ + s.bl_count[bits + 1] += 2; /* move one overflow item as its brother */ + s.bl_count[max_length]--; + /* The brother of the overflow item also moves one step up, + * but this does not affect bl_count[max_length] + */ + overflow -= 2; + } while (overflow > 0); + + /* Now recompute all bit lengths, scanning in increasing frequency. + * h is still equal to HEAP_SIZE. (It is simpler to reconstruct all + * lengths instead of fixing only the wrong ones. This idea is taken + * from 'ar' written by Haruhiko Okumura.) + */ + for (bits = max_length; bits !== 0; bits--) { + n = s.bl_count[bits]; + while (n !== 0) { + m = s.heap[--h]; + if (m > max_code) { + continue; + } + if (tree[m * 2 + 1]/*.Len*/ !== bits) { + // Trace((stderr,"code %d bits %d->%d\n", m, tree[m].Len, bits)); + s.opt_len += (bits - tree[m * 2 + 1]/*.Len*/) * tree[m * 2]/*.Freq*/; + tree[m * 2 + 1]/*.Len*/ = bits; + } + n--; + } + } + } + + + /* =========================================================================== + * Generate the codes for a given tree and bit counts (which need not be + * optimal). + * IN assertion: the array bl_count contains the bit length statistics for + * the given tree and the field len is set for all tree elements. + * OUT assertion: the field code is set for all tree elements of non + * zero code length. + */ + function gen_codes(tree, max_code, bl_count) + // ct_data *tree; /* the tree to decorate */ + // int max_code; /* largest code with non zero frequency */ + // ushf *bl_count; /* number of codes at each bit length */ + { + const next_code = new Array(MAX_BITS + 1); /* next code value for each bit length */ + let code = 0; /* running code value */ + let bits; /* bit index */ + let n; /* code index */ + + /* The distribution counts are first used to generate the code values + * without bit reversal. + */ + for (bits = 1; bits <= MAX_BITS; bits++) { + next_code[bits] = code = code + bl_count[bits - 1] << 1; + } + /* Check that the bit counts in bl_count are consistent. The last code + * must be all ones. + */ + //Assert (code + bl_count[MAX_BITS]-1 == (1< length code (0..28) */ + length = 0; + for (code = 0; code < LENGTH_CODES - 1; code++) { + base_length[code] = length; + for (n = 0; n < 1 << extra_lbits[code]; n++) { + _length_code[length++] = code; + } + } + //Assert (length == 256, "tr_static_init: length != 256"); + /* Note that the length 255 (match length 258) can be represented + * in two different ways: code 284 + 5 bits or code 285, so we + * overwrite length_code[255] to use the best encoding: + */ + _length_code[length - 1] = code; + + /* Initialize the mapping dist (0..32K) -> dist code (0..29) */ + dist = 0; + for (code = 0; code < 16; code++) { + base_dist[code] = dist; + for (n = 0; n < 1 << extra_dbits[code]; n++) { + _dist_code[dist++] = code; + } + } + //Assert (dist == 256, "tr_static_init: dist != 256"); + dist >>= 7; /* from now on, all distances are divided by 128 */ + for (; code < D_CODES; code++) { + base_dist[code] = dist << 7; + for (n = 0; n < 1 << extra_dbits[code] - 7; n++) { + _dist_code[256 + dist++] = code; + } + } + //Assert (dist == 256, "tr_static_init: 256+dist != 512"); + + /* Construct the codes of the static literal tree */ + for (bits = 0; bits <= MAX_BITS; bits++) { + bl_count[bits] = 0; + } + + n = 0; + while (n <= 143) { + static_ltree[n * 2 + 1]/*.Len*/ = 8; + n++; + bl_count[8]++; + } + while (n <= 255) { + static_ltree[n * 2 + 1]/*.Len*/ = 9; + n++; + bl_count[9]++; + } + while (n <= 279) { + static_ltree[n * 2 + 1]/*.Len*/ = 7; + n++; + bl_count[7]++; + } + while (n <= 287) { + static_ltree[n * 2 + 1]/*.Len*/ = 8; + n++; + bl_count[8]++; + } + /* Codes 286 and 287 do not exist, but we must include them in the + * tree construction to get a canonical Huffman tree (longest code + * all ones) + */ + gen_codes(static_ltree, L_CODES + 1, bl_count); + + /* The static distance tree is trivial: */ + for (n = 0; n < D_CODES; n++) { + static_dtree[n * 2 + 1]/*.Len*/ = 5; + static_dtree[n * 2]/*.Code*/ = bi_reverse(n, 5); + } + + // Now data ready and we can init static trees + static_l_desc = new StaticTreeDesc(static_ltree, extra_lbits, LITERALS + 1, L_CODES, MAX_BITS); + static_d_desc = new StaticTreeDesc(static_dtree, extra_dbits, 0, D_CODES, MAX_BITS); + static_bl_desc = new StaticTreeDesc(new Array(0), extra_blbits, 0, BL_CODES, MAX_BL_BITS); + + //static_init_done = true; + } + + + /* =========================================================================== + * Initialize a new block. + */ + function init_block(s) { + let n; /* iterates over tree elements */ + + /* Initialize the trees. */ + for (n = 0; n < L_CODES; n++) { + s.dyn_ltree[n * 2]/*.Freq*/ = 0; + } + for (n = 0; n < D_CODES; n++) { + s.dyn_dtree[n * 2]/*.Freq*/ = 0; + } + for (n = 0; n < BL_CODES; n++) { + s.bl_tree[n * 2]/*.Freq*/ = 0; + } + + s.dyn_ltree[END_BLOCK * 2]/*.Freq*/ = 1; + s.opt_len = s.static_len = 0; + s.last_lit = s.matches = 0; + } + + + /* =========================================================================== + * Flush the bit buffer and align the output on a byte boundary + */ + function bi_windup(s) { + if (s.bi_valid > 8) { + put_short(s, s.bi_buf); + } else if (s.bi_valid > 0) { + //put_byte(s, (Byte)s->bi_buf); + s.pending_buf[s.pending++] = s.bi_buf; + } + s.bi_buf = 0; + s.bi_valid = 0; + } + + /* =========================================================================== + * Copy a stored block, storing first the length and its + * one's complement if requested. + */ + function copy_block(s, buf, len, header) + //DeflateState *s; + //charf *buf; /* the input data */ + //unsigned len; /* its length */ + //int header; /* true if block header must be written */ + { + bi_windup(s); /* align on byte boundary */ + + if (header) { + put_short(s, len); + put_short(s, ~len); + } + // while (len--) { + // put_byte(s, *buf++); + // } + arraySet(s.pending_buf, s.window, buf, len, s.pending); + s.pending += len; + } + + /* =========================================================================== + * Compares to subtrees, using the tree depth as tie breaker when + * the subtrees have equal frequency. This minimizes the worst case length. + */ + function smaller(tree, n, m, depth) { + const _n2 = n * 2; + const _m2 = m * 2; + return tree[_n2]/*.Freq*/ < tree[_m2]/*.Freq*/ || + tree[_n2]/*.Freq*/ === tree[_m2]/*.Freq*/ && depth[n] <= depth[m]; + } + + /* =========================================================================== + * Restore the heap property by moving down the tree starting at node k, + * exchanging a node with the smallest of its two sons if necessary, stopping + * when the heap property is re-established (each father smaller than its + * two sons). + */ + function pqdownheap(s, tree, k) + // deflate_state *s; + // ct_data *tree; /* the tree to restore */ + // int k; /* node to move down */ + { + const v = s.heap[k]; + let j = k << 1; /* left son of k */ + while (j <= s.heap_len) { + /* Set j to the smallest of the two sons: */ + if (j < s.heap_len && + smaller(tree, s.heap[j + 1], s.heap[j], s.depth)) { + j++; + } + /* Exit if v is smaller than both sons */ + if (smaller(tree, v, s.heap[j], s.depth)) { + break; + } + + /* Exchange v with the smallest son */ + s.heap[k] = s.heap[j]; + k = j; + + /* And continue down the tree, setting j to the left son of k */ + j <<= 1; + } + s.heap[k] = v; + } + + + // inlined manually + // var SMALLEST = 1; + + /* =========================================================================== + * Send the block data compressed using the given Huffman trees + */ + function compress_block(s, ltree, dtree) + // deflate_state *s; + // const ct_data *ltree; /* literal tree */ + // const ct_data *dtree; /* distance tree */ + { + let dist; /* distance of matched string */ + let lc; /* match length or unmatched char (if dist == 0) */ + let lx = 0; /* running index in l_buf */ + let code; /* the code to send */ + let extra; /* number of extra bits to send */ + + if (s.last_lit !== 0) { + do { + dist = s.pending_buf[s.d_buf + lx * 2] << 8 | s.pending_buf[s.d_buf + lx * 2 + 1]; + lc = s.pending_buf[s.l_buf + lx]; + lx++; + + if (dist === 0) { + send_code(s, lc, ltree); /* send a literal byte */ + //Tracecv(isgraph(lc), (stderr," '%c' ", lc)); + } else { + /* Here, lc is the match length - MIN_MATCH */ + code = _length_code[lc]; + send_code(s, code + LITERALS + 1, ltree); /* send the length code */ + extra = extra_lbits[code]; + if (extra !== 0) { + lc -= base_length[code]; + send_bits(s, lc, extra); /* send the extra length bits */ + } + dist--; /* dist is now the match distance - 1 */ + code = d_code(dist); + //Assert (code < D_CODES, "bad d_code"); + + send_code(s, code, dtree); /* send the distance code */ + extra = extra_dbits[code]; + if (extra !== 0) { + dist -= base_dist[code]; + send_bits(s, dist, extra); /* send the extra distance bits */ + } + } /* literal or match pair ? */ + + /* Check that the overlay between pending_buf and d_buf+l_buf is ok: */ + //Assert((uInt)(s->pending) < s->lit_bufsize + 2*lx, + // "pendingBuf overflow"); + + } while (lx < s.last_lit); + } + + send_code(s, END_BLOCK, ltree); + } + + + /* =========================================================================== + * Construct one Huffman tree and assigns the code bit strings and lengths. + * Update the total bit length for the current block. + * IN assertion: the field freq is set for all tree elements. + * OUT assertions: the fields len and code are set to the optimal bit length + * and corresponding code. The length opt_len is updated; static_len is + * also updated if stree is not null. The field max_code is set. + */ + function build_tree(s, desc) + // deflate_state *s; + // tree_desc *desc; /* the tree descriptor */ + { + const tree = desc.dyn_tree; + const stree = desc.stat_desc.static_tree; + const has_stree = desc.stat_desc.has_stree; + const elems = desc.stat_desc.elems; + let n, m; /* iterate over heap elements */ + let max_code = -1; /* largest code with non zero frequency */ + let node; /* new node being created */ + + /* Construct the initial heap, with least frequent element in + * heap[SMALLEST]. The sons of heap[n] are heap[2*n] and heap[2*n+1]. + * heap[0] is not used. + */ + s.heap_len = 0; + s.heap_max = HEAP_SIZE; + + for (n = 0; n < elems; n++) { + if (tree[n * 2]/*.Freq*/ !== 0) { + s.heap[++s.heap_len] = max_code = n; + s.depth[n] = 0; + + } else { + tree[n * 2 + 1]/*.Len*/ = 0; + } + } + + /* The pkzip format requires that at least one distance code exists, + * and that at least one bit should be sent even if there is only one + * possible code. So to avoid special checks later on we force at least + * two codes of non zero frequency. + */ + while (s.heap_len < 2) { + node = s.heap[++s.heap_len] = max_code < 2 ? ++max_code : 0; + tree[node * 2]/*.Freq*/ = 1; + s.depth[node] = 0; + s.opt_len--; + + if (has_stree) { + s.static_len -= stree[node * 2 + 1]/*.Len*/; + } + /* node is 0 or 1 so it does not have extra bits */ + } + desc.max_code = max_code; + + /* The elements heap[heap_len/2+1 .. heap_len] are leaves of the tree, + * establish sub-heaps of increasing lengths: + */ + for (n = s.heap_len >> 1/*int /2*/; n >= 1; n--) { + pqdownheap(s, tree, n); + } + + /* Construct the Huffman tree by repeatedly combining the least two + * frequent nodes. + */ + node = elems; /* next internal node of the tree */ + do { + //pqremove(s, tree, n); /* n = node of least frequency */ + /*** pqremove ***/ + n = s.heap[1/*SMALLEST*/]; + s.heap[1/*SMALLEST*/] = s.heap[s.heap_len--]; + pqdownheap(s, tree, 1/*SMALLEST*/); + /***/ + + m = s.heap[1/*SMALLEST*/]; /* m = node of next least frequency */ + + s.heap[--s.heap_max] = n; /* keep the nodes sorted by frequency */ + s.heap[--s.heap_max] = m; + + /* Create a new node father of n and m */ + tree[node * 2]/*.Freq*/ = tree[n * 2]/*.Freq*/ + tree[m * 2]/*.Freq*/; + s.depth[node] = (s.depth[n] >= s.depth[m] ? s.depth[n] : s.depth[m]) + 1; + tree[n * 2 + 1]/*.Dad*/ = tree[m * 2 + 1]/*.Dad*/ = node; + + /* and insert the new node in the heap */ + s.heap[1/*SMALLEST*/] = node++; + pqdownheap(s, tree, 1/*SMALLEST*/); + + } while (s.heap_len >= 2); + + s.heap[--s.heap_max] = s.heap[1/*SMALLEST*/]; + + /* At this point, the fields freq and dad are set. We can now + * generate the bit lengths. + */ + gen_bitlen(s, desc); + + /* The field len is now set, we can generate the bit codes */ + gen_codes(tree, max_code, s.bl_count); + } + + + /* =========================================================================== + * Scan a literal or distance tree to determine the frequencies of the codes + * in the bit length tree. + */ + function scan_tree(s, tree, max_code) + // deflate_state *s; + // ct_data *tree; /* the tree to be scanned */ + // int max_code; /* and its largest code of non zero frequency */ + { + let n; /* iterates over all tree elements */ + let prevlen = -1; /* last emitted length */ + let curlen; /* length of current code */ + + let nextlen = tree[0 * 2 + 1]/*.Len*/; /* length of next code */ + + let count = 0; /* repeat count of the current code */ + let max_count = 7; /* max repeat count */ + let min_count = 4; /* min repeat count */ + + if (nextlen === 0) { + max_count = 138; + min_count = 3; + } + tree[(max_code + 1) * 2 + 1]/*.Len*/ = 0xffff; /* guard */ + + for (n = 0; n <= max_code; n++) { + curlen = nextlen; + nextlen = tree[(n + 1) * 2 + 1]/*.Len*/; + + if (++count < max_count && curlen === nextlen) { + continue; + + } else if (count < min_count) { + s.bl_tree[curlen * 2]/*.Freq*/ += count; + + } else if (curlen !== 0) { + + if (curlen !== prevlen) { + s.bl_tree[curlen * 2]/*.Freq*/++; + } + s.bl_tree[REP_3_6 * 2]/*.Freq*/++; + + } else if (count <= 10) { + s.bl_tree[REPZ_3_10 * 2]/*.Freq*/++; + + } else { + s.bl_tree[REPZ_11_138 * 2]/*.Freq*/++; + } + + count = 0; + prevlen = curlen; + + if (nextlen === 0) { + max_count = 138; + min_count = 3; + + } else if (curlen === nextlen) { + max_count = 6; + min_count = 3; + + } else { + max_count = 7; + min_count = 4; + } + } + } + + + /* =========================================================================== + * Send a literal or distance tree in compressed form, using the codes in + * bl_tree. + */ + function send_tree(s, tree, max_code) + // deflate_state *s; + // ct_data *tree; /* the tree to be scanned */ + // int max_code; /* and its largest code of non zero frequency */ + { + let n; /* iterates over all tree elements */ + let prevlen = -1; /* last emitted length */ + let curlen; /* length of current code */ + + let nextlen = tree[0 * 2 + 1]/*.Len*/; /* length of next code */ + + let count = 0; /* repeat count of the current code */ + let max_count = 7; /* max repeat count */ + let min_count = 4; /* min repeat count */ + + /* tree[max_code+1].Len = -1; */ /* guard already set */ + if (nextlen === 0) { + max_count = 138; + min_count = 3; + } + + for (n = 0; n <= max_code; n++) { + curlen = nextlen; + nextlen = tree[(n + 1) * 2 + 1]/*.Len*/; + + if (++count < max_count && curlen === nextlen) { + continue; + + } else if (count < min_count) { + do { + send_code(s, curlen, s.bl_tree); + } while (--count !== 0); + + } else if (curlen !== 0) { + if (curlen !== prevlen) { + send_code(s, curlen, s.bl_tree); + count--; + } + //Assert(count >= 3 && count <= 6, " 3_6?"); + send_code(s, REP_3_6, s.bl_tree); + send_bits(s, count - 3, 2); + + } else if (count <= 10) { + send_code(s, REPZ_3_10, s.bl_tree); + send_bits(s, count - 3, 3); + + } else { + send_code(s, REPZ_11_138, s.bl_tree); + send_bits(s, count - 11, 7); + } + + count = 0; + prevlen = curlen; + if (nextlen === 0) { + max_count = 138; + min_count = 3; + + } else if (curlen === nextlen) { + max_count = 6; + min_count = 3; + + } else { + max_count = 7; + min_count = 4; + } + } + } + + + /* =========================================================================== + * Construct the Huffman tree for the bit lengths and return the index in + * bl_order of the last bit length code to send. + */ + function build_bl_tree(s) { + let max_blindex; /* index of last bit length code of non zero freq */ + + /* Determine the bit length frequencies for literal and distance trees */ + scan_tree(s, s.dyn_ltree, s.l_desc.max_code); + scan_tree(s, s.dyn_dtree, s.d_desc.max_code); + + /* Build the bit length tree: */ + build_tree(s, s.bl_desc); + /* opt_len now includes the length of the tree representations, except + * the lengths of the bit lengths codes and the 5+5+4 bits for the counts. + */ + + /* Determine the number of bit length codes to send. The pkzip format + * requires that at least 4 bit length codes be sent. (appnote.txt says + * 3 but the actual value used is 4.) + */ + for (max_blindex = BL_CODES - 1; max_blindex >= 3; max_blindex--) { + if (s.bl_tree[bl_order[max_blindex] * 2 + 1]/*.Len*/ !== 0) { + break; + } + } + /* Update opt_len to include the bit length tree and counts */ + s.opt_len += 3 * (max_blindex + 1) + 5 + 5 + 4; + //Tracev((stderr, "\ndyn trees: dyn %ld, stat %ld", + // s->opt_len, s->static_len)); + + return max_blindex; + } + + + /* =========================================================================== + * Send the header for a block using dynamic Huffman trees: the counts, the + * lengths of the bit length codes, the literal tree and the distance tree. + * IN assertion: lcodes >= 257, dcodes >= 1, blcodes >= 4. + */ + function send_all_trees(s, lcodes, dcodes, blcodes) + // deflate_state *s; + // int lcodes, dcodes, blcodes; /* number of codes for each tree */ + { + let rank; /* index in bl_order */ + + //Assert (lcodes >= 257 && dcodes >= 1 && blcodes >= 4, "not enough codes"); + //Assert (lcodes <= L_CODES && dcodes <= D_CODES && blcodes <= BL_CODES, + // "too many codes"); + //Tracev((stderr, "\nbl counts: ")); + send_bits(s, lcodes - 257, 5); /* not +255 as stated in appnote.txt */ + send_bits(s, dcodes - 1, 5); + send_bits(s, blcodes - 4, 4); /* not -3 as stated in appnote.txt */ + for (rank = 0; rank < blcodes; rank++) { + //Tracev((stderr, "\nbl code %2d ", bl_order[rank])); + send_bits(s, s.bl_tree[bl_order[rank] * 2 + 1]/*.Len*/, 3); + } + //Tracev((stderr, "\nbl tree: sent %ld", s->bits_sent)); + + send_tree(s, s.dyn_ltree, lcodes - 1); /* literal tree */ + //Tracev((stderr, "\nlit tree: sent %ld", s->bits_sent)); + + send_tree(s, s.dyn_dtree, dcodes - 1); /* distance tree */ + //Tracev((stderr, "\ndist tree: sent %ld", s->bits_sent)); + } + + + /* =========================================================================== + * Check if the data type is TEXT or BINARY, using the following algorithm: + * - TEXT if the two conditions below are satisfied: + * a) There are no non-portable control characters belonging to the + * "black list" (0..6, 14..25, 28..31). + * b) There is at least one printable character belonging to the + * "white list" (9 {TAB}, 10 {LF}, 13 {CR}, 32..255). + * - BINARY otherwise. + * - The following partially-portable control characters form a + * "gray list" that is ignored in this detection algorithm: + * (7 {BEL}, 8 {BS}, 11 {VT}, 12 {FF}, 26 {SUB}, 27 {ESC}). + * IN assertion: the fields Freq of dyn_ltree are set. + */ + function detect_data_type(s) { + /* black_mask is the bit mask of black-listed bytes + * set bits 0..6, 14..25, and 28..31 + * 0xf3ffc07f = binary 11110011111111111100000001111111 + */ + let black_mask = 0xf3ffc07f; + let n; + + /* Check for non-textual ("black-listed") bytes. */ + for (n = 0; n <= 31; n++, black_mask >>>= 1) { + if (black_mask & 1 && s.dyn_ltree[n * 2]/*.Freq*/ !== 0) { + return Z_BINARY; + } + } + + /* Check for textual ("white-listed") bytes. */ + if (s.dyn_ltree[9 * 2]/*.Freq*/ !== 0 || s.dyn_ltree[10 * 2]/*.Freq*/ !== 0 || + s.dyn_ltree[13 * 2]/*.Freq*/ !== 0) { + return Z_TEXT; + } + for (n = 32; n < LITERALS; n++) { + if (s.dyn_ltree[n * 2]/*.Freq*/ !== 0) { + return Z_TEXT; + } + } + + /* There are no "black-listed" or "white-listed" bytes: + * this stream either is empty or has tolerated ("gray-listed") bytes only. + */ + return Z_BINARY; + } + + + let static_init_done = false; + + /* =========================================================================== + * Initialize the tree data structures for a new zlib stream. + */ + function _tr_init(s) { + + if (!static_init_done) { + tr_static_init(); + static_init_done = true; + } + + s.l_desc = new TreeDesc(s.dyn_ltree, static_l_desc); + s.d_desc = new TreeDesc(s.dyn_dtree, static_d_desc); + s.bl_desc = new TreeDesc(s.bl_tree, static_bl_desc); + + s.bi_buf = 0; + s.bi_valid = 0; + + /* Initialize the first block of the first file: */ + init_block(s); + } + + + /* =========================================================================== + * Send a stored block + */ + function _tr_stored_block(s, buf, stored_len, last) + //DeflateState *s; + //charf *buf; /* input block */ + //ulg stored_len; /* length of input block */ + //int last; /* one if this is the last block for a file */ + { + send_bits(s, (STORED_BLOCK << 1) + (last ? 1 : 0), 3); /* send block type */ + copy_block(s, buf, stored_len, true); /* with header */ + } + + + /* =========================================================================== + * Send one empty static block to give enough lookahead for inflate. + * This takes 10 bits, of which 7 may remain in the bit buffer. + */ + function _tr_align(s) { + send_bits(s, STATIC_TREES << 1, 3); + send_code(s, END_BLOCK, static_ltree); + bi_flush(s); + } + + + /* =========================================================================== + * Determine the best encoding for the current block: dynamic trees, static + * trees or store, and output the encoded block to the zip file. + */ + function _tr_flush_block(s, buf, stored_len, last) + //DeflateState *s; + //charf *buf; /* input block, or NULL if too old */ + //ulg stored_len; /* length of input block */ + //int last; /* one if this is the last block for a file */ + { + let opt_lenb, static_lenb; /* opt_len and static_len in bytes */ + let max_blindex = 0; /* index of last bit length code of non zero freq */ + + /* Build the Huffman trees unless a stored block is forced */ + if (s.level > 0) { + + /* Check if the file is binary or text */ + if (s.strm.data_type === Z_UNKNOWN) { + s.strm.data_type = detect_data_type(s); + } + + /* Construct the literal and distance trees */ + build_tree(s, s.l_desc); + // Tracev((stderr, "\nlit data: dyn %ld, stat %ld", s->opt_len, + // s->static_len)); + + build_tree(s, s.d_desc); + // Tracev((stderr, "\ndist data: dyn %ld, stat %ld", s->opt_len, + // s->static_len)); + /* At this point, opt_len and static_len are the total bit lengths of + * the compressed block data, excluding the tree representations. + */ + + /* Build the bit length tree for the above two trees, and get the index + * in bl_order of the last bit length code to send. + */ + max_blindex = build_bl_tree(s); + + /* Determine the best encoding. Compute the block lengths in bytes. */ + opt_lenb = s.opt_len + 3 + 7 >>> 3; + static_lenb = s.static_len + 3 + 7 >>> 3; + + // Tracev((stderr, "\nopt %lu(%lu) stat %lu(%lu) stored %lu lit %u ", + // opt_lenb, s->opt_len, static_lenb, s->static_len, stored_len, + // s->last_lit)); + + if (static_lenb <= opt_lenb) { + opt_lenb = static_lenb; + } + + } else { + // Assert(buf != (char*)0, "lost buf"); + opt_lenb = static_lenb = stored_len + 5; /* force a stored block */ + } + + if (stored_len + 4 <= opt_lenb && buf !== -1) { + /* 4: two words for the lengths */ + + /* The test buf != NULL is only necessary if LIT_BUFSIZE > WSIZE. + * Otherwise we can't have processed more than WSIZE input bytes since + * the last block flush, because compression would have been + * successful. If LIT_BUFSIZE <= WSIZE, it is never too late to + * transform a block into a stored block. + */ + _tr_stored_block(s, buf, stored_len, last); + + } else if (s.strategy === Z_FIXED || static_lenb === opt_lenb) { + + send_bits(s, (STATIC_TREES << 1) + (last ? 1 : 0), 3); + compress_block(s, static_ltree, static_dtree); + + } else { + send_bits(s, (DYN_TREES << 1) + (last ? 1 : 0), 3); + send_all_trees(s, s.l_desc.max_code + 1, s.d_desc.max_code + 1, max_blindex + 1); + compress_block(s, s.dyn_ltree, s.dyn_dtree); + } + // Assert (s->compressed_len == s->bits_sent, "bad compressed size"); + /* The above check is made mod 2^32, for files larger than 512 MB + * and uLong implemented on 32 bits. + */ + init_block(s); + + if (last) { + bi_windup(s); + } + // Tracev((stderr,"\ncomprlen %lu(%lu) ", s->compressed_len>>3, + // s->compressed_len-7*last)); + } + + /* =========================================================================== + * Save the match info and tally the frequency counts. Return true if + * the current block must be flushed. + */ + function _tr_tally(s, dist, lc) + // deflate_state *s; + // unsigned dist; /* distance of matched string */ + // unsigned lc; /* match length-MIN_MATCH or unmatched char (if dist==0) */ + { + //var out_length, in_length, dcode; + + s.pending_buf[s.d_buf + s.last_lit * 2] = dist >>> 8 & 0xff; + s.pending_buf[s.d_buf + s.last_lit * 2 + 1] = dist & 0xff; + + s.pending_buf[s.l_buf + s.last_lit] = lc & 0xff; + s.last_lit++; + + if (dist === 0) { + /* lc is the unmatched char */ + s.dyn_ltree[lc * 2]/*.Freq*/++; + } else { + s.matches++; + /* Here, lc is the match length - MIN_MATCH */ + dist--; /* dist = match distance - 1 */ + //Assert((ush)dist < (ush)MAX_DIST(s) && + // (ush)lc <= (ush)(MAX_MATCH-MIN_MATCH) && + // (ush)d_code(dist) < (ush)D_CODES, "_tr_tally: bad match"); + + s.dyn_ltree[(_length_code[lc] + LITERALS + 1) * 2]/*.Freq*/++; + s.dyn_dtree[d_code(dist) * 2]/*.Freq*/++; + } + + // (!) This block is disabled in zlib defaults, + // don't enable it for binary compatibility + + //#ifdef TRUNCATE_BLOCK + // /* Try to guess if it is profitable to stop the current block here */ + // if ((s.last_lit & 0x1fff) === 0 && s.level > 2) { + // /* Compute an upper bound for the compressed length */ + // out_length = s.last_lit*8; + // in_length = s.strstart - s.block_start; + // + // for (dcode = 0; dcode < D_CODES; dcode++) { + // out_length += s.dyn_dtree[dcode*2]/*.Freq*/ * (5 + extra_dbits[dcode]); + // } + // out_length >>>= 3; + // //Tracev((stderr,"\nlast_lit %u, in %ld, out ~%ld(%ld%%) ", + // // s->last_lit, in_length, out_length, + // // 100L - out_length*100L/in_length)); + // if (s.matches < (s.last_lit>>1)/*int /2*/ && out_length < (in_length>>1)/*int /2*/) { + // return true; + // } + // } + //#endif + + return s.last_lit === s.lit_bufsize - 1; + /* We avoid equality with lit_bufsize because of wraparound at 64K + * on 16 bit machines and because stored blocks are restricted to + * 64K-1 bytes. + */ + } + + // Note: adler32 takes 12% for level 0 and 2% for level 6. + // It isn't worth it to make additional optimizations as in original. + // Small size is preferable. + + // (C) 1995-2013 Jean-loup Gailly and Mark Adler + // (C) 2014-2017 Vitaly Puzrin and Andrey Tupitsin + // + // This software is provided 'as-is', without any express or implied + // warranty. In no event will the authors be held liable for any damages + // arising from the use of this software. + // + // Permission is granted to anyone to use this software for any purpose, + // including commercial applications, and to alter it and redistribute it + // freely, subject to the following restrictions: + // + // 1. The origin of this software must not be misrepresented; you must not + // claim that you wrote the original software. If you use this software + // in a product, an acknowledgment in the product documentation would be + // appreciated but is not required. + // 2. Altered source versions must be plainly marked as such, and must not be + // misrepresented as being the original software. + // 3. This notice may not be removed or altered from any source distribution. + + function adler32(adler, buf, len, pos) { + let s1 = adler & 0xffff |0, + s2 = adler >>> 16 & 0xffff |0, + n = 0; + + while (len !== 0) { + // Set limit ~ twice less than 5552, to keep + // s2 in 31-bits, because we force signed ints. + // in other case %= will fail. + n = len > 2000 ? 2000 : len; + len -= n; + + do { + s1 = s1 + buf[pos++] |0; + s2 = s2 + s1 |0; + } while (--n); + + s1 %= 65521; + s2 %= 65521; + } + + return s1 | s2 << 16 |0; + } + + // Note: we can't get significant speed boost here. + // So write code to minimize size - no pregenerated tables + // and array tools dependencies. + + // (C) 1995-2013 Jean-loup Gailly and Mark Adler + // (C) 2014-2017 Vitaly Puzrin and Andrey Tupitsin + // + // This software is provided 'as-is', without any express or implied + // warranty. In no event will the authors be held liable for any damages + // arising from the use of this software. + // + // Permission is granted to anyone to use this software for any purpose, + // including commercial applications, and to alter it and redistribute it + // freely, subject to the following restrictions: + // + // 1. The origin of this software must not be misrepresented; you must not + // claim that you wrote the original software. If you use this software + // in a product, an acknowledgment in the product documentation would be + // appreciated but is not required. + // 2. Altered source versions must be plainly marked as such, and must not be + // misrepresented as being the original software. + // 3. This notice may not be removed or altered from any source distribution. + + // Use ordinary array, since untyped makes no boost here + function makeTable() { + let c; + const table = []; + + for (let n = 0; n < 256; n++) { + c = n; + for (let k = 0; k < 8; k++) { + c = c & 1 ? 0xEDB88320 ^ c >>> 1 : c >>> 1; + } + table[n] = c; + } + + return table; + } + + // Create table on load. Just 255 signed longs. Not a problem. + const crcTable = makeTable(); + + + function crc32(crc, buf, len, pos) { + const t = crcTable, + end = pos + len; + + crc ^= -1; + + for (let i = pos; i < end; i++) { + crc = crc >>> 8 ^ t[(crc ^ buf[i]) & 0xFF]; + } + + return crc ^ -1; // >>> 0; + } + + // (C) 1995-2013 Jean-loup Gailly and Mark Adler + // (C) 2014-2017 Vitaly Puzrin and Andrey Tupitsin + // + // This software is provided 'as-is', without any express or implied + // warranty. In no event will the authors be held liable for any damages + // arising from the use of this software. + // + // Permission is granted to anyone to use this software for any purpose, + // including commercial applications, and to alter it and redistribute it + // freely, subject to the following restrictions: + // + // 1. The origin of this software must not be misrepresented; you must not + // claim that you wrote the original software. If you use this software + // in a product, an acknowledgment in the product documentation would be + // appreciated but is not required. + // 2. Altered source versions must be plainly marked as such, and must not be + // misrepresented as being the original software. + // 3. This notice may not be removed or altered from any source distribution. + + var msg = { + 2: "need dictionary", /* Z_NEED_DICT 2 */ + 1: "stream end", /* Z_STREAM_END 1 */ + 0: "", /* Z_OK 0 */ + "-1": "file error", /* Z_ERRNO (-1) */ + "-2": "stream error", /* Z_STREAM_ERROR (-2) */ + "-3": "data error", /* Z_DATA_ERROR (-3) */ + "-4": "insufficient memory", /* Z_MEM_ERROR (-4) */ + "-5": "buffer error", /* Z_BUF_ERROR (-5) */ + "-6": "incompatible version" /* Z_VERSION_ERROR (-6) */ + }; + + /*============================================================================*/ + + + const MAX_MEM_LEVEL = 9; + + + const LENGTH_CODES$1 = 29; + /* number of length codes, not counting the special END_BLOCK code */ + const LITERALS$1 = 256; + /* number of literal bytes 0..255 */ + const L_CODES$1 = LITERALS$1 + 1 + LENGTH_CODES$1; + /* number of Literal or Length codes, including the END_BLOCK code */ + const D_CODES$1 = 30; + /* number of distance codes */ + const BL_CODES$1 = 19; + /* number of codes used to transfer the bit lengths */ + const HEAP_SIZE$1 = 2 * L_CODES$1 + 1; + /* maximum heap size */ + const MAX_BITS$1 = 15; + /* All codes must not exceed MAX_BITS bits */ + + const MIN_MATCH$1 = 3; + const MAX_MATCH$1 = 258; + const MIN_LOOKAHEAD = (MAX_MATCH$1 + MIN_MATCH$1 + 1); + + const PRESET_DICT = 0x20; + + const INIT_STATE = 42; + const EXTRA_STATE = 69; + const NAME_STATE = 73; + const COMMENT_STATE = 91; + const HCRC_STATE = 103; + const BUSY_STATE = 113; + const FINISH_STATE = 666; + + const BS_NEED_MORE = 1; /* block not completed, need more input or more output */ + const BS_BLOCK_DONE = 2; /* block flush performed */ + const BS_FINISH_STARTED = 3; /* finish started, need only more output at next deflate */ + const BS_FINISH_DONE = 4; /* finish done, accept no more input or output */ + + const OS_CODE = 0x03; // Unix :) . Don't detect, use this default. + + function err(strm, errorCode) { + strm.msg = msg[errorCode]; + return errorCode; + } + + function rank(f) { + return ((f) << 1) - ((f) > 4 ? 9 : 0); + } + + function zero$2(buf) { let len = buf.length; while (--len >= 0) { buf[len] = 0; } } + + + /* ========================================================================= + * Flush as much pending output as possible. All deflate() output goes + * through this function so some applications may wish to modify it + * to avoid allocating a large strm->output buffer and copying into it. + * (See also read_buf()). + */ + function flush_pending(strm) { + const s = strm.state; + + //_tr_flush_bits(s); + let len = s.pending; + if (len > strm.avail_out) { + len = strm.avail_out; + } + if (len === 0) { return; } + + arraySet(strm.output, s.pending_buf, s.pending_out, len, strm.next_out); + strm.next_out += len; + s.pending_out += len; + strm.total_out += len; + strm.avail_out -= len; + s.pending -= len; + if (s.pending === 0) { + s.pending_out = 0; + } + } + + + function flush_block_only(s, last) { + _tr_flush_block(s, (s.block_start >= 0 ? s.block_start : -1), s.strstart - s.block_start, last); + s.block_start = s.strstart; + flush_pending(s.strm); + } + + + function put_byte(s, b) { + s.pending_buf[s.pending++] = b; + } + + + /* ========================================================================= + * Put a short in the pending buffer. The 16-bit value is put in MSB order. + * IN assertion: the stream state is correct and there is enough room in + * pending_buf. + */ + function putShortMSB(s, b) { + // put_byte(s, (Byte)(b >> 8)); + // put_byte(s, (Byte)(b & 0xff)); + s.pending_buf[s.pending++] = (b >>> 8) & 0xff; + s.pending_buf[s.pending++] = b & 0xff; + } + + + /* =========================================================================== + * Read a new buffer from the current input stream, update the adler32 + * and total number of bytes read. All deflate() input goes through + * this function so some applications may wish to modify it to avoid + * allocating a large strm->input buffer and copying from it. + * (See also flush_pending()). + */ + function read_buf(strm, buf, start, size) { + let len = strm.avail_in; + + if (len > size) { len = size; } + if (len === 0) { return 0; } + + strm.avail_in -= len; + + // zmemcpy(buf, strm->next_in, len); + arraySet(buf, strm.input, strm.next_in, len, start); + if (strm.state.wrap === 1) { + strm.adler = adler32(strm.adler, buf, len, start); + } + + else if (strm.state.wrap === 2) { + strm.adler = crc32(strm.adler, buf, len, start); + } + + strm.next_in += len; + strm.total_in += len; + + return len; + } + + + /* =========================================================================== + * Set match_start to the longest match starting at the given string and + * return its length. Matches shorter or equal to prev_length are discarded, + * in which case the result is equal to prev_length and match_start is + * garbage. + * IN assertions: cur_match is the head of the hash chain for the current + * string (strstart) and its distance is <= MAX_DIST, and prev_length >= 1 + * OUT assertion: the match length is not greater than s->lookahead. + */ + function longest_match(s, cur_match) { + let chain_length = s.max_chain_length; /* max hash chain length */ + let scan = s.strstart; /* current string */ + let match; /* matched string */ + let len; /* length of current match */ + let best_len = s.prev_length; /* best match length so far */ + let nice_match = s.nice_match; /* stop if match long enough */ + const limit = (s.strstart > (s.w_size - MIN_LOOKAHEAD)) ? + s.strstart - (s.w_size - MIN_LOOKAHEAD) : 0/*NIL*/; + + const _win = s.window; // shortcut + + const wmask = s.w_mask; + const prev = s.prev; + + /* Stop when cur_match becomes <= limit. To simplify the code, + * we prevent matches with the string of window index 0. + */ + + const strend = s.strstart + MAX_MATCH$1; + let scan_end1 = _win[scan + best_len - 1]; + let scan_end = _win[scan + best_len]; + + /* The code is optimized for HASH_BITS >= 8 and MAX_MATCH-2 multiple of 16. + * It is easy to get rid of this optimization if necessary. + */ + // Assert(s->hash_bits >= 8 && MAX_MATCH == 258, "Code too clever"); + + /* Do not waste too much time if we already have a good match: */ + if (s.prev_length >= s.good_match) { + chain_length >>= 2; + } + /* Do not look for matches beyond the end of the input. This is necessary + * to make deflate deterministic. + */ + if (nice_match > s.lookahead) { nice_match = s.lookahead; } + + // Assert((ulg)s->strstart <= s->window_size-MIN_LOOKAHEAD, "need lookahead"); + + do { + // Assert(cur_match < s->strstart, "no future"); + match = cur_match; + + /* Skip to next match if the match length cannot increase + * or if the match length is less than 2. Note that the checks below + * for insufficient lookahead only occur occasionally for performance + * reasons. Therefore uninitialized memory will be accessed, and + * conditional jumps will be made that depend on those values. + * However the length of the match is limited to the lookahead, so + * the output of deflate is not affected by the uninitialized values. + */ + + if (_win[match + best_len] !== scan_end || + _win[match + best_len - 1] !== scan_end1 || + _win[match] !== _win[scan] || + _win[++match] !== _win[scan + 1]) { + continue; + } + + /* The check at best_len-1 can be removed because it will be made + * again later. (This heuristic is not always a win.) + * It is not necessary to compare scan[2] and match[2] since they + * are always equal when the other bytes match, given that + * the hash keys are equal and that HASH_BITS >= 8. + */ + scan += 2; + match++; + // Assert(*scan == *match, "match[2]?"); + + /* We check for insufficient lookahead only every 8th comparison; + * the 256th check will be made at strstart+258. + */ + do { + /*jshint noempty:false*/ + } while (_win[++scan] === _win[++match] && _win[++scan] === _win[++match] && + _win[++scan] === _win[++match] && _win[++scan] === _win[++match] && + _win[++scan] === _win[++match] && _win[++scan] === _win[++match] && + _win[++scan] === _win[++match] && _win[++scan] === _win[++match] && + scan < strend); + + // Assert(scan <= s->window+(unsigned)(s->window_size-1), "wild scan"); + + len = MAX_MATCH$1 - (strend - scan); + scan = strend - MAX_MATCH$1; + + if (len > best_len) { + s.match_start = cur_match; + best_len = len; + if (len >= nice_match) { + break; + } + scan_end1 = _win[scan + best_len - 1]; + scan_end = _win[scan + best_len]; + } + } while ((cur_match = prev[cur_match & wmask]) > limit && --chain_length !== 0); + + if (best_len <= s.lookahead) { + return best_len; + } + return s.lookahead; + } + + + /* =========================================================================== + * Fill the window when the lookahead becomes insufficient. + * Updates strstart and lookahead. + * + * IN assertion: lookahead < MIN_LOOKAHEAD + * OUT assertions: strstart <= window_size-MIN_LOOKAHEAD + * At least one byte has been read, or avail_in == 0; reads are + * performed for at least two bytes (required for the zip translate_eol + * option -- not supported here). + */ + function fill_window(s) { + const _w_size = s.w_size; + let p, n, m, more, str; + + //Assert(s->lookahead < MIN_LOOKAHEAD, "already enough lookahead"); + + do { + more = s.window_size - s.lookahead - s.strstart; + + // JS ints have 32 bit, block below not needed + /* Deal with !@#$% 64K limit: */ + //if (sizeof(int) <= 2) { + // if (more == 0 && s->strstart == 0 && s->lookahead == 0) { + // more = wsize; + // + // } else if (more == (unsigned)(-1)) { + // /* Very unlikely, but possible on 16 bit machine if + // * strstart == 0 && lookahead == 1 (input done a byte at time) + // */ + // more--; + // } + //} + + + /* If the window is almost full and there is insufficient lookahead, + * move the upper half to the lower one to make room in the upper half. + */ + if (s.strstart >= _w_size + (_w_size - MIN_LOOKAHEAD)) { + + arraySet(s.window, s.window, _w_size, _w_size, 0); + s.match_start -= _w_size; + s.strstart -= _w_size; + /* we now have strstart >= MAX_DIST */ + s.block_start -= _w_size; + + /* Slide the hash table (could be avoided with 32 bit values + at the expense of memory usage). We slide even when level == 0 + to keep the hash table consistent if we switch back to level > 0 + later. (Using level 0 permanently is not an optimal usage of + zlib, so we don't care about this pathological case.) + */ + + n = s.hash_size; + p = n; + do { + m = s.head[--p]; + s.head[p] = (m >= _w_size ? m - _w_size : 0); + } while (--n); + + n = _w_size; + p = n; + do { + m = s.prev[--p]; + s.prev[p] = (m >= _w_size ? m - _w_size : 0); + /* If n is not on any hash chain, prev[n] is garbage but + * its value will never be used. + */ + } while (--n); + + more += _w_size; + } + if (s.strm.avail_in === 0) { + break; + } + + /* If there was no sliding: + * strstart <= WSIZE+MAX_DIST-1 && lookahead <= MIN_LOOKAHEAD - 1 && + * more == window_size - lookahead - strstart + * => more >= window_size - (MIN_LOOKAHEAD-1 + WSIZE + MAX_DIST-1) + * => more >= window_size - 2*WSIZE + 2 + * In the BIG_MEM or MMAP case (not yet supported), + * window_size == input_size + MIN_LOOKAHEAD && + * strstart + s->lookahead <= input_size => more >= MIN_LOOKAHEAD. + * Otherwise, window_size == 2*WSIZE so more >= 2. + * If there was sliding, more >= WSIZE. So in all cases, more >= 2. + */ + //Assert(more >= 2, "more < 2"); + n = read_buf(s.strm, s.window, s.strstart + s.lookahead, more); + s.lookahead += n; + + /* Initialize the hash value now that we have some input: */ + if (s.lookahead + s.insert >= MIN_MATCH$1) { + str = s.strstart - s.insert; + s.ins_h = s.window[str]; + + /* UPDATE_HASH(s, s->ins_h, s->window[str + 1]); */ + s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[str + 1]) & s.hash_mask; + //#if MIN_MATCH != 3 + // Call update_hash() MIN_MATCH-3 more times + //#endif + while (s.insert) { + /* UPDATE_HASH(s, s->ins_h, s->window[str + MIN_MATCH-1]); */ + s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[str + MIN_MATCH$1 - 1]) & s.hash_mask; + + s.prev[str & s.w_mask] = s.head[s.ins_h]; + s.head[s.ins_h] = str; + str++; + s.insert--; + if (s.lookahead + s.insert < MIN_MATCH$1) { + break; + } + } + } + /* If the whole input has less than MIN_MATCH bytes, ins_h is garbage, + * but this is not important since only literal bytes will be emitted. + */ + + } while (s.lookahead < MIN_LOOKAHEAD && s.strm.avail_in !== 0); + + /* If the WIN_INIT bytes after the end of the current data have never been + * written, then zero those bytes in order to avoid memory check reports of + * the use of uninitialized (or uninitialised as Julian writes) bytes by + * the longest match routines. Update the high water mark for the next + * time through here. WIN_INIT is set to MAX_MATCH since the longest match + * routines allow scanning to strstart + MAX_MATCH, ignoring lookahead. + */ + // if (s.high_water < s.window_size) { + // var curr = s.strstart + s.lookahead; + // var init = 0; + // + // if (s.high_water < curr) { + // /* Previous high water mark below current data -- zero WIN_INIT + // * bytes or up to end of window, whichever is less. + // */ + // init = s.window_size - curr; + // if (init > WIN_INIT) + // init = WIN_INIT; + // zmemzero(s->window + curr, (unsigned)init); + // s->high_water = curr + init; + // } + // else if (s->high_water < (ulg)curr + WIN_INIT) { + // /* High water mark at or above current data, but below current data + // * plus WIN_INIT -- zero out to current data plus WIN_INIT, or up + // * to end of window, whichever is less. + // */ + // init = (ulg)curr + WIN_INIT - s->high_water; + // if (init > s->window_size - s->high_water) + // init = s->window_size - s->high_water; + // zmemzero(s->window + s->high_water, (unsigned)init); + // s->high_water += init; + // } + // } + // + // Assert((ulg)s->strstart <= s->window_size - MIN_LOOKAHEAD, + // "not enough room for search"); + } + + /* =========================================================================== + * Copy without compression as much as possible from the input stream, return + * the current block state. + * This function does not insert new strings in the dictionary since + * uncompressible data is probably not useful. This function is used + * only for the level=0 compression option. + * NOTE: this function should be optimized to avoid extra copying from + * window to pending_buf. + */ + function deflate_stored(s, flush) { + /* Stored blocks are limited to 0xffff bytes, pending_buf is limited + * to pending_buf_size, and each stored block has a 5 byte header: + */ + let max_block_size = 0xffff; + + if (max_block_size > s.pending_buf_size - 5) { + max_block_size = s.pending_buf_size - 5; + } + + /* Copy as much as possible from input to output: */ + for (; ;) { + /* Fill the window as much as possible: */ + if (s.lookahead <= 1) { + + //Assert(s->strstart < s->w_size+MAX_DIST(s) || + // s->block_start >= (long)s->w_size, "slide too late"); + // if (!(s.strstart < s.w_size + (s.w_size - MIN_LOOKAHEAD) || + // s.block_start >= s.w_size)) { + // throw new Error("slide too late"); + // } + + fill_window(s); + if (s.lookahead === 0 && flush === Z_NO_FLUSH) { + return BS_NEED_MORE; + } + + if (s.lookahead === 0) { + break; + } + /* flush the current block */ + } + //Assert(s->block_start >= 0L, "block gone"); + // if (s.block_start < 0) throw new Error("block gone"); + + s.strstart += s.lookahead; + s.lookahead = 0; + + /* Emit a stored block if pending_buf will be full: */ + const max_start = s.block_start + max_block_size; + + if (s.strstart === 0 || s.strstart >= max_start) { + /* strstart == 0 is possible when wraparound on 16-bit machine */ + s.lookahead = s.strstart - max_start; + s.strstart = max_start; + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + + + } + /* Flush if we may have to slide, otherwise block_start may become + * negative and the data will be gone: + */ + if (s.strstart - s.block_start >= (s.w_size - MIN_LOOKAHEAD)) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + } + + s.insert = 0; + + if (flush === Z_FINISH) { + /*** FLUSH_BLOCK(s, 1); ***/ + flush_block_only(s, true); + if (s.strm.avail_out === 0) { + return BS_FINISH_STARTED; + } + /***/ + return BS_FINISH_DONE; + } + + if (s.strstart > s.block_start) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + + return BS_NEED_MORE; + } + + /* =========================================================================== + * Compress as much as possible from the input stream, return the current + * block state. + * This function does not perform lazy evaluation of matches and inserts + * new strings in the dictionary only for unmatched strings or for short + * matches. It is used only for the fast compression options. + */ + function deflate_fast(s, flush) { + let hash_head; /* head of the hash chain */ + let bflush; /* set if current block must be flushed */ + + for (; ;) { + /* Make sure that we always have enough lookahead, except + * at the end of the input file. We need MAX_MATCH bytes + * for the next match, plus MIN_MATCH bytes to insert the + * string following the next match. + */ + if (s.lookahead < MIN_LOOKAHEAD) { + fill_window(s); + if (s.lookahead < MIN_LOOKAHEAD && flush === Z_NO_FLUSH) { + return BS_NEED_MORE; + } + if (s.lookahead === 0) { + break; /* flush the current block */ + } + } + + /* Insert the string window[strstart .. strstart+2] in the + * dictionary, and set hash_head to the head of the hash chain: + */ + hash_head = 0/*NIL*/; + if (s.lookahead >= MIN_MATCH$1) { + /*** INSERT_STRING(s, s.strstart, hash_head); ***/ + s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + MIN_MATCH$1 - 1]) & s.hash_mask; + hash_head = s.prev[s.strstart & s.w_mask] = s.head[s.ins_h]; + s.head[s.ins_h] = s.strstart; + /***/ + } + + /* Find the longest match, discarding those <= prev_length. + * At this point we have always match_length < MIN_MATCH + */ + if (hash_head !== 0/*NIL*/ && ((s.strstart - hash_head) <= (s.w_size - MIN_LOOKAHEAD))) { + /* To simplify the code, we prevent matches with the string + * of window index 0 (in particular we have to avoid a match + * of the string with itself at the start of the input file). + */ + s.match_length = longest_match(s, hash_head); + /* longest_match() sets match_start */ + } + if (s.match_length >= MIN_MATCH$1) { + // check_match(s, s.strstart, s.match_start, s.match_length); // for debug only + + /*** _tr_tally_dist(s, s.strstart - s.match_start, + s.match_length - MIN_MATCH, bflush); ***/ + bflush = _tr_tally(s, s.strstart - s.match_start, s.match_length - MIN_MATCH$1); + + s.lookahead -= s.match_length; + + /* Insert new strings in the hash table only if the match length + * is not too large. This saves time but degrades compression. + */ + if (s.match_length <= s.max_lazy_match/*max_insert_length*/ && s.lookahead >= MIN_MATCH$1) { + s.match_length--; /* string at strstart already in table */ + do { + s.strstart++; + /*** INSERT_STRING(s, s.strstart, hash_head); ***/ + s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + MIN_MATCH$1 - 1]) & s.hash_mask; + hash_head = s.prev[s.strstart & s.w_mask] = s.head[s.ins_h]; + s.head[s.ins_h] = s.strstart; + /***/ + /* strstart never exceeds WSIZE-MAX_MATCH, so there are + * always MIN_MATCH bytes ahead. + */ + } while (--s.match_length !== 0); + s.strstart++; + } else { + s.strstart += s.match_length; + s.match_length = 0; + s.ins_h = s.window[s.strstart]; + /* UPDATE_HASH(s, s.ins_h, s.window[s.strstart+1]); */ + s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + 1]) & s.hash_mask; + + //#if MIN_MATCH != 3 + // Call UPDATE_HASH() MIN_MATCH-3 more times + //#endif + /* If lookahead < MIN_MATCH, ins_h is garbage, but it does not + * matter since it will be recomputed at next deflate call. + */ + } + } else { + /* No match, output a literal byte */ + //Tracevv((stderr,"%c", s.window[s.strstart])); + /*** _tr_tally_lit(s, s.window[s.strstart], bflush); ***/ + bflush = _tr_tally(s, 0, s.window[s.strstart]); + + s.lookahead--; + s.strstart++; + } + if (bflush) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + } + s.insert = ((s.strstart < (MIN_MATCH$1 - 1)) ? s.strstart : MIN_MATCH$1 - 1); + if (flush === Z_FINISH) { + /*** FLUSH_BLOCK(s, 1); ***/ + flush_block_only(s, true); + if (s.strm.avail_out === 0) { + return BS_FINISH_STARTED; + } + /***/ + return BS_FINISH_DONE; + } + if (s.last_lit) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + return BS_BLOCK_DONE; + } + + /* =========================================================================== + * Same as above, but achieves better compression. We use a lazy + * evaluation for matches: a match is finally adopted only if there is + * no better match at the next window position. + */ + function deflate_slow(s, flush) { + let hash_head; /* head of hash chain */ + let bflush; /* set if current block must be flushed */ + + let max_insert; + + /* Process the input block. */ + for (; ;) { + /* Make sure that we always have enough lookahead, except + * at the end of the input file. We need MAX_MATCH bytes + * for the next match, plus MIN_MATCH bytes to insert the + * string following the next match. + */ + if (s.lookahead < MIN_LOOKAHEAD) { + fill_window(s); + if (s.lookahead < MIN_LOOKAHEAD && flush === Z_NO_FLUSH) { + return BS_NEED_MORE; + } + if (s.lookahead === 0) { break; } /* flush the current block */ + } + + /* Insert the string window[strstart .. strstart+2] in the + * dictionary, and set hash_head to the head of the hash chain: + */ + hash_head = 0/*NIL*/; + if (s.lookahead >= MIN_MATCH$1) { + /*** INSERT_STRING(s, s.strstart, hash_head); ***/ + s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + MIN_MATCH$1 - 1]) & s.hash_mask; + hash_head = s.prev[s.strstart & s.w_mask] = s.head[s.ins_h]; + s.head[s.ins_h] = s.strstart; + /***/ + } + + /* Find the longest match, discarding those <= prev_length. + */ + s.prev_length = s.match_length; + s.prev_match = s.match_start; + s.match_length = MIN_MATCH$1 - 1; + + if (hash_head !== 0/*NIL*/ && s.prev_length < s.max_lazy_match && + s.strstart - hash_head <= (s.w_size - MIN_LOOKAHEAD)/*MAX_DIST(s)*/) { + /* To simplify the code, we prevent matches with the string + * of window index 0 (in particular we have to avoid a match + * of the string with itself at the start of the input file). + */ + s.match_length = longest_match(s, hash_head); + /* longest_match() sets match_start */ + + if (s.match_length <= 5 && + (s.strategy === Z_FILTERED || (s.match_length === MIN_MATCH$1 && s.strstart - s.match_start > 4096/*TOO_FAR*/))) { + + /* If prev_match is also MIN_MATCH, match_start is garbage + * but we will ignore the current match anyway. + */ + s.match_length = MIN_MATCH$1 - 1; + } + } + /* If there was a match at the previous step and the current + * match is not better, output the previous match: + */ + if (s.prev_length >= MIN_MATCH$1 && s.match_length <= s.prev_length) { + max_insert = s.strstart + s.lookahead - MIN_MATCH$1; + /* Do not insert strings in hash table beyond this. */ + + //check_match(s, s.strstart-1, s.prev_match, s.prev_length); + + /***_tr_tally_dist(s, s.strstart - 1 - s.prev_match, + s.prev_length - MIN_MATCH, bflush);***/ + bflush = _tr_tally(s, s.strstart - 1 - s.prev_match, s.prev_length - MIN_MATCH$1); + /* Insert in hash table all strings up to the end of the match. + * strstart-1 and strstart are already inserted. If there is not + * enough lookahead, the last two strings are not inserted in + * the hash table. + */ + s.lookahead -= s.prev_length - 1; + s.prev_length -= 2; + do { + if (++s.strstart <= max_insert) { + /*** INSERT_STRING(s, s.strstart, hash_head); ***/ + s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + MIN_MATCH$1 - 1]) & s.hash_mask; + hash_head = s.prev[s.strstart & s.w_mask] = s.head[s.ins_h]; + s.head[s.ins_h] = s.strstart; + /***/ + } + } while (--s.prev_length !== 0); + s.match_available = 0; + s.match_length = MIN_MATCH$1 - 1; + s.strstart++; + + if (bflush) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + + } else if (s.match_available) { + /* If there was no match at the previous position, output a + * single literal. If there was a match but the current match + * is longer, truncate the previous match to a single literal. + */ + //Tracevv((stderr,"%c", s->window[s->strstart-1])); + /*** _tr_tally_lit(s, s.window[s.strstart-1], bflush); ***/ + bflush = _tr_tally(s, 0, s.window[s.strstart - 1]); + + if (bflush) { + /*** FLUSH_BLOCK_ONLY(s, 0) ***/ + flush_block_only(s, false); + /***/ + } + s.strstart++; + s.lookahead--; + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + } else { + /* There is no previous match to compare with, wait for + * the next step to decide. + */ + s.match_available = 1; + s.strstart++; + s.lookahead--; + } + } + //Assert (flush != Z_NO_FLUSH, "no flush?"); + if (s.match_available) { + //Tracevv((stderr,"%c", s->window[s->strstart-1])); + /*** _tr_tally_lit(s, s.window[s.strstart-1], bflush); ***/ + bflush = _tr_tally(s, 0, s.window[s.strstart - 1]); + + s.match_available = 0; + } + s.insert = s.strstart < MIN_MATCH$1 - 1 ? s.strstart : MIN_MATCH$1 - 1; + if (flush === Z_FINISH) { + /*** FLUSH_BLOCK(s, 1); ***/ + flush_block_only(s, true); + if (s.strm.avail_out === 0) { + return BS_FINISH_STARTED; + } + /***/ + return BS_FINISH_DONE; + } + if (s.last_lit) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + + return BS_BLOCK_DONE; + } + + + /* =========================================================================== + * For Z_RLE, simply look for runs of bytes, generate matches only of distance + * one. Do not maintain a hash table. (It will be regenerated if this run of + * deflate switches away from Z_RLE.) + */ + function deflate_rle(s, flush) { + let bflush; /* set if current block must be flushed */ + let prev; /* byte at distance one to match */ + let scan, strend; /* scan goes up to strend for length of run */ + + const _win = s.window; + + for (; ;) { + /* Make sure that we always have enough lookahead, except + * at the end of the input file. We need MAX_MATCH bytes + * for the longest run, plus one for the unrolled loop. + */ + if (s.lookahead <= MAX_MATCH$1) { + fill_window(s); + if (s.lookahead <= MAX_MATCH$1 && flush === Z_NO_FLUSH) { + return BS_NEED_MORE; + } + if (s.lookahead === 0) { break; } /* flush the current block */ + } + + /* See how many times the previous byte repeats */ + s.match_length = 0; + if (s.lookahead >= MIN_MATCH$1 && s.strstart > 0) { + scan = s.strstart - 1; + prev = _win[scan]; + if (prev === _win[++scan] && prev === _win[++scan] && prev === _win[++scan]) { + strend = s.strstart + MAX_MATCH$1; + do { + /*jshint noempty:false*/ + } while (prev === _win[++scan] && prev === _win[++scan] && + prev === _win[++scan] && prev === _win[++scan] && + prev === _win[++scan] && prev === _win[++scan] && + prev === _win[++scan] && prev === _win[++scan] && + scan < strend); + s.match_length = MAX_MATCH$1 - (strend - scan); + if (s.match_length > s.lookahead) { + s.match_length = s.lookahead; + } + } + //Assert(scan <= s->window+(uInt)(s->window_size-1), "wild scan"); + } + + /* Emit match if have run of MIN_MATCH or longer, else emit literal */ + if (s.match_length >= MIN_MATCH$1) { + //check_match(s, s.strstart, s.strstart - 1, s.match_length); + + /*** _tr_tally_dist(s, 1, s.match_length - MIN_MATCH, bflush); ***/ + bflush = _tr_tally(s, 1, s.match_length - MIN_MATCH$1); + + s.lookahead -= s.match_length; + s.strstart += s.match_length; + s.match_length = 0; + } else { + /* No match, output a literal byte */ + //Tracevv((stderr,"%c", s->window[s->strstart])); + /*** _tr_tally_lit(s, s.window[s.strstart], bflush); ***/ + bflush = _tr_tally(s, 0, s.window[s.strstart]); + + s.lookahead--; + s.strstart++; + } + if (bflush) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + } + s.insert = 0; + if (flush === Z_FINISH) { + /*** FLUSH_BLOCK(s, 1); ***/ + flush_block_only(s, true); + if (s.strm.avail_out === 0) { + return BS_FINISH_STARTED; + } + /***/ + return BS_FINISH_DONE; + } + if (s.last_lit) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + return BS_BLOCK_DONE; + } + + /* =========================================================================== + * For Z_HUFFMAN_ONLY, do not look for matches. Do not maintain a hash table. + * (It will be regenerated if this run of deflate switches away from Huffman.) + */ + function deflate_huff(s, flush) { + let bflush; /* set if current block must be flushed */ + + for (; ;) { + /* Make sure that we have a literal to write. */ + if (s.lookahead === 0) { + fill_window(s); + if (s.lookahead === 0) { + if (flush === Z_NO_FLUSH) { + return BS_NEED_MORE; + } + break; /* flush the current block */ + } + } + + /* Output a literal byte */ + s.match_length = 0; + //Tracevv((stderr,"%c", s->window[s->strstart])); + /*** _tr_tally_lit(s, s.window[s.strstart], bflush); ***/ + bflush = _tr_tally(s, 0, s.window[s.strstart]); + s.lookahead--; + s.strstart++; + if (bflush) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + } + s.insert = 0; + if (flush === Z_FINISH) { + /*** FLUSH_BLOCK(s, 1); ***/ + flush_block_only(s, true); + if (s.strm.avail_out === 0) { + return BS_FINISH_STARTED; + } + /***/ + return BS_FINISH_DONE; + } + if (s.last_lit) { + /*** FLUSH_BLOCK(s, 0); ***/ + flush_block_only(s, false); + if (s.strm.avail_out === 0) { + return BS_NEED_MORE; + } + /***/ + } + return BS_BLOCK_DONE; + } + + /* Values for max_lazy_match, good_match and max_chain_length, depending on + * the desired pack level (0..9). The values given below have been tuned to + * exclude worst case performance for pathological files. Better values may be + * found for specific files. + */ + class Config { + constructor(good_length, max_lazy, nice_length, max_chain, func) { + this.good_length = good_length; + this.max_lazy = max_lazy; + this.nice_length = nice_length; + this.max_chain = max_chain; + this.func = func; + } + } + const configuration_table = [ + /* good lazy nice chain */ + new Config(0, 0, 0, 0, deflate_stored), /* 0 store only */ + new Config(4, 4, 8, 4, deflate_fast), /* 1 max speed, no lazy matches */ + new Config(4, 5, 16, 8, deflate_fast), /* 2 */ + new Config(4, 6, 32, 32, deflate_fast), /* 3 */ + + new Config(4, 4, 16, 16, deflate_slow), /* 4 lazy matches */ + new Config(8, 16, 32, 32, deflate_slow), /* 5 */ + new Config(8, 16, 128, 128, deflate_slow), /* 6 */ + new Config(8, 32, 128, 256, deflate_slow), /* 7 */ + new Config(32, 128, 258, 1024, deflate_slow), /* 8 */ + new Config(32, 258, 258, 4096, deflate_slow) /* 9 max compression */ + ]; + + + /* =========================================================================== + * Initialize the "longest match" routines for a new zlib stream + */ + function lm_init(s) { + s.window_size = 2 * s.w_size; + + /*** CLEAR_HASH(s); ***/ + zero$2(s.head); // Fill with NIL (= 0); + + /* Set the default configuration parameters: + */ + s.max_lazy_match = configuration_table[s.level].max_lazy; + s.good_match = configuration_table[s.level].good_length; + s.nice_match = configuration_table[s.level].nice_length; + s.max_chain_length = configuration_table[s.level].max_chain; + + s.strstart = 0; + s.block_start = 0; + s.lookahead = 0; + s.insert = 0; + s.match_length = s.prev_length = MIN_MATCH$1 - 1; + s.match_available = 0; + s.ins_h = 0; + } + + class DeflateState { + constructor() { + this.strm = null; /* pointer back to this zlib stream */ + this.status = 0; /* as the name implies */ + this.pending_buf = null; /* output still pending */ + this.pending_buf_size = 0; /* size of pending_buf */ + this.pending_out = 0; /* next pending byte to output to the stream */ + this.pending = 0; /* nb of bytes in the pending buffer */ + this.wrap = 0; /* bit 0 true for zlib, bit 1 true for gzip */ + this.gzhead = null; /* gzip header information to write */ + this.gzindex = 0; /* where in extra, name, or comment */ + this.method = Z_DEFLATED; /* can only be DEFLATED */ + this.last_flush = -1; /* value of flush param for previous deflate call */ + + this.w_size = 0; /* LZ77 window size (32K by default) */ + this.w_bits = 0; /* log2(w_size) (8..16) */ + this.w_mask = 0; /* w_size - 1 */ + + this.window = null; + /* Sliding window. Input bytes are read into the second half of the window, + * and move to the first half later to keep a dictionary of at least wSize + * bytes. With this organization, matches are limited to a distance of + * wSize-MAX_MATCH bytes, but this ensures that IO is always + * performed with a length multiple of the block size. + */ + + this.window_size = 0; + /* Actual size of window: 2*wSize, except when the user input buffer + * is directly used as sliding window. + */ + + this.prev = null; + /* Link to older string with same hash index. To limit the size of this + * array to 64K, this link is maintained only for the last 32K strings. + * An index in this array is thus a window index modulo 32K. + */ + + this.head = null; /* Heads of the hash chains or NIL. */ + + this.ins_h = 0; /* hash index of string to be inserted */ + this.hash_size = 0; /* number of elements in hash table */ + this.hash_bits = 0; /* log2(hash_size) */ + this.hash_mask = 0; /* hash_size-1 */ + + this.hash_shift = 0; + /* Number of bits by which ins_h must be shifted at each input + * step. It must be such that after MIN_MATCH steps, the oldest + * byte no longer takes part in the hash key, that is: + * hash_shift * MIN_MATCH >= hash_bits + */ + + this.block_start = 0; + /* Window position at the beginning of the current output block. Gets + * negative when the window is moved backwards. + */ + + this.match_length = 0; /* length of best match */ + this.prev_match = 0; /* previous match */ + this.match_available = 0; /* set if previous match exists */ + this.strstart = 0; /* start of string to insert */ + this.match_start = 0; /* start of matching string */ + this.lookahead = 0; /* number of valid bytes ahead in window */ + + this.prev_length = 0; + /* Length of the best match at previous step. Matches not greater than this + * are discarded. This is used in the lazy match evaluation. + */ + + this.max_chain_length = 0; + /* To speed up deflation, hash chains are never searched beyond this + * length. A higher limit improves compression ratio but degrades the + * speed. + */ + + this.max_lazy_match = 0; + /* Attempt to find a better match only when the current match is strictly + * smaller than this value. This mechanism is used only for compression + * levels >= 4. + */ + // That's alias to max_lazy_match, don't use directly + //this.max_insert_length = 0; + /* Insert new strings in the hash table only if the match length is not + * greater than this length. This saves time but degrades compression. + * max_insert_length is used only for compression levels <= 3. + */ + + this.level = 0; /* compression level (1..9) */ + this.strategy = 0; /* favor or force Huffman coding*/ + + this.good_match = 0; + /* Use a faster search when the previous match is longer than this */ + + this.nice_match = 0; /* Stop searching when current match exceeds this */ + + /* used by trees.c: */ + + /* Didn't use ct_data typedef below to suppress compiler warning */ + + // struct ct_data_s dyn_ltree[HEAP_SIZE]; /* literal and length tree */ + // struct ct_data_s dyn_dtree[2*D_CODES+1]; /* distance tree */ + // struct ct_data_s bl_tree[2*BL_CODES+1]; /* Huffman tree for bit lengths */ + + // Use flat array of DOUBLE size, with interleaved fata, + // because JS does not support effective + this.dyn_ltree = new Buf16(HEAP_SIZE$1 * 2); + this.dyn_dtree = new Buf16((2 * D_CODES$1 + 1) * 2); + this.bl_tree = new Buf16((2 * BL_CODES$1 + 1) * 2); + zero$2(this.dyn_ltree); + zero$2(this.dyn_dtree); + zero$2(this.bl_tree); + + this.l_desc = null; /* desc. for literal tree */ + this.d_desc = null; /* desc. for distance tree */ + this.bl_desc = null; /* desc. for bit length tree */ + + //ush bl_count[MAX_BITS+1]; + this.bl_count = new Buf16(MAX_BITS$1 + 1); + /* number of codes at each bit length for an optimal tree */ + + //int heap[2*L_CODES+1]; /* heap used to build the Huffman trees */ + this.heap = new Buf16(2 * L_CODES$1 + 1); /* heap used to build the Huffman trees */ + zero$2(this.heap); + + this.heap_len = 0; /* number of elements in the heap */ + this.heap_max = 0; /* element of largest frequency */ + /* The sons of heap[n] are heap[2*n] and heap[2*n+1]. heap[0] is not used. + * The same heap array is used to build all trees. + */ + + this.depth = new Buf16(2 * L_CODES$1 + 1); //uch depth[2*L_CODES+1]; + zero$2(this.depth); + /* Depth of each subtree used as tie breaker for trees of equal frequency + */ + + this.l_buf = 0; /* buffer index for literals or lengths */ + + this.lit_bufsize = 0; + /* Size of match buffer for literals/lengths. There are 4 reasons for + * limiting lit_bufsize to 64K: + * - frequencies can be kept in 16 bit counters + * - if compression is not successful for the first block, all input + * data is still in the window so we can still emit a stored block even + * when input comes from standard input. (This can also be done for + * all blocks if lit_bufsize is not greater than 32K.) + * - if compression is not successful for a file smaller than 64K, we can + * even emit a stored file instead of a stored block (saving 5 bytes). + * This is applicable only for zip (not gzip or zlib). + * - creating new Huffman trees less frequently may not provide fast + * adaptation to changes in the input data statistics. (Take for + * example a binary file with poorly compressible code followed by + * a highly compressible string table.) Smaller buffer sizes give + * fast adaptation but have of course the overhead of transmitting + * trees more frequently. + * - I can't count above 4 + */ + + this.last_lit = 0; /* running index in l_buf */ + + this.d_buf = 0; + /* Buffer index for distances. To simplify the code, d_buf and l_buf have + * the same number of elements. To use different lengths, an extra flag + * array would be necessary. + */ + + this.opt_len = 0; /* bit length of current block with optimal trees */ + this.static_len = 0; /* bit length of current block with static trees */ + this.matches = 0; /* number of string matches in current block */ + this.insert = 0; /* bytes at end of window left to insert */ + + + this.bi_buf = 0; + /* Output buffer. bits are inserted starting at the bottom (least + * significant bits). + */ + this.bi_valid = 0; + /* Number of valid bits in bi_buf. All bits above the last valid bit + * are always zero. + */ + + // Used for window memory init. We safely ignore it for JS. That makes + // sense only for pointers and memory check tools. + //this.high_water = 0; + /* High water mark offset in window for initialized bytes -- bytes above + * this are set to zero in order to avoid memory check warnings when + * longest match routines access bytes past the input. This is then + * updated to the new high water mark. + */ + } + } + + function deflateResetKeep(strm) { + let s; + + if (!strm || !strm.state) { + return err(strm, Z_STREAM_ERROR); + } + + strm.total_in = strm.total_out = 0; + strm.data_type = Z_UNKNOWN; + + s = strm.state; + s.pending = 0; + s.pending_out = 0; + + if (s.wrap < 0) { + s.wrap = -s.wrap; + /* was made negative by deflate(..., Z_FINISH); */ + } + s.status = (s.wrap ? INIT_STATE : BUSY_STATE); + strm.adler = (s.wrap === 2) ? + 0 // crc32(0, Z_NULL, 0) + : + 1; // adler32(0, Z_NULL, 0) + s.last_flush = Z_NO_FLUSH; + _tr_init(s); + return Z_OK; + } + + + function deflateReset(strm) { + const ret = deflateResetKeep(strm); + if (ret === Z_OK) { + lm_init(strm.state); + } + return ret; + } + + + function deflateSetHeader(strm, head) { + if (!strm || !strm.state) { return Z_STREAM_ERROR; } + if (strm.state.wrap !== 2) { return Z_STREAM_ERROR; } + strm.state.gzhead = head; + return Z_OK; + } + + + function deflateInit2(strm, level, method, windowBits, memLevel, strategy) { + if (!strm) { // === Z_NULL + return Z_STREAM_ERROR; + } + let wrap = 1; + + if (level === Z_DEFAULT_COMPRESSION) { + level = 6; + } + + if (windowBits < 0) { /* suppress zlib wrapper */ + wrap = 0; + windowBits = -windowBits; + } + + else if (windowBits > 15) { + wrap = 2; /* write gzip wrapper instead */ + windowBits -= 16; + } + + + if (memLevel < 1 || memLevel > MAX_MEM_LEVEL || method !== Z_DEFLATED || + windowBits < 8 || windowBits > 15 || level < 0 || level > 9 || + strategy < 0 || strategy > Z_FIXED) { + return err(strm, Z_STREAM_ERROR); + } + + + if (windowBits === 8) { + windowBits = 9; + } + /* until 256-byte window bug fixed */ + + const s = new DeflateState(); + + strm.state = s; + s.strm = strm; + + s.wrap = wrap; + s.gzhead = null; + s.w_bits = windowBits; + s.w_size = 1 << s.w_bits; + s.w_mask = s.w_size - 1; + + s.hash_bits = memLevel + 7; + s.hash_size = 1 << s.hash_bits; + s.hash_mask = s.hash_size - 1; + s.hash_shift = ~~((s.hash_bits + MIN_MATCH$1 - 1) / MIN_MATCH$1); + s.window = new Buf8(s.w_size * 2); + s.head = new Buf16(s.hash_size); + s.prev = new Buf16(s.w_size); + + // Don't need mem init magic for JS. + //s.high_water = 0; /* nothing written to s->window yet */ + + s.lit_bufsize = 1 << (memLevel + 6); /* 16K elements by default */ + + s.pending_buf_size = s.lit_bufsize * 4; + + //overlay = (ushf *) ZALLOC(strm, s->lit_bufsize, sizeof(ush)+2); + //s->pending_buf = (uchf *) overlay; + s.pending_buf = new Buf8(s.pending_buf_size); + + // It is offset from `s.pending_buf` (size is `s.lit_bufsize * 2`) + //s->d_buf = overlay + s->lit_bufsize/sizeof(ush); + s.d_buf = 1 * s.lit_bufsize; + + //s->l_buf = s->pending_buf + (1+sizeof(ush))*s->lit_bufsize; + s.l_buf = (1 + 2) * s.lit_bufsize; + + s.level = level; + s.strategy = strategy; + s.method = method; + + return deflateReset(strm); + } + + + function deflate(strm, flush) { + let old_flush, s; + let beg, val; // for gzip header write only + + if (!strm || !strm.state || + flush > Z_BLOCK || flush < 0) { + return strm ? err(strm, Z_STREAM_ERROR) : Z_STREAM_ERROR; + } + + s = strm.state; + + if (!strm.output || + (!strm.input && strm.avail_in !== 0) || + (s.status === FINISH_STATE && flush !== Z_FINISH)) { + return err(strm, (strm.avail_out === 0) ? Z_BUF_ERROR : Z_STREAM_ERROR); + } + + s.strm = strm; /* just in case */ + old_flush = s.last_flush; + s.last_flush = flush; + + /* Write the header */ + if (s.status === INIT_STATE) { + + if (s.wrap === 2) { // GZIP header + strm.adler = 0; //crc32(0L, Z_NULL, 0); + put_byte(s, 31); + put_byte(s, 139); + put_byte(s, 8); + if (!s.gzhead) { // s->gzhead == Z_NULL + put_byte(s, 0); + put_byte(s, 0); + put_byte(s, 0); + put_byte(s, 0); + put_byte(s, 0); + put_byte(s, s.level === 9 ? 2 : + (s.strategy >= Z_HUFFMAN_ONLY || s.level < 2 ? + 4 : 0)); + put_byte(s, OS_CODE); + s.status = BUSY_STATE; + } + else { + put_byte(s, (s.gzhead.text ? 1 : 0) + + (s.gzhead.hcrc ? 2 : 0) + + (!s.gzhead.extra ? 0 : 4) + + (!s.gzhead.name ? 0 : 8) + + (!s.gzhead.comment ? 0 : 16) + ); + put_byte(s, s.gzhead.time & 0xff); + put_byte(s, (s.gzhead.time >> 8) & 0xff); + put_byte(s, (s.gzhead.time >> 16) & 0xff); + put_byte(s, (s.gzhead.time >> 24) & 0xff); + put_byte(s, s.level === 9 ? 2 : + (s.strategy >= Z_HUFFMAN_ONLY || s.level < 2 ? + 4 : 0)); + put_byte(s, s.gzhead.os & 0xff); + if (s.gzhead.extra && s.gzhead.extra.length) { + put_byte(s, s.gzhead.extra.length & 0xff); + put_byte(s, (s.gzhead.extra.length >> 8) & 0xff); + } + if (s.gzhead.hcrc) { + strm.adler = crc32(strm.adler, s.pending_buf, s.pending, 0); + } + s.gzindex = 0; + s.status = EXTRA_STATE; + } + } + else // DEFLATE header + { + let header = (Z_DEFLATED + ((s.w_bits - 8) << 4)) << 8; + let level_flags = -1; + + if (s.strategy >= Z_HUFFMAN_ONLY || s.level < 2) { + level_flags = 0; + } else if (s.level < 6) { + level_flags = 1; + } else if (s.level === 6) { + level_flags = 2; + } else { + level_flags = 3; + } + header |= (level_flags << 6); + if (s.strstart !== 0) { header |= PRESET_DICT; } + header += 31 - (header % 31); + + s.status = BUSY_STATE; + putShortMSB(s, header); + + /* Save the adler32 of the preset dictionary: */ + if (s.strstart !== 0) { + putShortMSB(s, strm.adler >>> 16); + putShortMSB(s, strm.adler & 0xffff); + } + strm.adler = 1; // adler32(0L, Z_NULL, 0); + } + } + + //#ifdef GZIP + if (s.status === EXTRA_STATE) { + if (s.gzhead.extra/* != Z_NULL*/) { + beg = s.pending; /* start of bytes to update crc */ + + while (s.gzindex < (s.gzhead.extra.length & 0xffff)) { + if (s.pending === s.pending_buf_size) { + if (s.gzhead.hcrc && s.pending > beg) { + strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg); + } + flush_pending(strm); + beg = s.pending; + if (s.pending === s.pending_buf_size) { + break; + } + } + put_byte(s, s.gzhead.extra[s.gzindex] & 0xff); + s.gzindex++; + } + if (s.gzhead.hcrc && s.pending > beg) { + strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg); + } + if (s.gzindex === s.gzhead.extra.length) { + s.gzindex = 0; + s.status = NAME_STATE; + } + } + else { + s.status = NAME_STATE; + } + } + if (s.status === NAME_STATE) { + if (s.gzhead.name/* != Z_NULL*/) { + beg = s.pending; /* start of bytes to update crc */ + //int val; + + do { + if (s.pending === s.pending_buf_size) { + if (s.gzhead.hcrc && s.pending > beg) { + strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg); + } + flush_pending(strm); + beg = s.pending; + if (s.pending === s.pending_buf_size) { + val = 1; + break; + } + } + // JS specific: little magic to add zero terminator to end of string + if (s.gzindex < s.gzhead.name.length) { + val = s.gzhead.name.charCodeAt(s.gzindex++) & 0xff; + } else { + val = 0; + } + put_byte(s, val); + } while (val !== 0); + + if (s.gzhead.hcrc && s.pending > beg) { + strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg); + } + if (val === 0) { + s.gzindex = 0; + s.status = COMMENT_STATE; + } + } + else { + s.status = COMMENT_STATE; + } + } + if (s.status === COMMENT_STATE) { + if (s.gzhead.comment/* != Z_NULL*/) { + beg = s.pending; /* start of bytes to update crc */ + //int val; + + do { + if (s.pending === s.pending_buf_size) { + if (s.gzhead.hcrc && s.pending > beg) { + strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg); + } + flush_pending(strm); + beg = s.pending; + if (s.pending === s.pending_buf_size) { + val = 1; + break; + } + } + // JS specific: little magic to add zero terminator to end of string + if (s.gzindex < s.gzhead.comment.length) { + val = s.gzhead.comment.charCodeAt(s.gzindex++) & 0xff; + } else { + val = 0; + } + put_byte(s, val); + } while (val !== 0); + + if (s.gzhead.hcrc && s.pending > beg) { + strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg); + } + if (val === 0) { + s.status = HCRC_STATE; + } + } + else { + s.status = HCRC_STATE; + } + } + if (s.status === HCRC_STATE) { + if (s.gzhead.hcrc) { + if (s.pending + 2 > s.pending_buf_size) { + flush_pending(strm); + } + if (s.pending + 2 <= s.pending_buf_size) { + put_byte(s, strm.adler & 0xff); + put_byte(s, (strm.adler >> 8) & 0xff); + strm.adler = 0; //crc32(0L, Z_NULL, 0); + s.status = BUSY_STATE; + } + } + else { + s.status = BUSY_STATE; + } + } + //#endif + + /* Flush as much pending output as possible */ + if (s.pending !== 0) { + flush_pending(strm); + if (strm.avail_out === 0) { + /* Since avail_out is 0, deflate will be called again with + * more output space, but possibly with both pending and + * avail_in equal to zero. There won't be anything to do, + * but this is not an error situation so make sure we + * return OK instead of BUF_ERROR at next call of deflate: + */ + s.last_flush = -1; + return Z_OK; + } + + /* Make sure there is something to do and avoid duplicate consecutive + * flushes. For repeated and useless calls with Z_FINISH, we keep + * returning Z_STREAM_END instead of Z_BUF_ERROR. + */ + } else if (strm.avail_in === 0 && rank(flush) <= rank(old_flush) && + flush !== Z_FINISH) { + return err(strm, Z_BUF_ERROR); + } + + /* User must not provide more input after the first FINISH: */ + if (s.status === FINISH_STATE && strm.avail_in !== 0) { + return err(strm, Z_BUF_ERROR); + } + + /* Start a new block or continue the current one. + */ + if (strm.avail_in !== 0 || s.lookahead !== 0 || + (flush !== Z_NO_FLUSH && s.status !== FINISH_STATE)) { + var bstate = (s.strategy === Z_HUFFMAN_ONLY) ? deflate_huff(s, flush) : + (s.strategy === Z_RLE ? deflate_rle(s, flush) : + configuration_table[s.level].func(s, flush)); + + if (bstate === BS_FINISH_STARTED || bstate === BS_FINISH_DONE) { + s.status = FINISH_STATE; + } + if (bstate === BS_NEED_MORE || bstate === BS_FINISH_STARTED) { + if (strm.avail_out === 0) { + s.last_flush = -1; + /* avoid BUF_ERROR next call, see above */ + } + return Z_OK; + /* If flush != Z_NO_FLUSH && avail_out == 0, the next call + * of deflate should use the same flush parameter to make sure + * that the flush is complete. So we don't have to output an + * empty block here, this will be done at next call. This also + * ensures that for a very small output buffer, we emit at most + * one empty block. + */ + } + if (bstate === BS_BLOCK_DONE) { + if (flush === Z_PARTIAL_FLUSH) { + _tr_align(s); + } + else if (flush !== Z_BLOCK) { /* FULL_FLUSH or SYNC_FLUSH */ + + _tr_stored_block(s, 0, 0, false); + /* For a full flush, this empty block will be recognized + * as a special marker by inflate_sync(). + */ + if (flush === Z_FULL_FLUSH) { + /*** CLEAR_HASH(s); ***/ /* forget history */ + zero$2(s.head); // Fill with NIL (= 0); + + if (s.lookahead === 0) { + s.strstart = 0; + s.block_start = 0; + s.insert = 0; + } + } + } + flush_pending(strm); + if (strm.avail_out === 0) { + s.last_flush = -1; /* avoid BUF_ERROR at next call, see above */ + return Z_OK; + } + } + } + //Assert(strm->avail_out > 0, "bug2"); + //if (strm.avail_out <= 0) { throw new Error("bug2");} + + if (flush !== Z_FINISH) { return Z_OK; } + if (s.wrap <= 0) { return Z_STREAM_END; } + + /* Write the trailer */ + if (s.wrap === 2) { + put_byte(s, strm.adler & 0xff); + put_byte(s, (strm.adler >> 8) & 0xff); + put_byte(s, (strm.adler >> 16) & 0xff); + put_byte(s, (strm.adler >> 24) & 0xff); + put_byte(s, strm.total_in & 0xff); + put_byte(s, (strm.total_in >> 8) & 0xff); + put_byte(s, (strm.total_in >> 16) & 0xff); + put_byte(s, (strm.total_in >> 24) & 0xff); + } + else { + putShortMSB(s, strm.adler >>> 16); + putShortMSB(s, strm.adler & 0xffff); + } + + flush_pending(strm); + /* If avail_out is zero, the application will call deflate again + * to flush the rest. + */ + if (s.wrap > 0) { s.wrap = -s.wrap; } + /* write the trailer only once! */ + return s.pending !== 0 ? Z_OK : Z_STREAM_END; + } + + function deflateEnd(strm) { + let status; + + if (!strm/*== Z_NULL*/ || !strm.state/*== Z_NULL*/) { + return Z_STREAM_ERROR; + } + + status = strm.state.status; + if (status !== INIT_STATE && + status !== EXTRA_STATE && + status !== NAME_STATE && + status !== COMMENT_STATE && + status !== HCRC_STATE && + status !== BUSY_STATE && + status !== FINISH_STATE + ) { + return err(strm, Z_STREAM_ERROR); + } + + strm.state = null; + + return status === BUSY_STATE ? err(strm, Z_DATA_ERROR) : Z_OK; + } + + + /* ========================================================================= + * Initializes the compression dictionary from the given byte + * sequence without producing any compressed output. + */ + function deflateSetDictionary(strm, dictionary) { + let dictLength = dictionary.length; + + let s; + let str, n; + let wrap; + let avail; + let next; + let input; + let tmpDict; + + if (!strm/*== Z_NULL*/ || !strm.state/*== Z_NULL*/) { + return Z_STREAM_ERROR; + } + + s = strm.state; + wrap = s.wrap; + + if (wrap === 2 || (wrap === 1 && s.status !== INIT_STATE) || s.lookahead) { + return Z_STREAM_ERROR; + } + + /* when using zlib wrappers, compute Adler-32 for provided dictionary */ + if (wrap === 1) { + /* adler32(strm->adler, dictionary, dictLength); */ + strm.adler = adler32(strm.adler, dictionary, dictLength, 0); + } + + s.wrap = 0; /* avoid computing Adler-32 in read_buf */ + + /* if dictionary would fill window, just replace the history */ + if (dictLength >= s.w_size) { + if (wrap === 0) { /* already empty otherwise */ + /*** CLEAR_HASH(s); ***/ + zero$2(s.head); // Fill with NIL (= 0); + s.strstart = 0; + s.block_start = 0; + s.insert = 0; + } + /* use the tail */ + // dictionary = dictionary.slice(dictLength - s.w_size); + tmpDict = new Buf8(s.w_size); + arraySet(tmpDict, dictionary, dictLength - s.w_size, s.w_size, 0); + dictionary = tmpDict; + dictLength = s.w_size; + } + /* insert dictionary into window and hash */ + avail = strm.avail_in; + next = strm.next_in; + input = strm.input; + strm.avail_in = dictLength; + strm.next_in = 0; + strm.input = dictionary; + fill_window(s); + while (s.lookahead >= MIN_MATCH$1) { + str = s.strstart; + n = s.lookahead - (MIN_MATCH$1 - 1); + do { + /* UPDATE_HASH(s, s->ins_h, s->window[str + MIN_MATCH-1]); */ + s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[str + MIN_MATCH$1 - 1]) & s.hash_mask; + + s.prev[str & s.w_mask] = s.head[s.ins_h]; + + s.head[s.ins_h] = str; + str++; + } while (--n); + s.strstart = str; + s.lookahead = MIN_MATCH$1 - 1; + fill_window(s); + } + s.strstart += s.lookahead; + s.block_start = s.strstart; + s.insert = s.lookahead; + s.lookahead = 0; + s.match_length = s.prev_length = MIN_MATCH$1 - 1; + s.match_available = 0; + strm.next_in = next; + strm.input = input; + strm.avail_in = avail; + s.wrap = wrap; + return Z_OK; + } + + /* Not implemented + exports.deflateBound = deflateBound; + exports.deflateCopy = deflateCopy; + exports.deflateParams = deflateParams; + exports.deflatePending = deflatePending; + exports.deflatePrime = deflatePrime; + exports.deflateTune = deflateTune; + */ + + // String encode/decode helpers + + try { + String.fromCharCode.apply(null, [ 0 ]); + } catch (__) { + } + try { + String.fromCharCode.apply(null, new Uint8Array(1)); + } catch (__) { + } + + + // Table with utf8 lengths (calculated by first byte of sequence) + // Note, that 5 & 6-byte values and some 4-byte values can not be represented in JS, + // because max possible codepoint is 0x10ffff + const _utf8len = new Buf8(256); + for (let q = 0; q < 256; q++) { + _utf8len[q] = q >= 252 ? 6 : q >= 248 ? 5 : q >= 240 ? 4 : q >= 224 ? 3 : q >= 192 ? 2 : 1; + } + _utf8len[254] = _utf8len[254] = 1; // Invalid sequence start + + + // convert string to array (typed, when possible) + function string2buf (str) { + let c, c2, m_pos, i, buf_len = 0; + const str_len = str.length; + + // count binary size + for (m_pos = 0; m_pos < str_len; m_pos++) { + c = str.charCodeAt(m_pos); + if ((c & 0xfc00) === 0xd800 && m_pos + 1 < str_len) { + c2 = str.charCodeAt(m_pos + 1); + if ((c2 & 0xfc00) === 0xdc00) { + c = 0x10000 + (c - 0xd800 << 10) + (c2 - 0xdc00); + m_pos++; + } + } + buf_len += c < 0x80 ? 1 : c < 0x800 ? 2 : c < 0x10000 ? 3 : 4; + } + + // allocate buffer + const buf = new Buf8(buf_len); + + // convert + for (i = 0, m_pos = 0; i < buf_len; m_pos++) { + c = str.charCodeAt(m_pos); + if ((c & 0xfc00) === 0xd800 && m_pos + 1 < str_len) { + c2 = str.charCodeAt(m_pos + 1); + if ((c2 & 0xfc00) === 0xdc00) { + c = 0x10000 + (c - 0xd800 << 10) + (c2 - 0xdc00); + m_pos++; + } + } + if (c < 0x80) { + /* one byte */ + buf[i++] = c; + } else if (c < 0x800) { + /* two bytes */ + buf[i++] = 0xC0 | c >>> 6; + buf[i++] = 0x80 | c & 0x3f; + } else if (c < 0x10000) { + /* three bytes */ + buf[i++] = 0xE0 | c >>> 12; + buf[i++] = 0x80 | c >>> 6 & 0x3f; + buf[i++] = 0x80 | c & 0x3f; + } else { + /* four bytes */ + buf[i++] = 0xf0 | c >>> 18; + buf[i++] = 0x80 | c >>> 12 & 0x3f; + buf[i++] = 0x80 | c >>> 6 & 0x3f; + buf[i++] = 0x80 | c & 0x3f; + } + } + + return buf; + } + + + // Convert binary string (typed, when possible) + function binstring2buf (str) { + const buf = new Buf8(str.length); + for (let i = 0, len = buf.length; i < len; i++) { + buf[i] = str.charCodeAt(i); + } + return buf; + } + + // (C) 1995-2013 Jean-loup Gailly and Mark Adler + // (C) 2014-2017 Vitaly Puzrin and Andrey Tupitsin + // + // This software is provided 'as-is', without any express or implied + // warranty. In no event will the authors be held liable for any damages + // arising from the use of this software. + // + // Permission is granted to anyone to use this software for any purpose, + // including commercial applications, and to alter it and redistribute it + // freely, subject to the following restrictions: + // + // 1. The origin of this software must not be misrepresented; you must not + // claim that you wrote the original software. If you use this software + // in a product, an acknowledgment in the product documentation would be + // appreciated but is not required. + // 2. Altered source versions must be plainly marked as such, and must not be + // misrepresented as being the original software. + // 3. This notice may not be removed or altered from any source distribution. + + class ZStream { + constructor() { + /* next input byte */ + this.input = null; // JS specific, because we have no pointers + this.next_in = 0; + /* number of bytes available at input */ + this.avail_in = 0; + /* total number of input bytes read so far */ + this.total_in = 0; + /* next output byte should be put there */ + this.output = null; // JS specific, because we have no pointers + this.next_out = 0; + /* remaining free space at output */ + this.avail_out = 0; + /* total number of bytes output so far */ + this.total_out = 0; + /* last error message, NULL if no error */ + this.msg = ''/*Z_NULL*/; + /* not visible by applications */ + this.state = null; + /* best guess about the data type: binary or text */ + this.data_type = 2/*Z_UNKNOWN*/; + /* adler32 value of the uncompressed data */ + this.adler = 0; + } + } + + /* ===========================================================================*/ + + + /** + * class Deflate + * + * Generic JS-style wrapper for zlib calls. If you don't need + * streaming behaviour - use more simple functions: [[deflate]], + * [[deflateRaw]] and [[gzip]]. + **/ + + /* internal + * Deflate.chunks -> Array + * + * Chunks of output data, if [[Deflate#onData]] not overridden. + **/ + + /** + * Deflate.result -> Uint8Array|Array + * + * Compressed result, generated by default [[Deflate#onData]] + * and [[Deflate#onEnd]] handlers. Filled after you push last chunk + * (call [[Deflate#push]] with `Z_FINISH` / `true` param) or if you + * push a chunk with explicit flush (call [[Deflate#push]] with + * `Z_SYNC_FLUSH` param). + **/ + + /** + * Deflate.err -> Number + * + * Error code after deflate finished. 0 (Z_OK) on success. + * You will not need it in real life, because deflate errors + * are possible only on wrong options or bad `onData` / `onEnd` + * custom handlers. + **/ + + /** + * Deflate.msg -> String + * + * Error message, if [[Deflate.err]] != 0 + **/ + + + /** + * new Deflate(options) + * - options (Object): zlib deflate options. + * + * Creates new deflator instance with specified params. Throws exception + * on bad params. Supported options: + * + * - `level` + * - `windowBits` + * - `memLevel` + * - `strategy` + * - `dictionary` + * + * [http://zlib.net/manual.html#Advanced](http://zlib.net/manual.html#Advanced) + * for more information on these. + * + * Additional options, for internal needs: + * + * - `chunkSize` - size of generated data chunks (16K by default) + * - `raw` (Boolean) - do raw deflate + * - `gzip` (Boolean) - create gzip wrapper + * - `to` (String) - if equal to 'string', then result will be "binary string" + * (each char code [0..255]) + * - `header` (Object) - custom header for gzip + * - `text` (Boolean) - true if compressed data believed to be text + * - `time` (Number) - modification time, unix timestamp + * - `os` (Number) - operation system code + * - `extra` (Array) - array of bytes with extra data (max 65536) + * - `name` (String) - file name (binary string) + * - `comment` (String) - comment (binary string) + * - `hcrc` (Boolean) - true if header crc should be added + * + * ##### Example: + * + * ```javascript + * var pako = void('pako') + * , chunk1 = Uint8Array([1,2,3,4,5,6,7,8,9]) + * , chunk2 = Uint8Array([10,11,12,13,14,15,16,17,18,19]); + * + * var deflate = new pako.Deflate({ level: 3}); + * + * deflate.push(chunk1, false); + * deflate.push(chunk2, true); // true -> last chunk + * + * if (deflate.err) { throw new Error(deflate.err); } + * + * console.log(deflate.result); + * ``` + **/ + + class Deflate { + constructor(options) { + this.options = { + level: Z_DEFAULT_COMPRESSION, + method: Z_DEFLATED, + chunkSize: 16384, + windowBits: 15, + memLevel: 8, + strategy: Z_DEFAULT_STRATEGY, + ...(options || {}) + }; + + const opt = this.options; + + if (opt.raw && (opt.windowBits > 0)) { + opt.windowBits = -opt.windowBits; + } + + else if (opt.gzip && (opt.windowBits > 0) && (opt.windowBits < 16)) { + opt.windowBits += 16; + } + + this.err = 0; // error code, if happens (0 = Z_OK) + this.msg = ''; // error message + this.ended = false; // used to avoid multiple onEnd() calls + this.chunks = []; // chunks of compressed data + + this.strm = new ZStream(); + this.strm.avail_out = 0; + + var status = deflateInit2( + this.strm, + opt.level, + opt.method, + opt.windowBits, + opt.memLevel, + opt.strategy + ); + + if (status !== Z_OK) { + throw new Error(msg[status]); + } + + if (opt.header) { + deflateSetHeader(this.strm, opt.header); + } + + if (opt.dictionary) { + let dict; + // Convert data if needed + if (typeof opt.dictionary === 'string') { + // If we need to compress text, change encoding to utf8. + dict = string2buf(opt.dictionary); + } else if (opt.dictionary instanceof ArrayBuffer) { + dict = new Uint8Array(opt.dictionary); + } else { + dict = opt.dictionary; + } + + status = deflateSetDictionary(this.strm, dict); + + if (status !== Z_OK) { + throw new Error(msg[status]); + } + + this._dict_set = true; + } + } + + /** + * Deflate#push(data[, mode]) -> Boolean + * - data (Uint8Array|Array|ArrayBuffer|String): input data. Strings will be + * converted to utf8 byte sequence. + * - mode (Number|Boolean): 0..6 for corresponding Z_NO_FLUSH..Z_TREE modes. + * See constants. Skipped or `false` means Z_NO_FLUSH, `true` means Z_FINISH. + * + * Sends input data to deflate pipe, generating [[Deflate#onData]] calls with + * new compressed chunks. Returns `true` on success. The last data block must have + * mode Z_FINISH (or `true`). That will flush internal pending buffers and call + * [[Deflate#onEnd]]. For interim explicit flushes (without ending the stream) you + * can use mode Z_SYNC_FLUSH, keeping the compression context. + * + * On fail call [[Deflate#onEnd]] with error code and return false. + * + * We strongly recommend to use `Uint8Array` on input for best speed (output + * array format is detected automatically). Also, don't skip last param and always + * use the same type in your code (boolean or number). That will improve JS speed. + * + * For regular `Array`-s make sure all elements are [0..255]. + * + * ##### Example + * + * ```javascript + * push(chunk, false); // push one of data chunks + * ... + * push(chunk, true); // push last chunk + * ``` + **/ + push(data, mode) { + const { strm, options: { chunkSize } } = this; + var status, _mode; + + if (this.ended) { return false; } + + _mode = (mode === ~~mode) ? mode : ((mode === true) ? Z_FINISH : Z_NO_FLUSH); + + // Convert data if needed + if (typeof data === 'string') { + // If we need to compress text, change encoding to utf8. + strm.input = string2buf(data); + } else if (data instanceof ArrayBuffer) { + strm.input = new Uint8Array(data); + } else { + strm.input = data; + } + + strm.next_in = 0; + strm.avail_in = strm.input.length; + + do { + if (strm.avail_out === 0) { + strm.output = new Buf8(chunkSize); + strm.next_out = 0; + strm.avail_out = chunkSize; + } + status = deflate(strm, _mode); /* no bad return value */ + + if (status !== Z_STREAM_END && status !== Z_OK) { + this.onEnd(status); + this.ended = true; + return false; + } + if (strm.avail_out === 0 || (strm.avail_in === 0 && (_mode === Z_FINISH || _mode === Z_SYNC_FLUSH))) { + this.onData(shrinkBuf(strm.output, strm.next_out)); + } + } while ((strm.avail_in > 0 || strm.avail_out === 0) && status !== Z_STREAM_END); + + // Finalize on the last chunk. + if (_mode === Z_FINISH) { + status = deflateEnd(this.strm); + this.onEnd(status); + this.ended = true; + return status === Z_OK; + } + + // callback interim results if Z_SYNC_FLUSH. + if (_mode === Z_SYNC_FLUSH) { + this.onEnd(Z_OK); + strm.avail_out = 0; + return true; + } + + return true; + }; + /** + * Deflate#onData(chunk) -> Void + * - chunk (Uint8Array|Array|String): output data. Type of array depends + * on js engine support. When string output requested, each chunk + * will be string. + * + * By default, stores data blocks in `chunks[]` property and glue + * those in `onEnd`. Override this handler, if you need another behaviour. + **/ + onData(chunk) { + this.chunks.push(chunk); + }; + + /** + * Deflate#onEnd(status) -> Void + * - status (Number): deflate status. 0 (Z_OK) on success, + * other if not. + * + * Called once after you tell deflate that the input stream is + * complete (Z_FINISH) or should be flushed (Z_SYNC_FLUSH) + * or if an error happened. By default - join collected chunks, + * free memory and fill `results` / `err` properties. + **/ + onEnd(status) { + // On success - join + if (status === Z_OK) { + this.result = flattenChunks(this.chunks); + } + this.chunks = []; + this.err = status; + this.msg = this.strm.msg; + }; + } + + // (C) 1995-2013 Jean-loup Gailly and Mark Adler + // (C) 2014-2017 Vitaly Puzrin and Andrey Tupitsin + // + // This software is provided 'as-is', without any express or implied + // warranty. In no event will the authors be held liable for any damages + // arising from the use of this software. + // + // Permission is granted to anyone to use this software for any purpose, + // including commercial applications, and to alter it and redistribute it + // freely, subject to the following restrictions: + // + // 1. The origin of this software must not be misrepresented; you must not + // claim that you wrote the original software. If you use this software + // in a product, an acknowledgment in the product documentation would be + // appreciated but is not required. + // 2. Altered source versions must be plainly marked as such, and must not be + // misrepresented as being the original software. + // 3. This notice may not be removed or altered from any source distribution. + + // See state defs from inflate.js + const BAD = 30; /* got a data error -- remain here until reset */ + const TYPE = 12; /* i: waiting for type bits, including last-flag bit */ + + /* + Decode literal, length, and distance codes and write out the resulting + literal and match bytes until either not enough input or output is + available, an end-of-block is encountered, or a data error is encountered. + When large enough input and output buffers are supplied to inflate(), for + example, a 16K input buffer and a 64K output buffer, more than 95% of the + inflate execution time is spent in this routine. + + Entry assumptions: + + state.mode === LEN + strm.avail_in >= 6 + strm.avail_out >= 258 + start >= strm.avail_out + state.bits < 8 + + On return, state.mode is one of: + + LEN -- ran out of enough output space or enough available input + TYPE -- reached end of block code, inflate() to interpret next block + BAD -- error in block data + + Notes: + + - The maximum input bits used by a length/distance pair is 15 bits for the + length code, 5 bits for the length extra, 15 bits for the distance code, + and 13 bits for the distance extra. This totals 48 bits, or six bytes. + Therefore if strm.avail_in >= 6, then there is enough input to avoid + checking for available input while decoding. + + - The maximum bytes that a single length/distance pair can output is 258 + bytes, which is the maximum length that can be coded. inflate_fast() + requires strm.avail_out >= 258 for each loop to avoid checking for + output space. + */ + function inflate_fast(strm, start) { + let _in; /* local strm.input */ + let _out; /* local strm.output */ + // Use `s_window` instead `window`, avoid conflict with instrumentation tools + let hold; /* local strm.hold */ + let bits; /* local strm.bits */ + let here; /* retrieved table entry */ + let op; /* code bits, operation, extra bits, or */ + /* window position, window bytes to copy */ + let len; /* match length, unused bytes */ + let dist; /* match distance */ + let from; /* where to copy match from */ + let from_source; + + + + /* copy state to local variables */ + const state = strm.state; + //here = state.here; + _in = strm.next_in; + const input = strm.input; + const last = _in + (strm.avail_in - 5); + _out = strm.next_out; + const output = strm.output; + const beg = _out - (start - strm.avail_out); + const end = _out + (strm.avail_out - 257); + //#ifdef INFLATE_STRICT + const dmax = state.dmax; + //#endif + const wsize = state.wsize; + const whave = state.whave; + const wnext = state.wnext; + const s_window = state.window; + hold = state.hold; + bits = state.bits; + const lcode = state.lencode; + const dcode = state.distcode; + const lmask = (1 << state.lenbits) - 1; + const dmask = (1 << state.distbits) - 1; + + + /* decode literals and length/distances until end-of-block or not enough + input data or output space */ + + top: + do { + if (bits < 15) { + hold += input[_in++] << bits; + bits += 8; + hold += input[_in++] << bits; + bits += 8; + } + + here = lcode[hold & lmask]; + + dolen: + for (;;) { // Goto emulation + op = here >>> 24/*here.bits*/; + hold >>>= op; + bits -= op; + op = here >>> 16 & 0xff/*here.op*/; + if (op === 0) { /* literal */ + //Tracevv((stderr, here.val >= 0x20 && here.val < 0x7f ? + // "inflate: literal '%c'\n" : + // "inflate: literal 0x%02x\n", here.val)); + output[_out++] = here & 0xffff/*here.val*/; + } else if (op & 16) { /* length base */ + len = here & 0xffff/*here.val*/; + op &= 15; /* number of extra bits */ + if (op) { + if (bits < op) { + hold += input[_in++] << bits; + bits += 8; + } + len += hold & (1 << op) - 1; + hold >>>= op; + bits -= op; + } + //Tracevv((stderr, "inflate: length %u\n", len)); + if (bits < 15) { + hold += input[_in++] << bits; + bits += 8; + hold += input[_in++] << bits; + bits += 8; + } + here = dcode[hold & dmask]; + + dodist: + for (;;) { // goto emulation + op = here >>> 24/*here.bits*/; + hold >>>= op; + bits -= op; + op = here >>> 16 & 0xff/*here.op*/; + + if (op & 16) { /* distance base */ + dist = here & 0xffff/*here.val*/; + op &= 15; /* number of extra bits */ + if (bits < op) { + hold += input[_in++] << bits; + bits += 8; + if (bits < op) { + hold += input[_in++] << bits; + bits += 8; + } + } + dist += hold & (1 << op) - 1; + //#ifdef INFLATE_STRICT + if (dist > dmax) { + strm.msg = "invalid distance too far back"; + state.mode = BAD; + break top; + } + //#endif + hold >>>= op; + bits -= op; + //Tracevv((stderr, "inflate: distance %u\n", dist)); + op = _out - beg; /* max distance in output */ + if (dist > op) { /* see if copy from window */ + op = dist - op; /* distance back in window */ + if (op > whave) { + if (state.sane) { + strm.msg = "invalid distance too far back"; + state.mode = BAD; + break top; + } + + // (!) This block is disabled in zlib defaults, + // don't enable it for binary compatibility + //#ifdef INFLATE_ALLOW_INVALID_DISTANCE_TOOFAR_ARRR + // if (len <= op - whave) { + // do { + // output[_out++] = 0; + // } while (--len); + // continue top; + // } + // len -= op - whave; + // do { + // output[_out++] = 0; + // } while (--op > whave); + // if (op === 0) { + // from = _out - dist; + // do { + // output[_out++] = output[from++]; + // } while (--len); + // continue top; + // } + //#endif + } + from = 0; // window index + from_source = s_window; + if (wnext === 0) { /* very common case */ + from += wsize - op; + if (op < len) { /* some from window */ + len -= op; + do { + output[_out++] = s_window[from++]; + } while (--op); + from = _out - dist; /* rest from output */ + from_source = output; + } + } else if (wnext < op) { /* wrap around window */ + from += wsize + wnext - op; + op -= wnext; + if (op < len) { /* some from end of window */ + len -= op; + do { + output[_out++] = s_window[from++]; + } while (--op); + from = 0; + if (wnext < len) { /* some from start of window */ + op = wnext; + len -= op; + do { + output[_out++] = s_window[from++]; + } while (--op); + from = _out - dist; /* rest from output */ + from_source = output; + } + } + } else { /* contiguous in window */ + from += wnext - op; + if (op < len) { /* some from window */ + len -= op; + do { + output[_out++] = s_window[from++]; + } while (--op); + from = _out - dist; /* rest from output */ + from_source = output; + } + } + while (len > 2) { + output[_out++] = from_source[from++]; + output[_out++] = from_source[from++]; + output[_out++] = from_source[from++]; + len -= 3; + } + if (len) { + output[_out++] = from_source[from++]; + if (len > 1) { + output[_out++] = from_source[from++]; + } + } + } else { + from = _out - dist; /* copy direct from output */ + do { /* minimum length is three */ + output[_out++] = output[from++]; + output[_out++] = output[from++]; + output[_out++] = output[from++]; + len -= 3; + } while (len > 2); + if (len) { + output[_out++] = output[from++]; + if (len > 1) { + output[_out++] = output[from++]; + } + } + } + } else if ((op & 64) === 0) { /* 2nd level distance code */ + here = dcode[(here & 0xffff)/*here.val*/ + (hold & (1 << op) - 1)]; + continue dodist; + } else { + strm.msg = "invalid distance code"; + state.mode = BAD; + break top; + } + + break; // need to emulate goto via "continue" + } + } else if ((op & 64) === 0) { /* 2nd level length code */ + here = lcode[(here & 0xffff)/*here.val*/ + (hold & (1 << op) - 1)]; + continue dolen; + } else if (op & 32) { /* end-of-block */ + //Tracevv((stderr, "inflate: end of block\n")); + state.mode = TYPE; + break top; + } else { + strm.msg = "invalid literal/length code"; + state.mode = BAD; + break top; + } + + break; // need to emulate goto via "continue" + } + } while (_in < last && _out < end); + + /* return unused bytes (on entry, bits < 8, so in won't go too far back) */ + len = bits >> 3; + _in -= len; + bits -= len << 3; + hold &= (1 << bits) - 1; + + /* update state and return */ + strm.next_in = _in; + strm.next_out = _out; + strm.avail_in = _in < last ? 5 + (last - _in) : 5 - (_in - last); + strm.avail_out = _out < end ? 257 + (end - _out) : 257 - (_out - end); + state.hold = hold; + state.bits = bits; + return; + } + + const MAXBITS = 15; + const ENOUGH_LENS = 852; + const ENOUGH_DISTS = 592; + //var ENOUGH = (ENOUGH_LENS+ENOUGH_DISTS); + + const CODES = 0; + const LENS = 1; + const DISTS = 2; + + const lbase = [ /* Length codes 257..285 base */ + 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31, + 35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0 + ]; + + const lext = [ /* Length codes 257..285 extra */ + 16, 16, 16, 16, 16, 16, 16, 16, 17, 17, 17, 17, 18, 18, 18, 18, + 19, 19, 19, 19, 20, 20, 20, 20, 21, 21, 21, 21, 16, 72, 78 + ]; + + const dbase = [ /* Distance codes 0..29 base */ + 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193, + 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145, + 8193, 12289, 16385, 24577, 0, 0 + ]; + + const dext = [ /* Distance codes 0..29 extra */ + 16, 16, 16, 16, 17, 17, 18, 18, 19, 19, 20, 20, 21, 21, 22, 22, + 23, 23, 24, 24, 25, 25, 26, 26, 27, 27, + 28, 28, 29, 29, 64, 64 + ]; + + function inflate_table(type, lens, lens_index, codes, table, table_index, work, opts) { + const bits = opts.bits; + //here = opts.here; /* table entry for duplication */ + + let len = 0; /* a code's length in bits */ + let sym = 0; /* index of code symbols */ + let min = 0, max = 0; /* minimum and maximum code lengths */ + let root = 0; /* number of index bits for root table */ + let curr = 0; /* number of index bits for current table */ + let drop = 0; /* code bits to drop for sub-table */ + let left = 0; /* number of prefix codes available */ + let used = 0; /* code entries in table used */ + let huff = 0; /* Huffman code */ + let incr; /* for incrementing code, index */ + let fill; /* index for replicating entries */ + let low; /* low bits for current root entry */ + let next; /* next available space in table */ + let base = null; /* base value table to use */ + let base_index = 0; + // var shoextra; /* extra bits table to use */ + let end; /* use base and extra for symbol > end */ + const count = new Buf16(MAXBITS + 1); //[MAXBITS+1]; /* number of codes of each length */ + const offs = new Buf16(MAXBITS + 1); //[MAXBITS+1]; /* offsets in table for each length */ + let extra = null; + let extra_index = 0; + + let here_bits, here_op, here_val; + + /* + Process a set of code lengths to create a canonical Huffman code. The + code lengths are lens[0..codes-1]. Each length corresponds to the + symbols 0..codes-1. The Huffman code is generated by first sorting the + symbols by length from short to long, and retaining the symbol order + for codes with equal lengths. Then the code starts with all zero bits + for the first code of the shortest length, and the codes are integer + increments for the same length, and zeros are appended as the length + increases. For the deflate format, these bits are stored backwards + from their more natural integer increment ordering, and so when the + decoding tables are built in the large loop below, the integer codes + are incremented backwards. + + This routine assumes, but does not check, that all of the entries in + lens[] are in the range 0..MAXBITS. The caller must assure this. + 1..MAXBITS is interpreted as that code length. zero means that that + symbol does not occur in this code. + + The codes are sorted by computing a count of codes for each length, + creating from that a table of starting indices for each length in the + sorted table, and then entering the symbols in order in the sorted + table. The sorted table is work[], with that space being provided by + the caller. + + The length counts are used for other purposes as well, i.e. finding + the minimum and maximum length codes, determining if there are any + codes at all, checking for a valid set of lengths, and looking ahead + at length counts to determine sub-table sizes when building the + decoding tables. + */ + + /* accumulate lengths for codes (assumes lens[] all in 0..MAXBITS) */ + for (len = 0; len <= MAXBITS; len++) { + count[len] = 0; + } + for (sym = 0; sym < codes; sym++) { + count[lens[lens_index + sym]]++; + } + + /* bound code lengths, force root to be within code lengths */ + root = bits; + for (max = MAXBITS; max >= 1; max--) { + if (count[max] !== 0) { + break; + } + } + if (root > max) { + root = max; + } + if (max === 0) { /* no symbols to code at all */ + //table.op[opts.table_index] = 64; //here.op = (var char)64; /* invalid code marker */ + //table.bits[opts.table_index] = 1; //here.bits = (var char)1; + //table.val[opts.table_index++] = 0; //here.val = (var short)0; + table[table_index++] = 1 << 24 | 64 << 16 | 0; + + + //table.op[opts.table_index] = 64; + //table.bits[opts.table_index] = 1; + //table.val[opts.table_index++] = 0; + table[table_index++] = 1 << 24 | 64 << 16 | 0; + + opts.bits = 1; + return 0; /* no symbols, but wait for decoding to report error */ + } + for (min = 1; min < max; min++) { + if (count[min] !== 0) { + break; + } + } + if (root < min) { + root = min; + } + + /* check for an over-subscribed or incomplete set of lengths */ + left = 1; + for (len = 1; len <= MAXBITS; len++) { + left <<= 1; + left -= count[len]; + if (left < 0) { + return -1; + } /* over-subscribed */ + } + if (left > 0 && (type === CODES || max !== 1)) { + return -1; /* incomplete set */ + } + + /* generate offsets into symbol table for each length for sorting */ + offs[1] = 0; + for (len = 1; len < MAXBITS; len++) { + offs[len + 1] = offs[len] + count[len]; + } + + /* sort symbols by length, by symbol order within each length */ + for (sym = 0; sym < codes; sym++) { + if (lens[lens_index + sym] !== 0) { + work[offs[lens[lens_index + sym]]++] = sym; + } + } + + /* + Create and fill in decoding tables. In this loop, the table being + filled is at next and has curr index bits. The code being used is huff + with length len. That code is converted to an index by dropping drop + bits off of the bottom. For codes where len is less than drop + curr, + those top drop + curr - len bits are incremented through all values to + fill the table with replicated entries. + + root is the number of index bits for the root table. When len exceeds + root, sub-tables are created pointed to by the root entry with an index + of the low root bits of huff. This is saved in low to check for when a + new sub-table should be started. drop is zero when the root table is + being filled, and drop is root when sub-tables are being filled. + + When a new sub-table is needed, it is necessary to look ahead in the + code lengths to determine what size sub-table is needed. The length + counts are used for this, and so count[] is decremented as codes are + entered in the tables. + + used keeps track of how many table entries have been allocated from the + provided *table space. It is checked for LENS and DIST tables against + the constants ENOUGH_LENS and ENOUGH_DISTS to guard against changes in + the initial root table size constants. See the comments in inftrees.h + for more information. + + sym increments through all symbols, and the loop terminates when + all codes of length max, i.e. all codes, have been processed. This + routine permits incomplete codes, so another loop after this one fills + in the rest of the decoding tables with invalid code markers. + */ + + /* set up for code type */ + // poor man optimization - use if-else instead of switch, + // to avoid deopts in old v8 + if (type === CODES) { + base = extra = work; /* dummy value--not used */ + end = 19; + + } else if (type === LENS) { + base = lbase; + base_index -= 257; + extra = lext; + extra_index -= 257; + end = 256; + + } else { /* DISTS */ + base = dbase; + extra = dext; + end = -1; + } + + /* initialize opts for loop */ + huff = 0; /* starting code */ + sym = 0; /* starting code symbol */ + len = min; /* starting code length */ + next = table_index; /* current table to fill in */ + curr = root; /* current table index bits */ + drop = 0; /* current bits to drop from code for index */ + low = -1; /* trigger new sub-table when len > root */ + used = 1 << root; /* use root table entries */ + const mask = used - 1; /* mask for comparing low */ + + /* check available table space */ + if (type === LENS && used > ENOUGH_LENS || + type === DISTS && used > ENOUGH_DISTS) { + return 1; + } + + /* process all codes and make table entries */ + for (;;) { + /* create table entry */ + here_bits = len - drop; + if (work[sym] < end) { + here_op = 0; + here_val = work[sym]; + } else if (work[sym] > end) { + here_op = extra[extra_index + work[sym]]; + here_val = base[base_index + work[sym]]; + } else { + here_op = 32 + 64; /* end of block */ + here_val = 0; + } + + /* replicate for those indices with low len bits equal to huff */ + incr = 1 << len - drop; + fill = 1 << curr; + min = fill; /* save offset to next table */ + do { + fill -= incr; + table[next + (huff >> drop) + fill] = here_bits << 24 | here_op << 16 | here_val |0; + } while (fill !== 0); + + /* backwards increment the len-bit code huff */ + incr = 1 << len - 1; + while (huff & incr) { + incr >>= 1; + } + if (incr !== 0) { + huff &= incr - 1; + huff += incr; + } else { + huff = 0; + } + + /* go to next symbol, update count, len */ + sym++; + if (--count[len] === 0) { + if (len === max) { + break; + } + len = lens[lens_index + work[sym]]; + } + + /* create new sub-table if needed */ + if (len > root && (huff & mask) !== low) { + /* if first time, transition to sub-tables */ + if (drop === 0) { + drop = root; + } + + /* increment past last table */ + next += min; /* here min is 1 << curr */ + + /* determine length of next table */ + curr = len - drop; + left = 1 << curr; + while (curr + drop < max) { + left -= count[curr + drop]; + if (left <= 0) { + break; + } + curr++; + left <<= 1; + } + + /* check for enough space */ + used += 1 << curr; + if (type === LENS && used > ENOUGH_LENS || + type === DISTS && used > ENOUGH_DISTS) { + return 1; + } + + /* point entry in root table to sub-table */ + low = huff & mask; + /*table.op[low] = curr; + table.bits[low] = root; + table.val[low] = next - opts.table_index;*/ + table[low] = root << 24 | curr << 16 | next - table_index |0; + } + } + + /* fill in remaining table entry if code is incomplete (guaranteed to have + at most one remaining entry, since if the code is incomplete, the + maximum code length that was allowed to get this far is one bit) */ + if (huff !== 0) { + //table.op[next + huff] = 64; /* invalid code marker */ + //table.bits[next + huff] = len - drop; + //table.val[next + huff] = 0; + table[next + huff] = len - drop << 24 | 64 << 16 |0; + } + + /* set return parameters */ + //opts.table_index += used; + opts.bits = root; + return 0; + } + + const CODES$1 = 0; + const LENS$1 = 1; + const DISTS$1 = 2; + + /* STATES ====================================================================*/ + /* ===========================================================================*/ + + + const HEAD = 1; /* i: waiting for magic header */ + const FLAGS = 2; /* i: waiting for method and flags (gzip) */ + const TIME = 3; /* i: waiting for modification time (gzip) */ + const OS = 4; /* i: waiting for extra flags and operating system (gzip) */ + const EXLEN = 5; /* i: waiting for extra length (gzip) */ + const EXTRA = 6; /* i: waiting for extra bytes (gzip) */ + const NAME = 7; /* i: waiting for end of file name (gzip) */ + const COMMENT = 8; /* i: waiting for end of comment (gzip) */ + const HCRC = 9; /* i: waiting for header crc (gzip) */ + const DICTID = 10; /* i: waiting for dictionary check value */ + const DICT = 11; /* waiting for inflateSetDictionary() call */ + const TYPE$1 = 12; /* i: waiting for type bits, including last-flag bit */ + const TYPEDO = 13; /* i: same, but skip check to exit inflate on new block */ + const STORED = 14; /* i: waiting for stored size (length and complement) */ + const COPY_ = 15; /* i/o: same as COPY below, but only first time in */ + const COPY = 16; /* i/o: waiting for input or output to copy stored block */ + const TABLE = 17; /* i: waiting for dynamic block table lengths */ + const LENLENS = 18; /* i: waiting for code length code lengths */ + const CODELENS = 19; /* i: waiting for length/lit and distance code lengths */ + const LEN_ = 20; /* i: same as LEN below, but only first time in */ + const LEN = 21; /* i: waiting for length/lit/eob code */ + const LENEXT = 22; /* i: waiting for length extra bits */ + const DIST = 23; /* i: waiting for distance code */ + const DISTEXT = 24; /* i: waiting for distance extra bits */ + const MATCH = 25; /* o: waiting for output space to copy string */ + const LIT = 26; /* o: waiting for output space to write literal */ + const CHECK = 27; /* i: waiting for 32-bit check value */ + const LENGTH = 28; /* i: waiting for 32-bit length (gzip) */ + const DONE = 29; /* finished check, done -- remain here until reset */ + const BAD$1 = 30; /* got a data error -- remain here until reset */ + //const MEM = 31; /* got an inflate() memory error -- remain here until reset */ + const SYNC = 32; /* looking for synchronization bytes to restart inflate() */ + + /* ===========================================================================*/ + + + + const ENOUGH_LENS$1 = 852; + const ENOUGH_DISTS$1 = 592; + + + function zswap32(q) { + return (((q >>> 24) & 0xff) + + ((q >>> 8) & 0xff00) + + ((q & 0xff00) << 8) + + ((q & 0xff) << 24)); + } + + + class InflateState { + constructor() { + this.mode = 0; /* current inflate mode */ + this.last = false; /* true if processing last block */ + this.wrap = 0; /* bit 0 true for zlib, bit 1 true for gzip */ + this.havedict = false; /* true if dictionary provided */ + this.flags = 0; /* gzip header method and flags (0 if zlib) */ + this.dmax = 0; /* zlib header max distance (INFLATE_STRICT) */ + this.check = 0; /* protected copy of check value */ + this.total = 0; /* protected copy of output count */ + // TODO: may be {} + this.head = null; /* where to save gzip header information */ + + /* sliding window */ + this.wbits = 0; /* log base 2 of requested window size */ + this.wsize = 0; /* window size or zero if not using window */ + this.whave = 0; /* valid bytes in the window */ + this.wnext = 0; /* window write index */ + this.window = null; /* allocated sliding window, if needed */ + + /* bit accumulator */ + this.hold = 0; /* input bit accumulator */ + this.bits = 0; /* number of bits in "in" */ + + /* for string and stored block copying */ + this.length = 0; /* literal or length of data to copy */ + this.offset = 0; /* distance back to copy string from */ + + /* for table and code decoding */ + this.extra = 0; /* extra bits needed */ + + /* fixed and dynamic code tables */ + this.lencode = null; /* starting table for length/literal codes */ + this.distcode = null; /* starting table for distance codes */ + this.lenbits = 0; /* index bits for lencode */ + this.distbits = 0; /* index bits for distcode */ + + /* dynamic table building */ + this.ncode = 0; /* number of code length code lengths */ + this.nlen = 0; /* number of length code lengths */ + this.ndist = 0; /* number of distance code lengths */ + this.have = 0; /* number of code lengths in lens[] */ + this.next = null; /* next available space in codes[] */ + + this.lens = new Buf16(320); /* temporary storage for code lengths */ + this.work = new Buf16(288); /* work area for code table building */ + + /* + because we don't have pointers in js, we use lencode and distcode directly + as buffers so we don't need codes + */ + //this.codes = new utils.Buf32(ENOUGH); /* space for code tables */ + this.lendyn = null; /* dynamic table for length/literal codes (JS specific) */ + this.distdyn = null; /* dynamic table for distance codes (JS specific) */ + this.sane = 0; /* if false, allow invalid distance too far */ + this.back = 0; /* bits back of last unprocessed length/lit */ + this.was = 0; /* initial length of match */ + } + } + + function inflateResetKeep(strm) { + let state; + + if (!strm || !strm.state) { return Z_STREAM_ERROR; } + state = strm.state; + strm.total_in = strm.total_out = state.total = 0; + strm.msg = ''; /*Z_NULL*/ + if (state.wrap) { /* to support ill-conceived Java test suite */ + strm.adler = state.wrap & 1; + } + state.mode = HEAD; + state.last = 0; + state.havedict = 0; + state.dmax = 32768; + state.head = null/*Z_NULL*/; + state.hold = 0; + state.bits = 0; + //state.lencode = state.distcode = state.next = state.codes; + state.lencode = state.lendyn = new Buf32(ENOUGH_LENS$1); + state.distcode = state.distdyn = new Buf32(ENOUGH_DISTS$1); + + state.sane = 1; + state.back = -1; + //Tracev((stderr, "inflate: reset\n")); + return Z_OK; + } + + function inflateReset(strm) { + let state; + + if (!strm || !strm.state) { return Z_STREAM_ERROR; } + state = strm.state; + state.wsize = 0; + state.whave = 0; + state.wnext = 0; + return inflateResetKeep(strm); + + } + + function inflateReset2(strm, windowBits) { + let wrap; + let state; + + /* get the state */ + if (!strm || !strm.state) { return Z_STREAM_ERROR; } + state = strm.state; + + /* extract wrap request from windowBits parameter */ + if (windowBits < 0) { + wrap = 0; + windowBits = -windowBits; + } + else { + wrap = (windowBits >> 4) + 1; + if (windowBits < 48) { + windowBits &= 15; + } + } + + /* set number of window bits, free window if different */ + if (windowBits && (windowBits < 8 || windowBits > 15)) { + return Z_STREAM_ERROR; + } + if (state.window !== null && state.wbits !== windowBits) { + state.window = null; + } + + /* update state and reset the rest of it */ + state.wrap = wrap; + state.wbits = windowBits; + return inflateReset(strm); + } + + function inflateInit2(strm, windowBits) { + let ret; + let state; + + if (!strm) { return Z_STREAM_ERROR; } + //strm.msg = Z_NULL; /* in case we return an error */ + + state = new InflateState(); + + //if (state === Z_NULL) return Z_MEM_ERROR; + //Tracev((stderr, "inflate: allocated\n")); + strm.state = state; + state.window = null/*Z_NULL*/; + ret = inflateReset2(strm, windowBits); + if (ret !== Z_OK) { + strm.state = null/*Z_NULL*/; + } + return ret; + } + + + /* + Return state with length and distance decoding tables and index sizes set to + fixed code decoding. Normally this returns fixed tables from inffixed.h. + If BUILDFIXED is defined, then instead this routine builds the tables the + first time it's called, and returns those tables the first time and + thereafter. This reduces the size of the code by about 2K bytes, in + exchange for a little execution time. However, BUILDFIXED should not be + used for threaded applications, since the rewriting of the tables and virgin + may not be thread-safe. + */ + let virgin = true; + + let lenfix, distfix; // We have no pointers in JS, so keep tables separate + + function fixedtables(state) { + /* build fixed huffman tables if first call (may not be thread safe) */ + if (virgin) { + let sym; + + lenfix = new Buf32(512); + distfix = new Buf32(32); + + /* literal/length table */ + sym = 0; + while (sym < 144) { state.lens[sym++] = 8; } + while (sym < 256) { state.lens[sym++] = 9; } + while (sym < 280) { state.lens[sym++] = 7; } + while (sym < 288) { state.lens[sym++] = 8; } + + inflate_table(LENS$1, state.lens, 0, 288, lenfix, 0, state.work, { bits: 9 }); + + /* distance table */ + sym = 0; + while (sym < 32) { state.lens[sym++] = 5; } + + inflate_table(DISTS$1, state.lens, 0, 32, distfix, 0, state.work, { bits: 5 }); + + /* do this just once */ + virgin = false; + } + + state.lencode = lenfix; + state.lenbits = 9; + state.distcode = distfix; + state.distbits = 5; + } + + + /* + Update the window with the last wsize (normally 32K) bytes written before + returning. If window does not exist yet, create it. This is only called + when a window is already in use, or when output has been written during this + inflate call, but the end of the deflate stream has not been reached yet. + It is also called to create a window for dictionary data when a dictionary + is loaded. + + Providing output buffers larger than 32K to inflate() should provide a speed + advantage, since only the last 32K of output is copied to the sliding window + upon return from inflate(), and since all distances after the first 32K of + output will fall in the output data, making match copies simpler and faster. + The advantage may be dependent on the size of the processor's data caches. + */ + function updatewindow(strm, src, end, copy) { + let dist; + const state = strm.state; + + /* if it hasn't been done already, allocate space for the window */ + if (state.window === null) { + state.wsize = 1 << state.wbits; + state.wnext = 0; + state.whave = 0; + + state.window = new Buf8(state.wsize); + } + + /* copy state->wsize or less output bytes into the circular window */ + if (copy >= state.wsize) { + arraySet(state.window, src, end - state.wsize, state.wsize, 0); + state.wnext = 0; + state.whave = state.wsize; + } + else { + dist = state.wsize - state.wnext; + if (dist > copy) { + dist = copy; + } + //zmemcpy(state->window + state->wnext, end - copy, dist); + arraySet(state.window, src, end - copy, dist, state.wnext); + copy -= dist; + if (copy) { + //zmemcpy(state->window, end - copy, copy); + arraySet(state.window, src, end - copy, copy, 0); + state.wnext = copy; + state.whave = state.wsize; + } + else { + state.wnext += dist; + if (state.wnext === state.wsize) { state.wnext = 0; } + if (state.whave < state.wsize) { state.whave += dist; } + } + } + return 0; + } + + function inflate(strm, flush) { + let state; + let input, output; // input/output buffers + let next; /* next input INDEX */ + let put; /* next output INDEX */ + let have, left; /* available input and output */ + let hold; /* bit buffer */ + let bits; /* bits in bit buffer */ + let _in, _out; /* save starting available input and output */ + let copy; /* number of stored or match bytes to copy */ + let from; /* where to copy match bytes from */ + let from_source; + let here = 0; /* current decoding table entry */ + let here_bits, here_op, here_val; // paked "here" denormalized (JS specific) + //var last; /* parent table entry */ + let last_bits, last_op, last_val; // paked "last" denormalized (JS specific) + let len; /* length to copy for repeats, bits to drop */ + let ret; /* return code */ + let hbuf = new Buf8(4); /* buffer for gzip header crc calculation */ + let opts; + + let n; // temporary var for NEED_BITS + + const order = /* permutation of code lengths */ + [ 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15 ]; + + + if (!strm || !strm.state || !strm.output || + (!strm.input && strm.avail_in !== 0)) { + return Z_STREAM_ERROR; + } + + state = strm.state; + if (state.mode === TYPE$1) { state.mode = TYPEDO; } /* skip check */ + + + //--- LOAD() --- + put = strm.next_out; + output = strm.output; + left = strm.avail_out; + next = strm.next_in; + input = strm.input; + have = strm.avail_in; + hold = state.hold; + bits = state.bits; + //--- + + _in = have; + _out = left; + ret = Z_OK; + + inf_leave: // goto emulation + for (;;) { + switch (state.mode) { + case HEAD: + if (state.wrap === 0) { + state.mode = TYPEDO; + break; + } + //=== NEEDBITS(16); + while (bits < 16) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + if ((state.wrap & 2) && hold === 0x8b1f) { /* gzip header */ + state.check = 0/*crc32(0L, Z_NULL, 0)*/; + //=== CRC2(state.check, hold); + hbuf[0] = hold & 0xff; + hbuf[1] = (hold >>> 8) & 0xff; + state.check = crc32(state.check, hbuf, 2, 0); + //===// + + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + state.mode = FLAGS; + break; + } + state.flags = 0; /* expect zlib header */ + if (state.head) { + state.head.done = false; + } + if (!(state.wrap & 1) || /* check if zlib header allowed */ + (((hold & 0xff)/*BITS(8)*/ << 8) + (hold >> 8)) % 31) { + strm.msg = 'incorrect header check'; + state.mode = BAD$1; + break; + } + if ((hold & 0x0f)/*BITS(4)*/ !== Z_DEFLATED) { + strm.msg = 'unknown compression method'; + state.mode = BAD$1; + break; + } + //--- DROPBITS(4) ---// + hold >>>= 4; + bits -= 4; + //---// + len = (hold & 0x0f)/*BITS(4)*/ + 8; + if (state.wbits === 0) { + state.wbits = len; + } + else if (len > state.wbits) { + strm.msg = 'invalid window size'; + state.mode = BAD$1; + break; + } + state.dmax = 1 << len; + //Tracev((stderr, "inflate: zlib header ok\n")); + strm.adler = state.check = 1/*adler32(0L, Z_NULL, 0)*/; + state.mode = hold & 0x200 ? DICTID : TYPE$1; + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + break; + case FLAGS: + //=== NEEDBITS(16); */ + while (bits < 16) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + state.flags = hold; + if ((state.flags & 0xff) !== Z_DEFLATED) { + strm.msg = 'unknown compression method'; + state.mode = BAD$1; + break; + } + if (state.flags & 0xe000) { + strm.msg = 'unknown header flags set'; + state.mode = BAD$1; + break; + } + if (state.head) { + state.head.text = ((hold >> 8) & 1); + } + if (state.flags & 0x0200) { + //=== CRC2(state.check, hold); + hbuf[0] = hold & 0xff; + hbuf[1] = (hold >>> 8) & 0xff; + state.check = crc32(state.check, hbuf, 2, 0); + //===// + } + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + state.mode = TIME; + /* falls through */ + case TIME: + //=== NEEDBITS(32); */ + while (bits < 32) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + if (state.head) { + state.head.time = hold; + } + if (state.flags & 0x0200) { + //=== CRC4(state.check, hold) + hbuf[0] = hold & 0xff; + hbuf[1] = (hold >>> 8) & 0xff; + hbuf[2] = (hold >>> 16) & 0xff; + hbuf[3] = (hold >>> 24) & 0xff; + state.check = crc32(state.check, hbuf, 4, 0); + //=== + } + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + state.mode = OS; + /* falls through */ + case OS: + //=== NEEDBITS(16); */ + while (bits < 16) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + if (state.head) { + state.head.xflags = (hold & 0xff); + state.head.os = (hold >> 8); + } + if (state.flags & 0x0200) { + //=== CRC2(state.check, hold); + hbuf[0] = hold & 0xff; + hbuf[1] = (hold >>> 8) & 0xff; + state.check = crc32(state.check, hbuf, 2, 0); + //===// + } + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + state.mode = EXLEN; + /* falls through */ + case EXLEN: + if (state.flags & 0x0400) { + //=== NEEDBITS(16); */ + while (bits < 16) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + state.length = hold; + if (state.head) { + state.head.extra_len = hold; + } + if (state.flags & 0x0200) { + //=== CRC2(state.check, hold); + hbuf[0] = hold & 0xff; + hbuf[1] = (hold >>> 8) & 0xff; + state.check = crc32(state.check, hbuf, 2, 0); + //===// + } + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + } + else if (state.head) { + state.head.extra = null/*Z_NULL*/; + } + state.mode = EXTRA; + /* falls through */ + case EXTRA: + if (state.flags & 0x0400) { + copy = state.length; + if (copy > have) { copy = have; } + if (copy) { + if (state.head) { + len = state.head.extra_len - state.length; + if (!state.head.extra) { + // Use untyped array for more convenient processing later + state.head.extra = new Array(state.head.extra_len); + } + arraySet( + state.head.extra, + input, + next, + // extra field is limited to 65536 bytes + // - no need for additional size check + copy, + /*len + copy > state.head.extra_max - len ? state.head.extra_max : copy,*/ + len + ); + //zmemcpy(state.head.extra + len, next, + // len + copy > state.head.extra_max ? + // state.head.extra_max - len : copy); + } + if (state.flags & 0x0200) { + state.check = crc32(state.check, input, copy, next); + } + have -= copy; + next += copy; + state.length -= copy; + } + if (state.length) { break inf_leave; } + } + state.length = 0; + state.mode = NAME; + /* falls through */ + case NAME: + if (state.flags & 0x0800) { + if (have === 0) { break inf_leave; } + copy = 0; + do { + // TODO: 2 or 1 bytes? + len = input[next + copy++]; + /* use constant limit because in js we should not preallocate memory */ + if (state.head && len && + (state.length < 65536 /*state.head.name_max*/)) { + state.head.name += String.fromCharCode(len); + } + } while (len && copy < have); + + if (state.flags & 0x0200) { + state.check = crc32(state.check, input, copy, next); + } + have -= copy; + next += copy; + if (len) { break inf_leave; } + } + else if (state.head) { + state.head.name = null; + } + state.length = 0; + state.mode = COMMENT; + /* falls through */ + case COMMENT: + if (state.flags & 0x1000) { + if (have === 0) { break inf_leave; } + copy = 0; + do { + len = input[next + copy++]; + /* use constant limit because in js we should not preallocate memory */ + if (state.head && len && + (state.length < 65536 /*state.head.comm_max*/)) { + state.head.comment += String.fromCharCode(len); + } + } while (len && copy < have); + if (state.flags & 0x0200) { + state.check = crc32(state.check, input, copy, next); + } + have -= copy; + next += copy; + if (len) { break inf_leave; } + } + else if (state.head) { + state.head.comment = null; + } + state.mode = HCRC; + /* falls through */ + case HCRC: + if (state.flags & 0x0200) { + //=== NEEDBITS(16); */ + while (bits < 16) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + if (hold !== (state.check & 0xffff)) { + strm.msg = 'header crc mismatch'; + state.mode = BAD$1; + break; + } + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + } + if (state.head) { + state.head.hcrc = ((state.flags >> 9) & 1); + state.head.done = true; + } + strm.adler = state.check = 0; + state.mode = TYPE$1; + break; + case DICTID: + //=== NEEDBITS(32); */ + while (bits < 32) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + strm.adler = state.check = zswap32(hold); + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + state.mode = DICT; + /* falls through */ + case DICT: + if (state.havedict === 0) { + //--- RESTORE() --- + strm.next_out = put; + strm.avail_out = left; + strm.next_in = next; + strm.avail_in = have; + state.hold = hold; + state.bits = bits; + //--- + return Z_NEED_DICT; + } + strm.adler = state.check = 1/*adler32(0L, Z_NULL, 0)*/; + state.mode = TYPE$1; + /* falls through */ + case TYPE$1: + if (flush === Z_BLOCK || flush === Z_TREES) { break inf_leave; } + /* falls through */ + case TYPEDO: + if (state.last) { + //--- BYTEBITS() ---// + hold >>>= bits & 7; + bits -= bits & 7; + //---// + state.mode = CHECK; + break; + } + //=== NEEDBITS(3); */ + while (bits < 3) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + state.last = (hold & 0x01)/*BITS(1)*/; + //--- DROPBITS(1) ---// + hold >>>= 1; + bits -= 1; + //---// + + switch ((hold & 0x03)/*BITS(2)*/) { + case 0: /* stored block */ + //Tracev((stderr, "inflate: stored block%s\n", + // state.last ? " (last)" : "")); + state.mode = STORED; + break; + case 1: /* fixed block */ + fixedtables(state); + //Tracev((stderr, "inflate: fixed codes block%s\n", + // state.last ? " (last)" : "")); + state.mode = LEN_; /* decode codes */ + if (flush === Z_TREES) { + //--- DROPBITS(2) ---// + hold >>>= 2; + bits -= 2; + //---// + break inf_leave; + } + break; + case 2: /* dynamic block */ + //Tracev((stderr, "inflate: dynamic codes block%s\n", + // state.last ? " (last)" : "")); + state.mode = TABLE; + break; + case 3: + strm.msg = 'invalid block type'; + state.mode = BAD$1; + } + //--- DROPBITS(2) ---// + hold >>>= 2; + bits -= 2; + //---// + break; + case STORED: + //--- BYTEBITS() ---// /* go to byte boundary */ + hold >>>= bits & 7; + bits -= bits & 7; + //---// + //=== NEEDBITS(32); */ + while (bits < 32) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + if ((hold & 0xffff) !== ((hold >>> 16) ^ 0xffff)) { + strm.msg = 'invalid stored block lengths'; + state.mode = BAD$1; + break; + } + state.length = hold & 0xffff; + //Tracev((stderr, "inflate: stored length %u\n", + // state.length)); + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + state.mode = COPY_; + if (flush === Z_TREES) { break inf_leave; } + /* falls through */ + case COPY_: + state.mode = COPY; + /* falls through */ + case COPY: + copy = state.length; + if (copy) { + if (copy > have) { copy = have; } + if (copy > left) { copy = left; } + if (copy === 0) { break inf_leave; } + //--- zmemcpy(put, next, copy); --- + arraySet(output, input, next, copy, put); + //---// + have -= copy; + next += copy; + left -= copy; + put += copy; + state.length -= copy; + break; + } + //Tracev((stderr, "inflate: stored end\n")); + state.mode = TYPE$1; + break; + case TABLE: + //=== NEEDBITS(14); */ + while (bits < 14) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + state.nlen = (hold & 0x1f)/*BITS(5)*/ + 257; + //--- DROPBITS(5) ---// + hold >>>= 5; + bits -= 5; + //---// + state.ndist = (hold & 0x1f)/*BITS(5)*/ + 1; + //--- DROPBITS(5) ---// + hold >>>= 5; + bits -= 5; + //---// + state.ncode = (hold & 0x0f)/*BITS(4)*/ + 4; + //--- DROPBITS(4) ---// + hold >>>= 4; + bits -= 4; + //---// + //#ifndef PKZIP_BUG_WORKAROUND + if (state.nlen > 286 || state.ndist > 30) { + strm.msg = 'too many length or distance symbols'; + state.mode = BAD$1; + break; + } + //#endif + //Tracev((stderr, "inflate: table sizes ok\n")); + state.have = 0; + state.mode = LENLENS; + /* falls through */ + case LENLENS: + while (state.have < state.ncode) { + //=== NEEDBITS(3); + while (bits < 3) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + state.lens[order[state.have++]] = (hold & 0x07);//BITS(3); + //--- DROPBITS(3) ---// + hold >>>= 3; + bits -= 3; + //---// + } + while (state.have < 19) { + state.lens[order[state.have++]] = 0; + } + // We have separate tables & no pointers. 2 commented lines below not needed. + //state.next = state.codes; + //state.lencode = state.next; + // Switch to use dynamic table + state.lencode = state.lendyn; + state.lenbits = 7; + + opts = { bits: state.lenbits }; + ret = inflate_table(CODES$1, state.lens, 0, 19, state.lencode, 0, state.work, opts); + state.lenbits = opts.bits; + + if (ret) { + strm.msg = 'invalid code lengths set'; + state.mode = BAD$1; + break; + } + //Tracev((stderr, "inflate: code lengths ok\n")); + state.have = 0; + state.mode = CODELENS; + /* falls through */ + case CODELENS: + while (state.have < state.nlen + state.ndist) { + for (;;) { + here = state.lencode[hold & ((1 << state.lenbits) - 1)];/*BITS(state.lenbits)*/ + here_bits = here >>> 24; + here_op = (here >>> 16) & 0xff; + here_val = here & 0xffff; + + if ((here_bits) <= bits) { break; } + //--- PULLBYTE() ---// + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + //---// + } + if (here_val < 16) { + //--- DROPBITS(here.bits) ---// + hold >>>= here_bits; + bits -= here_bits; + //---// + state.lens[state.have++] = here_val; + } + else { + if (here_val === 16) { + //=== NEEDBITS(here.bits + 2); + n = here_bits + 2; + while (bits < n) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + //--- DROPBITS(here.bits) ---// + hold >>>= here_bits; + bits -= here_bits; + //---// + if (state.have === 0) { + strm.msg = 'invalid bit length repeat'; + state.mode = BAD$1; + break; + } + len = state.lens[state.have - 1]; + copy = 3 + (hold & 0x03);//BITS(2); + //--- DROPBITS(2) ---// + hold >>>= 2; + bits -= 2; + //---// + } + else if (here_val === 17) { + //=== NEEDBITS(here.bits + 3); + n = here_bits + 3; + while (bits < n) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + //--- DROPBITS(here.bits) ---// + hold >>>= here_bits; + bits -= here_bits; + //---// + len = 0; + copy = 3 + (hold & 0x07);//BITS(3); + //--- DROPBITS(3) ---// + hold >>>= 3; + bits -= 3; + //---// + } + else { + //=== NEEDBITS(here.bits + 7); + n = here_bits + 7; + while (bits < n) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + //--- DROPBITS(here.bits) ---// + hold >>>= here_bits; + bits -= here_bits; + //---// + len = 0; + copy = 11 + (hold & 0x7f);//BITS(7); + //--- DROPBITS(7) ---// + hold >>>= 7; + bits -= 7; + //---// + } + if (state.have + copy > state.nlen + state.ndist) { + strm.msg = 'invalid bit length repeat'; + state.mode = BAD$1; + break; + } + while (copy--) { + state.lens[state.have++] = len; + } + } + } + + /* handle error breaks in while */ + if (state.mode === BAD$1) { break; } + + /* check for end-of-block code (better have one) */ + if (state.lens[256] === 0) { + strm.msg = 'invalid code -- missing end-of-block'; + state.mode = BAD$1; + break; + } + + /* build code tables -- note: do not change the lenbits or distbits + values here (9 and 6) without reading the comments in inftrees.h + concerning the ENOUGH constants, which depend on those values */ + state.lenbits = 9; + + opts = { bits: state.lenbits }; + ret = inflate_table(LENS$1, state.lens, 0, state.nlen, state.lencode, 0, state.work, opts); + // We have separate tables & no pointers. 2 commented lines below not needed. + // state.next_index = opts.table_index; + state.lenbits = opts.bits; + // state.lencode = state.next; + + if (ret) { + strm.msg = 'invalid literal/lengths set'; + state.mode = BAD$1; + break; + } + + state.distbits = 6; + //state.distcode.copy(state.codes); + // Switch to use dynamic table + state.distcode = state.distdyn; + opts = { bits: state.distbits }; + ret = inflate_table(DISTS$1, state.lens, state.nlen, state.ndist, state.distcode, 0, state.work, opts); + // We have separate tables & no pointers. 2 commented lines below not needed. + // state.next_index = opts.table_index; + state.distbits = opts.bits; + // state.distcode = state.next; + + if (ret) { + strm.msg = 'invalid distances set'; + state.mode = BAD$1; + break; + } + //Tracev((stderr, 'inflate: codes ok\n')); + state.mode = LEN_; + if (flush === Z_TREES) { break inf_leave; } + /* falls through */ + case LEN_: + state.mode = LEN; + /* falls through */ + case LEN: + if (have >= 6 && left >= 258) { + //--- RESTORE() --- + strm.next_out = put; + strm.avail_out = left; + strm.next_in = next; + strm.avail_in = have; + state.hold = hold; + state.bits = bits; + //--- + inflate_fast(strm, _out); + //--- LOAD() --- + put = strm.next_out; + output = strm.output; + left = strm.avail_out; + next = strm.next_in; + input = strm.input; + have = strm.avail_in; + hold = state.hold; + bits = state.bits; + //--- + + if (state.mode === TYPE$1) { + state.back = -1; + } + break; + } + state.back = 0; + for (;;) { + here = state.lencode[hold & ((1 << state.lenbits) - 1)]; /*BITS(state.lenbits)*/ + here_bits = here >>> 24; + here_op = (here >>> 16) & 0xff; + here_val = here & 0xffff; + + if (here_bits <= bits) { break; } + //--- PULLBYTE() ---// + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + //---// + } + if (here_op && (here_op & 0xf0) === 0) { + last_bits = here_bits; + last_op = here_op; + last_val = here_val; + for (;;) { + here = state.lencode[last_val + + ((hold & ((1 << (last_bits + last_op)) - 1))/*BITS(last.bits + last.op)*/ >> last_bits)]; + here_bits = here >>> 24; + here_op = (here >>> 16) & 0xff; + here_val = here & 0xffff; + + if ((last_bits + here_bits) <= bits) { break; } + //--- PULLBYTE() ---// + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + //---// + } + //--- DROPBITS(last.bits) ---// + hold >>>= last_bits; + bits -= last_bits; + //---// + state.back += last_bits; + } + //--- DROPBITS(here.bits) ---// + hold >>>= here_bits; + bits -= here_bits; + //---// + state.back += here_bits; + state.length = here_val; + if (here_op === 0) { + //Tracevv((stderr, here.val >= 0x20 && here.val < 0x7f ? + // "inflate: literal '%c'\n" : + // "inflate: literal 0x%02x\n", here.val)); + state.mode = LIT; + break; + } + if (here_op & 32) { + //Tracevv((stderr, "inflate: end of block\n")); + state.back = -1; + state.mode = TYPE$1; + break; + } + if (here_op & 64) { + strm.msg = 'invalid literal/length code'; + state.mode = BAD$1; + break; + } + state.extra = here_op & 15; + state.mode = LENEXT; + /* falls through */ + case LENEXT: + if (state.extra) { + //=== NEEDBITS(state.extra); + n = state.extra; + while (bits < n) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + state.length += hold & ((1 << state.extra) - 1)/*BITS(state.extra)*/; + //--- DROPBITS(state.extra) ---// + hold >>>= state.extra; + bits -= state.extra; + //---// + state.back += state.extra; + } + //Tracevv((stderr, "inflate: length %u\n", state.length)); + state.was = state.length; + state.mode = DIST; + /* falls through */ + case DIST: + for (;;) { + here = state.distcode[hold & ((1 << state.distbits) - 1)];/*BITS(state.distbits)*/ + here_bits = here >>> 24; + here_op = (here >>> 16) & 0xff; + here_val = here & 0xffff; + + if ((here_bits) <= bits) { break; } + //--- PULLBYTE() ---// + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + //---// + } + if ((here_op & 0xf0) === 0) { + last_bits = here_bits; + last_op = here_op; + last_val = here_val; + for (;;) { + here = state.distcode[last_val + + ((hold & ((1 << (last_bits + last_op)) - 1))/*BITS(last.bits + last.op)*/ >> last_bits)]; + here_bits = here >>> 24; + here_op = (here >>> 16) & 0xff; + here_val = here & 0xffff; + + if ((last_bits + here_bits) <= bits) { break; } + //--- PULLBYTE() ---// + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + //---// + } + //--- DROPBITS(last.bits) ---// + hold >>>= last_bits; + bits -= last_bits; + //---// + state.back += last_bits; + } + //--- DROPBITS(here.bits) ---// + hold >>>= here_bits; + bits -= here_bits; + //---// + state.back += here_bits; + if (here_op & 64) { + strm.msg = 'invalid distance code'; + state.mode = BAD$1; + break; + } + state.offset = here_val; + state.extra = (here_op) & 15; + state.mode = DISTEXT; + /* falls through */ + case DISTEXT: + if (state.extra) { + //=== NEEDBITS(state.extra); + n = state.extra; + while (bits < n) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + state.offset += hold & ((1 << state.extra) - 1)/*BITS(state.extra)*/; + //--- DROPBITS(state.extra) ---// + hold >>>= state.extra; + bits -= state.extra; + //---// + state.back += state.extra; + } + //#ifdef INFLATE_STRICT + if (state.offset > state.dmax) { + strm.msg = 'invalid distance too far back'; + state.mode = BAD$1; + break; + } + //#endif + //Tracevv((stderr, "inflate: distance %u\n", state.offset)); + state.mode = MATCH; + /* falls through */ + case MATCH: + if (left === 0) { break inf_leave; } + copy = _out - left; + if (state.offset > copy) { /* copy from window */ + copy = state.offset - copy; + if (copy > state.whave) { + if (state.sane) { + strm.msg = 'invalid distance too far back'; + state.mode = BAD$1; + break; + } + // (!) This block is disabled in zlib defaults, + // don't enable it for binary compatibility + //#ifdef INFLATE_ALLOW_INVALID_DISTANCE_TOOFAR_ARRR + // Trace((stderr, "inflate.c too far\n")); + // copy -= state.whave; + // if (copy > state.length) { copy = state.length; } + // if (copy > left) { copy = left; } + // left -= copy; + // state.length -= copy; + // do { + // output[put++] = 0; + // } while (--copy); + // if (state.length === 0) { state.mode = LEN; } + // break; + //#endif + } + if (copy > state.wnext) { + copy -= state.wnext; + from = state.wsize - copy; + } + else { + from = state.wnext - copy; + } + if (copy > state.length) { copy = state.length; } + from_source = state.window; + } + else { /* copy from output */ + from_source = output; + from = put - state.offset; + copy = state.length; + } + if (copy > left) { copy = left; } + left -= copy; + state.length -= copy; + do { + output[put++] = from_source[from++]; + } while (--copy); + if (state.length === 0) { state.mode = LEN; } + break; + case LIT: + if (left === 0) { break inf_leave; } + output[put++] = state.length; + left--; + state.mode = LEN; + break; + case CHECK: + if (state.wrap) { + //=== NEEDBITS(32); + while (bits < 32) { + if (have === 0) { break inf_leave; } + have--; + // Use '|' instead of '+' to make sure that result is signed + hold |= input[next++] << bits; + bits += 8; + } + //===// + _out -= left; + strm.total_out += _out; + state.total += _out; + if (_out) { + strm.adler = state.check = + /*UPDATE(state.check, put - _out, _out);*/ + (state.flags ? crc32(state.check, output, _out, put - _out) : adler32(state.check, output, _out, put - _out)); + + } + _out = left; + // NB: crc32 stored as signed 32-bit int, zswap32 returns signed too + if ((state.flags ? hold : zswap32(hold)) !== state.check) { + strm.msg = 'incorrect data check'; + state.mode = BAD$1; + break; + } + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + //Tracev((stderr, "inflate: check matches trailer\n")); + } + state.mode = LENGTH; + /* falls through */ + case LENGTH: + if (state.wrap && state.flags) { + //=== NEEDBITS(32); + while (bits < 32) { + if (have === 0) { break inf_leave; } + have--; + hold += input[next++] << bits; + bits += 8; + } + //===// + if (hold !== (state.total & 0xffffffff)) { + strm.msg = 'incorrect length check'; + state.mode = BAD$1; + break; + } + //=== INITBITS(); + hold = 0; + bits = 0; + //===// + //Tracev((stderr, "inflate: length matches trailer\n")); + } + state.mode = DONE; + /* falls through */ + case DONE: + ret = Z_STREAM_END; + break inf_leave; + case BAD$1: + ret = Z_DATA_ERROR; + break inf_leave; + // case MEM: + // return Z_MEM_ERROR; + case SYNC: + /* falls through */ + default: + return Z_STREAM_ERROR; + } + } + + // inf_leave <- here is real place for "goto inf_leave", emulated via "break inf_leave" + + /* + Return from inflate(), updating the total counts and the check value. + If there was no progress during the inflate() call, return a buffer + error. Call updatewindow() to create and/or update the window state. + Note: a memory error from inflate() is non-recoverable. + */ + + //--- RESTORE() --- + strm.next_out = put; + strm.avail_out = left; + strm.next_in = next; + strm.avail_in = have; + state.hold = hold; + state.bits = bits; + //--- + + if (state.wsize || (_out !== strm.avail_out && state.mode < BAD$1 && + (state.mode < CHECK || flush !== Z_FINISH))) { + if (updatewindow(strm, strm.output, strm.next_out, _out - strm.avail_out)) ; + } + _in -= strm.avail_in; + _out -= strm.avail_out; + strm.total_in += _in; + strm.total_out += _out; + state.total += _out; + if (state.wrap && _out) { + strm.adler = state.check = /*UPDATE(state.check, strm.next_out - _out, _out);*/ + (state.flags ? crc32(state.check, output, _out, strm.next_out - _out) : adler32(state.check, output, _out, strm.next_out - _out)); + } + strm.data_type = state.bits + (state.last ? 64 : 0) + + (state.mode === TYPE$1 ? 128 : 0) + + (state.mode === LEN_ || state.mode === COPY_ ? 256 : 0); + if (((_in === 0 && _out === 0) || flush === Z_FINISH) && ret === Z_OK) { + ret = Z_BUF_ERROR; + } + return ret; + } + + function inflateEnd(strm) { + + if (!strm || !strm.state /*|| strm->zfree == (free_func)0*/) { + return Z_STREAM_ERROR; + } + + const state = strm.state; + if (state.window) { + state.window = null; + } + strm.state = null; + return Z_OK; + } + + function inflateGetHeader(strm, head) { + let state; + + /* check state */ + if (!strm || !strm.state) { return Z_STREAM_ERROR; } + state = strm.state; + if ((state.wrap & 2) === 0) { return Z_STREAM_ERROR; } + + /* save header structure */ + state.head = head; + head.done = false; + return Z_OK; + } + + function inflateSetDictionary(strm, dictionary) { + const dictLength = dictionary.length; + + let state; + let dictid; + + /* check state */ + if (!strm /* == Z_NULL */ || !strm.state /* == Z_NULL */) { return Z_STREAM_ERROR; } + state = strm.state; + + if (state.wrap !== 0 && state.mode !== DICT) { + return Z_STREAM_ERROR; + } + + /* check for correct dictionary identifier */ + if (state.mode === DICT) { + dictid = 1; /* adler32(0, null, 0)*/ + /* dictid = adler32(dictid, dictionary, dictLength); */ + dictid = adler32(dictid, dictionary, dictLength, 0); + if (dictid !== state.check) { + return Z_DATA_ERROR; + } + } + /* copy dictionary to window using updatewindow(), which will amend the + existing dictionary if appropriate */ + updatewindow(strm, dictionary, dictLength, dictLength); + // if (ret) { + // state.mode = MEM; + // return Z_MEM_ERROR; + // } + state.havedict = 1; + // Tracev((stderr, "inflate: dictionary set\n")); + return Z_OK; + } + + /* Not implemented + exports.inflateCopy = inflateCopy; + exports.inflateGetDictionary = inflateGetDictionary; + exports.inflateMark = inflateMark; + exports.inflatePrime = inflatePrime; + exports.inflateSync = inflateSync; + exports.inflateSyncPoint = inflateSyncPoint; + exports.inflateUndermine = inflateUndermine; + */ + + // (C) 1995-2013 Jean-loup Gailly and Mark Adler + // (C) 2014-2017 Vitaly Puzrin and Andrey Tupitsin + // + // This software is provided 'as-is', without any express or implied + // warranty. In no event will the authors be held liable for any damages + // arising from the use of this software. + // + // Permission is granted to anyone to use this software for any purpose, + // including commercial applications, and to alter it and redistribute it + // freely, subject to the following restrictions: + // + // 1. The origin of this software must not be misrepresented; you must not + // claim that you wrote the original software. If you use this software + // in a product, an acknowledgment in the product documentation would be + // appreciated but is not required. + // 2. Altered source versions must be plainly marked as such, and must not be + // misrepresented as being the original software. + // 3. This notice may not be removed or altered from any source distribution. + + class GZheader { + constructor() { + /* true if compressed data believed to be text */ + this.text = 0; + /* modification time */ + this.time = 0; + /* extra flags (not used when writing a gzip file) */ + this.xflags = 0; + /* operating system */ + this.os = 0; + /* pointer to extra field or Z_NULL if none */ + this.extra = null; + /* extra field length (valid if extra != Z_NULL) */ + this.extra_len = 0; // Actually, we don't need it in JS, + // but leave for few code modifications + + // + // Setup limits is not necessary because in js we should not preallocate memory + // for inflate use constant limit in 65536 bytes + // + + /* space at extra (only when reading header) */ + // this.extra_max = 0; + /* pointer to zero-terminated file name or Z_NULL */ + this.name = ''; + /* space at name (only when reading header) */ + // this.name_max = 0; + /* pointer to zero-terminated comment or Z_NULL */ + this.comment = ''; + /* space at comment (only when reading header) */ + // this.comm_max = 0; + /* true if there was or will be a header crc */ + this.hcrc = 0; + /* true when done reading gzip header (not used when writing a gzip file) */ + this.done = false; + } + } + + /** + * class Inflate + * + * Generic JS-style wrapper for zlib calls. If you don't need + * streaming behaviour - use more simple functions: [[inflate]] + * and [[inflateRaw]]. + **/ + + /* internal + * inflate.chunks -> Array + * + * Chunks of output data, if [[Inflate#onData]] not overridden. + **/ + + /** + * Inflate.result -> Uint8Array|Array|String + * + * Uncompressed result, generated by default [[Inflate#onData]] + * and [[Inflate#onEnd]] handlers. Filled after you push last chunk + * (call [[Inflate#push]] with `Z_FINISH` / `true` param) or if you + * push a chunk with explicit flush (call [[Inflate#push]] with + * `Z_SYNC_FLUSH` param). + **/ + + /** + * Inflate.err -> Number + * + * Error code after inflate finished. 0 (Z_OK) on success. + * Should be checked if broken data possible. + **/ + + /** + * Inflate.msg -> String + * + * Error message, if [[Inflate.err]] != 0 + **/ + + + /** + * new Inflate(options) + * - options (Object): zlib inflate options. + * + * Creates new inflator instance with specified params. Throws exception + * on bad params. Supported options: + * + * - `windowBits` + * - `dictionary` + * + * [http://zlib.net/manual.html#Advanced](http://zlib.net/manual.html#Advanced) + * for more information on these. + * + * Additional options, for internal needs: + * + * - `chunkSize` - size of generated data chunks (16K by default) + * - `raw` (Boolean) - do raw inflate + * - `to` (String) - if equal to 'string', then result will be converted + * from utf8 to utf16 (javascript) string. When string output requested, + * chunk length can differ from `chunkSize`, depending on content. + * + * By default, when no options set, autodetect deflate/gzip data format via + * wrapper header. + * + * ##### Example: + * + * ```javascript + * var pako = void('pako') + * , chunk1 = Uint8Array([1,2,3,4,5,6,7,8,9]) + * , chunk2 = Uint8Array([10,11,12,13,14,15,16,17,18,19]); + * + * var inflate = new pako.Inflate({ level: 3}); + * + * inflate.push(chunk1, false); + * inflate.push(chunk2, true); // true -> last chunk + * + * if (inflate.err) { throw new Error(inflate.err); } + * + * console.log(inflate.result); + * ``` + **/ + class Inflate { + constructor(options) { + this.options = { + chunkSize: 16384, + windowBits: 0, + ...(options || {}) + }; + + const opt = this.options; + + // Force window size for `raw` data, if not set directly, + // because we have no header for autodetect. + if (opt.raw && (opt.windowBits >= 0) && (opt.windowBits < 16)) { + opt.windowBits = -opt.windowBits; + if (opt.windowBits === 0) { opt.windowBits = -15; } + } + + // If `windowBits` not defined (and mode not raw) - set autodetect flag for gzip/deflate + if ((opt.windowBits >= 0) && (opt.windowBits < 16) && + !(options && options.windowBits)) { + opt.windowBits += 32; + } + + // Gzip header has no info about windows size, we can do autodetect only + // for deflate. So, if window size not set, force it to max when gzip possible + if ((opt.windowBits > 15) && (opt.windowBits < 48)) { + // bit 3 (16) -> gzipped data + // bit 4 (32) -> autodetect gzip/deflate + if ((opt.windowBits & 15) === 0) { + opt.windowBits |= 15; + } + } + + this.err = 0; // error code, if happens (0 = Z_OK) + this.msg = ''; // error message + this.ended = false; // used to avoid multiple onEnd() calls + this.chunks = []; // chunks of compressed data + + this.strm = new ZStream(); + this.strm.avail_out = 0; + + let status = inflateInit2( + this.strm, + opt.windowBits + ); + + if (status !== Z_OK) { + throw new Error(msg[status]); + } + + this.header = new GZheader(); + + inflateGetHeader(this.strm, this.header); + + // Setup dictionary + if (opt.dictionary) { + // Convert data if needed + if (typeof opt.dictionary === 'string') { + opt.dictionary = string2buf(opt.dictionary); + } else if (opt.dictionary instanceof ArrayBuffer) { + opt.dictionary = new Uint8Array(opt.dictionary); + } + if (opt.raw) { //In raw mode we need to set the dictionary early + status = inflateSetDictionary(this.strm, opt.dictionary); + if (status !== Z_OK) { + throw new Error(msg[status]); + } + } + } + } + /** + * Inflate#push(data[, mode]) -> Boolean + * - data (Uint8Array|Array|ArrayBuffer|String): input data + * - mode (Number|Boolean): 0..6 for corresponding Z_NO_FLUSH..Z_TREE modes. + * See constants. Skipped or `false` means Z_NO_FLUSH, `true` means Z_FINISH. + * + * Sends input data to inflate pipe, generating [[Inflate#onData]] calls with + * new output chunks. Returns `true` on success. The last data block must have + * mode Z_FINISH (or `true`). That will flush internal pending buffers and call + * [[Inflate#onEnd]]. For interim explicit flushes (without ending the stream) you + * can use mode Z_SYNC_FLUSH, keeping the decompression context. + * + * On fail call [[Inflate#onEnd]] with error code and return false. + * + * We strongly recommend to use `Uint8Array` on input for best speed (output + * format is detected automatically). Also, don't skip last param and always + * use the same type in your code (boolean or number). That will improve JS speed. + * + * For regular `Array`-s make sure all elements are [0..255]. + * + * ##### Example + * + * ```javascript + * push(chunk, false); // push one of data chunks + * ... + * push(chunk, true); // push last chunk + * ``` + **/ + push(data, mode) { + const { strm, options: { chunkSize, dictionary } } = this; + let status, _mode; + + // Flag to properly process Z_BUF_ERROR on testing inflate call + // when we check that all output data was flushed. + let allowBufError = false; + + if (this.ended) { return false; } + _mode = (mode === ~~mode) ? mode : ((mode === true) ? Z_FINISH : Z_NO_FLUSH); + + // Convert data if needed + if (typeof data === 'string') { + // Only binary strings can be decompressed on practice + strm.input = binstring2buf(data); + } else if (data instanceof ArrayBuffer) { + strm.input = new Uint8Array(data); + } else { + strm.input = data; + } + + strm.next_in = 0; + strm.avail_in = strm.input.length; + + do { + if (strm.avail_out === 0) { + strm.output = new Buf8(chunkSize); + strm.next_out = 0; + strm.avail_out = chunkSize; + } + + status = inflate(strm, Z_NO_FLUSH); /* no bad return value */ + + if (status === Z_NEED_DICT && dictionary) { + status = inflateSetDictionary(this.strm, dictionary); + } + + if (status === Z_BUF_ERROR && allowBufError === true) { + status = Z_OK; + allowBufError = false; + } + + if (status !== Z_STREAM_END && status !== Z_OK) { + this.onEnd(status); + this.ended = true; + return false; + } + + if (strm.next_out) { + if (strm.avail_out === 0 || status === Z_STREAM_END || (strm.avail_in === 0 && (_mode === Z_FINISH || _mode === Z_SYNC_FLUSH))) { + this.onData(shrinkBuf(strm.output, strm.next_out)); + } + } + + // When no more input data, we should check that internal inflate buffers + // are flushed. The only way to do it when avail_out = 0 - run one more + // inflate pass. But if output data not exists, inflate return Z_BUF_ERROR. + // Here we set flag to process this error properly. + // + // NOTE. Deflate does not return error in this case and does not needs such + // logic. + if (strm.avail_in === 0 && strm.avail_out === 0) { + allowBufError = true; + } + + } while ((strm.avail_in > 0 || strm.avail_out === 0) && status !== Z_STREAM_END); + + if (status === Z_STREAM_END) { + _mode = Z_FINISH; + } + + // Finalize on the last chunk. + if (_mode === Z_FINISH) { + status = inflateEnd(this.strm); + this.onEnd(status); + this.ended = true; + return status === Z_OK; + } + + // callback interim results if Z_SYNC_FLUSH. + if (_mode === Z_SYNC_FLUSH) { + this.onEnd(Z_OK); + strm.avail_out = 0; + return true; + } + + return true; + }; + + /** + * Inflate#onData(chunk) -> Void + * - chunk (Uint8Array|Array|String): output data. Type of array depends + * on js engine support. When string output requested, each chunk + * will be string. + * + * By default, stores data blocks in `chunks[]` property and glue + * those in `onEnd`. Override this handler, if you need another behaviour. + **/ + onData(chunk) { + this.chunks.push(chunk); + }; + + + + /** + * Inflate#onEnd(status) -> Void + * - status (Number): inflate status. 0 (Z_OK) on success, + * other if not. + * + * Called either after you tell inflate that the input stream is + * complete (Z_FINISH) or should be flushed (Z_SYNC_FLUSH) + * or if an error happened. By default - join collected chunks, + * free memory and fill `results` / `err` properties. + **/ + onEnd(status) { + // On success - join + if (status === Z_OK) { + this.result = flattenChunks(this.chunks); + } + this.chunks = []; + this.err = status; + this.msg = this.strm.msg; + }; + } + + /* + node-bzip - a pure-javascript Node.JS module for decoding bzip2 data + + Copyright (C) 2012 Eli Skeggs + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with this library; if not, see + http://www.gnu.org/licenses/lgpl-2.1.html + + Adapted from bzip2.js, copyright 2011 antimatter15 (antimatter15@gmail.com). + + Based on micro-bunzip by Rob Landley (rob@landley.net). + + Based on bzip2 decompression code by Julian R Seward (jseward@acm.org), + which also acknowledges contributions by Mike Burrows, David Wheeler, + Peter Fenwick, Alistair Moffat, Radford Neal, Ian H. Witten, + Robert Sedgewick, and Jon L. Bentley. + */ + + var BITMASK = [0x00, 0x01, 0x03, 0x07, 0x0F, 0x1F, 0x3F, 0x7F, 0xFF]; + + // offset in bytes + var BitReader = function(stream) { + this.stream = stream; + this.bitOffset = 0; + this.curByte = 0; + this.hasByte = false; + }; + + BitReader.prototype._ensureByte = function() { + if (!this.hasByte) { + this.curByte = this.stream.readByte(); + this.hasByte = true; + } + }; + + // reads bits from the buffer + BitReader.prototype.read = function(bits) { + var result = 0; + while (bits > 0) { + this._ensureByte(); + var remaining = 8 - this.bitOffset; + // if we're in a byte + if (bits >= remaining) { + result <<= remaining; + result |= BITMASK[remaining] & this.curByte; + this.hasByte = false; + this.bitOffset = 0; + bits -= remaining; + } else { + result <<= bits; + var shift = remaining - bits; + result |= (this.curByte & (BITMASK[bits] << shift)) >> shift; + this.bitOffset += bits; + bits = 0; + } + } + return result; + }; + + // seek to an arbitrary point in the buffer (expressed in bits) + BitReader.prototype.seek = function(pos) { + var n_bit = pos % 8; + var n_byte = (pos - n_bit) / 8; + this.bitOffset = n_bit; + this.stream.seek(n_byte); + this.hasByte = false; + }; + + // reads 6 bytes worth of data using the read method + BitReader.prototype.pi = function() { + var buf = new Uint8Array(6), i; + for (i = 0; i < buf.length; i++) { + buf[i] = this.read(8); + } + return bufToHex(buf); + }; + + function bufToHex(buf) { + return Array.prototype.map.call(buf, x => ('00' + x.toString(16)).slice(-2)).join(''); + } + + var bitreader = BitReader; + + /* very simple input/output stream interface */ + var Stream = function() { + }; + + // input streams ////////////// + /** Returns the next byte, or -1 for EOF. */ + Stream.prototype.readByte = function() { + throw new Error("abstract method readByte() not implemented"); + }; + /** Attempts to fill the buffer; returns number of bytes read, or + * -1 for EOF. */ + Stream.prototype.read = function(buffer, bufOffset, length) { + var bytesRead = 0; + while (bytesRead < length) { + var c = this.readByte(); + if (c < 0) { // EOF + return (bytesRead===0) ? -1 : bytesRead; + } + buffer[bufOffset++] = c; + bytesRead++; + } + return bytesRead; + }; + Stream.prototype.seek = function(new_pos) { + throw new Error("abstract method seek() not implemented"); + }; + + // output streams /////////// + Stream.prototype.writeByte = function(_byte) { + throw new Error("abstract method readByte() not implemented"); + }; + Stream.prototype.write = function(buffer, bufOffset, length) { + var i; + for (i=0; i>> 0; // return an unsigned value + }; + + /** + * Update the CRC with a single byte + * @param value The value to update the CRC with + */ + this.updateCRC = function(value) { + crc = (crc << 8) ^ crc32Lookup[((crc >>> 24) ^ value) & 0xff]; + }; + + /** + * Update the CRC with a sequence of identical bytes + * @param value The value to update the CRC with + * @param count The number of bytes + */ + this.updateCRCRun = function(value, count) { + while (count-- > 0) { + crc = (crc << 8) ^ crc32Lookup[((crc >>> 24) ^ value) & 0xff]; + } + }; + }; + return CRC32; + })(); + + /* + seek-bzip - a pure-javascript module for seeking within bzip2 data + + Copyright (C) 2013 C. Scott Ananian + Copyright (C) 2012 Eli Skeggs + Copyright (C) 2011 Kevin Kwok + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with this library; if not, see + http://www.gnu.org/licenses/lgpl-2.1.html + + Adapted from node-bzip, copyright 2012 Eli Skeggs. + Adapted from bzip2.js, copyright 2011 Kevin Kwok (antimatter15@gmail.com). + + Based on micro-bunzip by Rob Landley (rob@landley.net). + + Based on bzip2 decompression code by Julian R Seward (jseward@acm.org), + which also acknowledges contributions by Mike Burrows, David Wheeler, + Peter Fenwick, Alistair Moffat, Radford Neal, Ian H. Witten, + Robert Sedgewick, and Jon L. Bentley. + */ + + + + + + var MAX_HUFCODE_BITS = 20; + var MAX_SYMBOLS = 258; + var SYMBOL_RUNA = 0; + var SYMBOL_RUNB = 1; + var MIN_GROUPS = 2; + var MAX_GROUPS = 6; + var GROUP_SIZE = 50; + + var WHOLEPI = "314159265359"; + var SQRTPI = "177245385090"; + + var mtf = function(array, index) { + var src = array[index], i; + for (i = index; i > 0; i--) { + array[i] = array[i-1]; + } + array[0] = src; + return src; + }; + + var Err = { + OK: 0, + LAST_BLOCK: -1, + NOT_BZIP_DATA: -2, + UNEXPECTED_INPUT_EOF: -3, + UNEXPECTED_OUTPUT_EOF: -4, + DATA_ERROR: -5, + OUT_OF_MEMORY: -6, + OBSOLETE_INPUT: -7, + END_OF_BLOCK: -8 + }; + var ErrorMessages = {}; + ErrorMessages[Err.LAST_BLOCK] = "Bad file checksum"; + ErrorMessages[Err.NOT_BZIP_DATA] = "Not bzip data"; + ErrorMessages[Err.UNEXPECTED_INPUT_EOF] = "Unexpected input EOF"; + ErrorMessages[Err.UNEXPECTED_OUTPUT_EOF] = "Unexpected output EOF"; + ErrorMessages[Err.DATA_ERROR] = "Data error"; + ErrorMessages[Err.OUT_OF_MEMORY] = "Out of memory"; + ErrorMessages[Err.OBSOLETE_INPUT] = "Obsolete (pre 0.9.5) bzip format not supported."; + + var _throw = function(status, optDetail) { + var msg = ErrorMessages[status] || 'unknown error'; + if (optDetail) { msg += ': '+optDetail; } + var e = new TypeError(msg); + e.errorCode = status; + throw e; + }; + + var Bunzip = function(inputStream, outputStream) { + this.writePos = this.writeCurrent = this.writeCount = 0; + + this._start_bunzip(inputStream, outputStream); + }; + Bunzip.prototype._init_block = function() { + var moreBlocks = this._get_next_block(); + if ( !moreBlocks ) { + this.writeCount = -1; + return false; /* no more blocks */ + } + this.blockCRC = new crc32$1(); + return true; + }; + /* XXX micro-bunzip uses (inputStream, inputBuffer, len) as arguments */ + Bunzip.prototype._start_bunzip = function(inputStream, outputStream) { + /* Ensure that file starts with "BZh['1'-'9']." */ + var buf = new Uint8Array(4); + if (inputStream.read(buf, 0, 4) !== 4 || + String.fromCharCode(buf[0], buf[1], buf[2]) !== 'BZh') + _throw(Err.NOT_BZIP_DATA, 'bad magic'); + + var level = buf[3] - 0x30; + if (level < 1 || level > 9) + _throw(Err.NOT_BZIP_DATA, 'level out of range'); + + this.reader = new bitreader(inputStream); + + /* Fourth byte (ascii '1'-'9'), indicates block size in units of 100k of + uncompressed data. Allocate intermediate buffer for block. */ + this.dbufSize = 100000 * level; + this.nextoutput = 0; + this.outputStream = outputStream; + this.streamCRC = 0; + }; + Bunzip.prototype._get_next_block = function() { + var i, j, k; + var reader = this.reader; + // this is get_next_block() function from micro-bunzip: + /* Read in header signature and CRC, then validate signature. + (last block signature means CRC is for whole file, return now) */ + var h = reader.pi(); + if (h === SQRTPI) { // last block + return false; /* no more blocks */ + } + if (h !== WHOLEPI) + _throw(Err.NOT_BZIP_DATA); + this.targetBlockCRC = reader.read(32) >>> 0; // (convert to unsigned) + this.streamCRC = (this.targetBlockCRC ^ + ((this.streamCRC << 1) | (this.streamCRC>>>31))) >>> 0; + /* We can add support for blockRandomised if anybody complains. There was + some code for this in busybox 1.0.0-pre3, but nobody ever noticed that + it didn't actually work. */ + if (reader.read(1)) + _throw(Err.OBSOLETE_INPUT); + var origPointer = reader.read(24); + if (origPointer > this.dbufSize) + _throw(Err.DATA_ERROR, 'initial position out of bounds'); + /* mapping table: if some byte values are never used (encoding things + like ascii text), the compression code removes the gaps to have fewer + symbols to deal with, and writes a sparse bitfield indicating which + values were present. We make a translation table to convert the symbols + back to the corresponding bytes. */ + var t = reader.read(16); + var symToByte = new Uint8Array(256), symTotal = 0; + for (i = 0; i < 16; i++) { + if (t & (1 << (0xF - i))) { + var o = i * 16; + k = reader.read(16); + for (j = 0; j < 16; j++) + if (k & (1 << (0xF - j))) + symToByte[symTotal++] = o + j; + } + } + + /* How many different huffman coding groups does this block use? */ + var groupCount = reader.read(3); + if (groupCount < MIN_GROUPS || groupCount > MAX_GROUPS) + _throw(Err.DATA_ERROR); + /* nSelectors: Every GROUP_SIZE many symbols we select a new huffman coding + group. Read in the group selector list, which is stored as MTF encoded + bit runs. (MTF=Move To Front, as each value is used it's moved to the + start of the list.) */ + var nSelectors = reader.read(15); + if (nSelectors === 0) + _throw(Err.DATA_ERROR); + + var mtfSymbol = new Uint8Array(256); + for (i = 0; i < groupCount; i++) + mtfSymbol[i] = i; + + var selectors = new Uint8Array(nSelectors); // was 32768... + + for (i = 0; i < nSelectors; i++) { + /* Get next value */ + for (j = 0; reader.read(1); j++) + if (j >= groupCount) _throw(Err.DATA_ERROR); + /* Decode MTF to get the next selector */ + selectors[i] = mtf(mtfSymbol, j); + } + + /* Read the huffman coding tables for each group, which code for symTotal + literal symbols, plus two run symbols (RUNA, RUNB) */ + var symCount = symTotal + 2; + var groups = [], hufGroup; + for (j = 0; j < groupCount; j++) { + var length = new Uint8Array(symCount), temp = new Uint16Array(MAX_HUFCODE_BITS + 1); + /* Read huffman code lengths for each symbol. They're stored in + a way similar to mtf; record a starting value for the first symbol, + and an offset from the previous value for everys symbol after that. */ + t = reader.read(5); // lengths + for (i = 0; i < symCount; i++) { + for (;;) { + if (t < 1 || t > MAX_HUFCODE_BITS) _throw(Err.DATA_ERROR); + /* If first bit is 0, stop. Else second bit indicates whether + to increment or decrement the value. */ + if(!reader.read(1)) + break; + if(!reader.read(1)) + t++; + else + t--; + } + length[i] = t; + } + + /* Find largest and smallest lengths in this group */ + var minLen, maxLen; + minLen = maxLen = length[0]; + for (i = 1; i < symCount; i++) { + if (length[i] > maxLen) + maxLen = length[i]; + else if (length[i] < minLen) + minLen = length[i]; + } + + /* Calculate permute[], base[], and limit[] tables from length[]. + * + * permute[] is the lookup table for converting huffman coded symbols + * into decoded symbols. base[] is the amount to subtract from the + * value of a huffman symbol of a given length when using permute[]. + * + * limit[] indicates the largest numerical value a symbol with a given + * number of bits can have. This is how the huffman codes can vary in + * length: each code with a value>limit[length] needs another bit. + */ + hufGroup = {}; + groups.push(hufGroup); + hufGroup.permute = new Uint16Array(MAX_SYMBOLS); + hufGroup.limit = new Uint32Array(MAX_HUFCODE_BITS + 2); + hufGroup.base = new Uint32Array(MAX_HUFCODE_BITS + 1); + hufGroup.minLen = minLen; + hufGroup.maxLen = maxLen; + /* Calculate permute[]. Concurently, initialize temp[] and limit[]. */ + var pp = 0; + for (i = minLen; i <= maxLen; i++) { + temp[i] = hufGroup.limit[i] = 0; + for (t = 0; t < symCount; t++) + if (length[t] === i) + hufGroup.permute[pp++] = t; + } + /* Count symbols coded for at each bit length */ + for (i = 0; i < symCount; i++) + temp[length[i]]++; + /* Calculate limit[] (the largest symbol-coding value at each bit + * length, which is (previous limit<<1)+symbols at this level), and + * base[] (number of symbols to ignore at each bit length, which is + * limit minus the cumulative count of symbols coded for already). */ + pp = t = 0; + for (i = minLen; i < maxLen; i++) { + pp += temp[i]; + /* We read the largest possible symbol size and then unget bits + after determining how many we need, and those extra bits could + be set to anything. (They're noise from future symbols.) At + each level we're really only interested in the first few bits, + so here we set all the trailing to-be-ignored bits to 1 so they + don't affect the value>limit[length] comparison. */ + hufGroup.limit[i] = pp - 1; + pp <<= 1; + t += temp[i]; + hufGroup.base[i + 1] = pp - t; + } + hufGroup.limit[maxLen + 1] = Number.MAX_VALUE; /* Sentinal value for reading next sym. */ + hufGroup.limit[maxLen] = pp + temp[maxLen] - 1; + hufGroup.base[minLen] = 0; + } + /* We've finished reading and digesting the block header. Now read this + block's huffman coded symbols from the file and undo the huffman coding + and run length encoding, saving the result into dbuf[dbufCount++]=uc */ + + /* Initialize symbol occurrence counters and symbol Move To Front table */ + var byteCount = new Uint32Array(256); + for (i = 0; i < 256; i++) + mtfSymbol[i] = i; + /* Loop through compressed symbols. */ + var runPos = 0, dbufCount = 0, selector = 0, uc; + var dbuf = this.dbuf = new Uint32Array(this.dbufSize); + symCount = 0; + for (;;) { + /* Determine which huffman coding group to use. */ + if (!(symCount--)) { + symCount = GROUP_SIZE - 1; + if (selector >= nSelectors) { _throw(Err.DATA_ERROR); } + hufGroup = groups[selectors[selector++]]; + } + /* Read next huffman-coded symbol. */ + i = hufGroup.minLen; + j = reader.read(i); + for (;;i++) { + if (i > hufGroup.maxLen) { _throw(Err.DATA_ERROR); } + if (j <= hufGroup.limit[i]) + break; + j = (j << 1) | reader.read(1); + } + /* Huffman decode value to get nextSym (with bounds checking) */ + j -= hufGroup.base[i]; + if (j < 0 || j >= MAX_SYMBOLS) { _throw(Err.DATA_ERROR); } + var nextSym = hufGroup.permute[j]; + /* We have now decoded the symbol, which indicates either a new literal + byte, or a repeated run of the most recent literal byte. First, + check if nextSym indicates a repeated run, and if so loop collecting + how many times to repeat the last literal. */ + if (nextSym === SYMBOL_RUNA || nextSym === SYMBOL_RUNB) { + /* If this is the start of a new run, zero out counter */ + if (!runPos){ + runPos = 1; + t = 0; + } + /* Neat trick that saves 1 symbol: instead of or-ing 0 or 1 at + each bit position, add 1 or 2 instead. For example, + 1011 is 1<<0 + 1<<1 + 2<<2. 1010 is 2<<0 + 2<<1 + 1<<2. + You can make any bit pattern that way using 1 less symbol than + the basic or 0/1 method (except all bits 0, which would use no + symbols, but a run of length 0 doesn't mean anything in this + context). Thus space is saved. */ + if (nextSym === SYMBOL_RUNA) + t += runPos; + else + t += 2 * runPos; + runPos <<= 1; + continue; + } + /* When we hit the first non-run symbol after a run, we now know + how many times to repeat the last literal, so append that many + copies to our buffer of decoded symbols (dbuf) now. (The last + literal used is the one at the head of the mtfSymbol array.) */ + if (runPos){ + runPos = 0; + if (dbufCount + t > this.dbufSize) { _throw(Err.DATA_ERROR); } + uc = symToByte[mtfSymbol[0]]; + byteCount[uc] += t; + while (t--) + dbuf[dbufCount++] = uc; + } + /* Is this the terminating symbol? */ + if (nextSym > symTotal) + break; + /* At this point, nextSym indicates a new literal character. Subtract + one to get the position in the MTF array at which this literal is + currently to be found. (Note that the result can't be -1 or 0, + because 0 and 1 are RUNA and RUNB. But another instance of the + first symbol in the mtf array, position 0, would have been handled + as part of a run above. Therefore 1 unused mtf position minus + 2 non-literal nextSym values equals -1.) */ + if (dbufCount >= this.dbufSize) { _throw(Err.DATA_ERROR); } + i = nextSym - 1; + uc = mtf(mtfSymbol, i); + uc = symToByte[uc]; + /* We have our literal byte. Save it into dbuf. */ + byteCount[uc]++; + dbuf[dbufCount++] = uc; + } + /* At this point, we've read all the huffman-coded symbols (and repeated + runs) for this block from the input stream, and decoded them into the + intermediate buffer. There are dbufCount many decoded bytes in dbuf[]. + Now undo the Burrows-Wheeler transform on dbuf. + See http://dogma.net/markn/articles/bwt/bwt.htm + */ + if (origPointer < 0 || origPointer >= dbufCount) { _throw(Err.DATA_ERROR); } + /* Turn byteCount into cumulative occurrence counts of 0 to n-1. */ + j = 0; + for (i = 0; i < 256; i++) { + k = j + byteCount[i]; + byteCount[i] = j; + j = k; + } + /* Figure out what order dbuf would be in if we sorted it. */ + for (i = 0; i < dbufCount; i++) { + uc = dbuf[i] & 0xff; + dbuf[byteCount[uc]] |= (i << 8); + byteCount[uc]++; + } + /* Decode first byte by hand to initialize "previous" byte. Note that it + doesn't get output, and if the first three characters are identical + it doesn't qualify as a run (hence writeRunCountdown=5). */ + var pos = 0, current = 0, run = 0; + if (dbufCount) { + pos = dbuf[origPointer]; + current = (pos & 0xff); + pos >>= 8; + run = -1; + } + this.writePos = pos; + this.writeCurrent = current; + this.writeCount = dbufCount; + this.writeRun = run; + + return true; /* more blocks to come */ + }; + /* Undo burrows-wheeler transform on intermediate buffer to produce output. + If start_bunzip was initialized with out_fd=-1, then up to len bytes of + data are written to outbuf. Return value is number of bytes written or + error (all errors are negative numbers). If out_fd!=-1, outbuf and len + are ignored, data is written to out_fd and return is RETVAL_OK or error. + */ + Bunzip.prototype._read_bunzip = function(outputBuffer, len) { + var copies, previous, outbyte; + /* james@jamestaylor.org: writeCount goes to -1 when the buffer is fully + decoded, which results in this returning RETVAL_LAST_BLOCK, also + equal to -1... Confusing, I'm returning 0 here to indicate no + bytes written into the buffer */ + if (this.writeCount < 0) { return 0; } + var dbuf = this.dbuf, pos = this.writePos, current = this.writeCurrent; + var dbufCount = this.writeCount; this.outputsize; + var run = this.writeRun; + + while (dbufCount) { + dbufCount--; + previous = current; + pos = dbuf[pos]; + current = pos & 0xff; + pos >>= 8; + if (run++ === 3){ + copies = current; + outbyte = previous; + current = -1; + } else { + copies = 1; + outbyte = current; + } + this.blockCRC.updateCRCRun(outbyte, copies); + while (copies--) { + this.outputStream.writeByte(outbyte); + this.nextoutput++; + } + if (current != previous) + run = 0; + } + this.writeCount = dbufCount; + // check CRC + if (this.blockCRC.getCRC() !== this.targetBlockCRC) { + _throw(Err.DATA_ERROR, "Bad block CRC "+ + "(got "+this.blockCRC.getCRC().toString(16)+ + " expected "+this.targetBlockCRC.toString(16)+")"); + } + return this.nextoutput; + }; + + var coerceInputStream = function(input) { + if ('readByte' in input) { return input; } + var inputStream = new stream(); + inputStream.pos = 0; + inputStream.readByte = function() { return input[this.pos++]; }; + inputStream.seek = function(pos) { this.pos = pos; }; + inputStream.eof = function() { return this.pos >= input.length; }; + return inputStream; + }; + var coerceOutputStream = function(output) { + var outputStream = new stream(); + var resizeOk = true; + if (output) { + if (typeof(output)==='number') { + outputStream.buffer = new Uint8Array(output); + resizeOk = false; + } else if ('writeByte' in output) { + return output; + } else { + outputStream.buffer = output; + resizeOk = false; + } + } else { + outputStream.buffer = new Uint8Array(16384); + } + outputStream.pos = 0; + outputStream.writeByte = function(_byte) { + if (resizeOk && this.pos >= this.buffer.length) { + var newBuffer = new Uint8Array(this.buffer.length*2); + newBuffer.set(this.buffer); + this.buffer = newBuffer; + } + this.buffer[this.pos++] = _byte; + }; + outputStream.getBuffer = function() { + // trim buffer + if (this.pos !== this.buffer.length) { + if (!resizeOk) + throw new TypeError('outputsize does not match decoded input'); + var newBuffer = new Uint8Array(this.pos); + newBuffer.set(this.buffer.subarray(0, this.pos)); + this.buffer = newBuffer; + } + return this.buffer; + }; + outputStream._coerced = true; + return outputStream; + }; + + /* Static helper functions */ + // 'input' can be a stream or a buffer + // 'output' can be a stream or a buffer or a number (buffer size) + const decode$2 = function(input, output, multistream) { + // make a stream from a buffer, if necessary + var inputStream = coerceInputStream(input); + var outputStream = coerceOutputStream(output); + + var bz = new Bunzip(inputStream, outputStream); + while (true) { + if ('eof' in inputStream && inputStream.eof()) break; + if (bz._init_block()) { + bz._read_bunzip(); + } else { + var targetStreamCRC = bz.reader.read(32) >>> 0; // (convert to unsigned) + if (targetStreamCRC !== bz.streamCRC) { + _throw(Err.DATA_ERROR, "Bad stream CRC "+ + "(got "+bz.streamCRC.toString(16)+ + " expected "+targetStreamCRC.toString(16)+")"); + } + if (multistream && + 'eof' in inputStream && + !inputStream.eof()) { + // note that start_bunzip will also resync the bit reader to next byte + bz._start_bunzip(inputStream, outputStream); + } else break; + } + } + if ('getBuffer' in outputStream) + return outputStream.getBuffer(); + }; + const decodeBlock = function(input, pos, output) { + // make a stream from a buffer, if necessary + var inputStream = coerceInputStream(input); + var outputStream = coerceOutputStream(output); + var bz = new Bunzip(inputStream, outputStream); + bz.reader.seek(pos); + /* Fill the decode buffer for the block */ + var moreBlocks = bz._get_next_block(); + if (moreBlocks) { + /* Init the CRC for writing */ + bz.blockCRC = new crc32$1(); + + /* Zero this so the current byte from before the seek is not written */ + bz.writeCopies = 0; + + /* Decompress the block and write to stdout */ + bz._read_bunzip(); + // XXX keep writing? + } + if ('getBuffer' in outputStream) + return outputStream.getBuffer(); + }; + /* Reads bzip2 file from stream or buffer `input`, and invoke + * `callback(position, size)` once for each bzip2 block, + * where position gives the starting position (in *bits*) + * and size gives uncompressed size of the block (in *bytes*). */ + const table = function(input, callback, multistream) { + // make a stream from a buffer, if necessary + var inputStream = new stream(); + inputStream.delegate = coerceInputStream(input); + inputStream.pos = 0; + inputStream.readByte = function() { + this.pos++; + return this.delegate.readByte(); + }; + if (inputStream.delegate.eof) { + inputStream.eof = inputStream.delegate.eof.bind(inputStream.delegate); + } + var outputStream = new stream(); + outputStream.pos = 0; + outputStream.writeByte = function() { this.pos++; }; + + var bz = new Bunzip(inputStream, outputStream); + var blockSize = bz.dbufSize; + while (true) { + if ('eof' in inputStream && inputStream.eof()) break; + + var position = inputStream.pos*8 + bz.reader.bitOffset; + if (bz.reader.hasByte) { position -= 8; } + + if (bz._init_block()) { + var start = outputStream.pos; + bz._read_bunzip(); + callback(position, outputStream.pos - start); + } else { + bz.reader.read(32); // (but we ignore the crc) + if (multistream && + 'eof' in inputStream && + !inputStream.eof()) { + // note that start_bunzip will also resync the bit reader to next byte + bz._start_bunzip(inputStream, outputStream); + console.assert(bz.dbufSize === blockSize, + "shouldn't change block size within multistream file"); + } else break; + } + } + }; + + var lib = { + Bunzip, + Stream: stream, + Err, + decode: decode$2, + decodeBlock, + table + }; + var lib_4 = lib.decode; + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Implementation of the Literal Data Packet (Tag 11) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.9|RFC4880 5.9}: + * A Literal Data packet contains the body of a message; data that is not to be + * further interpreted. + */ + class LiteralDataPacket { + static get tag() { + return enums.packet.literalData; + } + + /** + * @param {Date} date - The creation date of the literal package + */ + constructor(date = new Date()) { + this.format = enums.literal.utf8; // default format for literal data packets + this.date = util.normalizeDate(date); + this.text = null; // textual data representation + this.data = null; // literal data representation + this.filename = ''; + } + + /** + * Set the packet data to a javascript native string, end of line + * will be normalized to \r\n and by default text is converted to UTF8 + * @param {String | ReadableStream} text - Any native javascript string + * @param {enums.literal} [format] - The format of the string of bytes + */ + setText(text, format = enums.literal.utf8) { + this.format = format; + this.text = text; + this.data = null; + } + + /** + * Returns literal data packets as native JavaScript string + * with normalized end of line to \n + * @param {Boolean} [clone] - Whether to return a clone so that getBytes/getText can be called again + * @returns {String | ReadableStream} Literal data as text. + */ + getText(clone = false) { + if (this.text === null || util.isStream(this.text)) { // Assume that this.text has been read + this.text = util.decodeUTF8(util.nativeEOL(this.getBytes(clone))); + } + return this.text; + } + + /** + * Set the packet data to value represented by the provided string of bytes. + * @param {Uint8Array | ReadableStream} bytes - The string of bytes + * @param {enums.literal} format - The format of the string of bytes + */ + setBytes(bytes, format) { + this.format = format; + this.data = bytes; + this.text = null; + } + + + /** + * Get the byte sequence representing the literal packet data + * @param {Boolean} [clone] - Whether to return a clone so that getBytes/getText can be called again + * @returns {Uint8Array | ReadableStream} A sequence of bytes. + */ + getBytes(clone = false) { + if (this.data === null) { + // encode UTF8 and normalize EOL to \r\n + this.data = util.canonicalizeEOL(util.encodeUTF8(this.text)); + } + if (clone) { + return passiveClone(this.data); + } + return this.data; + } + + + /** + * Sets the filename of the literal packet data + * @param {String} filename - Any native javascript string + */ + setFilename(filename) { + this.filename = filename; + } + + + /** + * Get the filename of the literal packet data + * @returns {String} Filename. + */ + getFilename() { + return this.filename; + } + + /** + * Parsing function for a literal data packet (tag 11). + * + * @param {Uint8Array | ReadableStream} input - Payload of a tag 11 packet + * @returns {Promise} Object representation. + * @async + */ + async read(bytes) { + await parse(bytes, async reader => { + // - A one-octet field that describes how the data is formatted. + const format = await reader.readByte(); // enums.literal + + const filename_len = await reader.readByte(); + this.filename = util.decodeUTF8(await reader.readBytes(filename_len)); + + this.date = util.readDate(await reader.readBytes(4)); + + let data = reader.remainder(); + if (isArrayStream(data)) data = await readToEnd(data); + this.setBytes(data, format); + }); + } + + /** + * Creates a Uint8Array representation of the packet, excluding the data + * + * @returns {Uint8Array} Uint8Array representation of the packet. + */ + writeHeader() { + const filename = util.encodeUTF8(this.filename); + const filename_length = new Uint8Array([filename.length]); + + const format = new Uint8Array([this.format]); + const date = util.writeDate(this.date); + + return util.concatUint8Array([format, filename_length, filename, date]); + } + + /** + * Creates a Uint8Array representation of the packet + * + * @returns {Uint8Array | ReadableStream} Uint8Array representation of the packet. + */ + write() { + const header = this.writeHeader(); + const data = this.getBytes(); + + return util.concat([header, data]); + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + // Symbol to store cryptographic validity of the signature, to avoid recomputing multiple times on verification. + const verified = Symbol('verified'); + + // GPG puts the Issuer and Signature subpackets in the unhashed area. + // Tampering with those invalidates the signature, so we still trust them and parse them. + // All other unhashed subpackets are ignored. + const allowedUnhashedSubpackets = new Set([ + enums.signatureSubpacket.issuer, + enums.signatureSubpacket.issuerFingerprint, + enums.signatureSubpacket.embeddedSignature + ]); + + /** + * Implementation of the Signature Packet (Tag 2) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.2|RFC4480 5.2}: + * A Signature packet describes a binding between some public key and + * some data. The most common signatures are a signature of a file or a + * block of text, and a signature that is a certification of a User ID. + */ + class SignaturePacket { + static get tag() { + return enums.packet.signature; + } + + constructor() { + this.version = null; + /** @type {enums.signature} */ + this.signatureType = null; + /** @type {enums.hash} */ + this.hashAlgorithm = null; + /** @type {enums.publicKey} */ + this.publicKeyAlgorithm = null; + + this.signatureData = null; + this.unhashedSubpackets = []; + this.signedHashValue = null; + + this.created = null; + this.signatureExpirationTime = null; + this.signatureNeverExpires = true; + this.exportable = null; + this.trustLevel = null; + this.trustAmount = null; + this.regularExpression = null; + this.revocable = null; + this.keyExpirationTime = null; + this.keyNeverExpires = null; + this.preferredSymmetricAlgorithms = null; + this.revocationKeyClass = null; + this.revocationKeyAlgorithm = null; + this.revocationKeyFingerprint = null; + this.issuerKeyID = new KeyID(); + this.rawNotations = []; + this.notations = {}; + this.preferredHashAlgorithms = null; + this.preferredCompressionAlgorithms = null; + this.keyServerPreferences = null; + this.preferredKeyServer = null; + this.isPrimaryUserID = null; + this.policyURI = null; + this.keyFlags = null; + this.signersUserID = null; + this.reasonForRevocationFlag = null; + this.reasonForRevocationString = null; + this.features = null; + this.signatureTargetPublicKeyAlgorithm = null; + this.signatureTargetHashAlgorithm = null; + this.signatureTargetHash = null; + this.embeddedSignature = null; + this.issuerKeyVersion = null; + this.issuerFingerprint = null; + this.preferredAEADAlgorithms = null; + + this.revoked = null; + this[verified] = null; + } + + /** + * parsing function for a signature packet (tag 2). + * @param {String} bytes - Payload of a tag 2 packet + * @returns {SignaturePacket} Object representation. + */ + read(bytes) { + let i = 0; + this.version = bytes[i++]; + + if (this.version !== 4 && this.version !== 5) { + throw new UnsupportedError(`Version ${this.version} of the signature packet is unsupported.`); + } + + this.signatureType = bytes[i++]; + this.publicKeyAlgorithm = bytes[i++]; + this.hashAlgorithm = bytes[i++]; + + // hashed subpackets + i += this.readSubPackets(bytes.subarray(i, bytes.length), true); + if (!this.created) { + throw new Error('Missing signature creation time subpacket.'); + } + + // A V4 signature hashes the packet body + // starting from its first field, the version number, through the end + // of the hashed subpacket data. Thus, the fields hashed are the + // signature version, the signature type, the public-key algorithm, the + // hash algorithm, the hashed subpacket length, and the hashed + // subpacket body. + this.signatureData = bytes.subarray(0, i); + + // unhashed subpackets + i += this.readSubPackets(bytes.subarray(i, bytes.length), false); + + // Two-octet field holding left 16 bits of signed hash value. + this.signedHashValue = bytes.subarray(i, i + 2); + i += 2; + + this.params = mod.signature.parseSignatureParams(this.publicKeyAlgorithm, bytes.subarray(i, bytes.length)); + } + + /** + * @returns {Uint8Array | ReadableStream} + */ + writeParams() { + if (this.params instanceof Promise) { + return fromAsync( + async () => mod.serializeParams(this.publicKeyAlgorithm, await this.params) + ); + } + return mod.serializeParams(this.publicKeyAlgorithm, this.params); + } + + write() { + const arr = []; + arr.push(this.signatureData); + arr.push(this.writeUnhashedSubPackets()); + arr.push(this.signedHashValue); + arr.push(this.writeParams()); + return util.concat(arr); + } + + /** + * Signs provided data. This needs to be done prior to serialization. + * @param {SecretKeyPacket} key - Private key used to sign the message. + * @param {Object} data - Contains packets to be signed. + * @param {Date} [date] - The signature creation time. + * @param {Boolean} [detached] - Whether to create a detached signature + * @throws {Error} if signing failed + * @async + */ + async sign(key, data, date = new Date(), detached = false) { + if (key.version === 5) { + this.version = 5; + } else { + this.version = 4; + } + const arr = [new Uint8Array([this.version, this.signatureType, this.publicKeyAlgorithm, this.hashAlgorithm])]; + + this.created = util.normalizeDate(date); + this.issuerKeyVersion = key.version; + this.issuerFingerprint = key.getFingerprintBytes(); + this.issuerKeyID = key.getKeyID(); + + // Add hashed subpackets + arr.push(this.writeHashedSubPackets()); + + // Remove unhashed subpackets, in case some allowed unhashed + // subpackets existed, in order not to duplicate them (in both + // the hashed and unhashed subpackets) when re-signing. + this.unhashedSubpackets = []; + + this.signatureData = util.concat(arr); + + const toHash = this.toHash(this.signatureType, data, detached); + const hash = await this.hash(this.signatureType, data, toHash, detached); + + this.signedHashValue = slice(clone(hash), 0, 2); + const signed = async () => mod.signature.sign( + this.publicKeyAlgorithm, this.hashAlgorithm, key.publicParams, key.privateParams, toHash, await readToEnd(hash) + ); + if (util.isStream(hash)) { + this.params = signed(); + } else { + this.params = await signed(); + + // Store the fact that this signature is valid, e.g. for when we call `await + // getLatestValidSignature(this.revocationSignatures, key, data)` later. + // Note that this only holds up if the key and data passed to verify are the + // same as the ones passed to sign. + this[verified] = true; + } + } + + /** + * Creates Uint8Array of bytes of all subpacket data except Issuer and Embedded Signature subpackets + * @returns {Uint8Array} Subpacket data. + */ + writeHashedSubPackets() { + const sub = enums.signatureSubpacket; + const arr = []; + let bytes; + if (this.created === null) { + throw new Error('Missing signature creation time'); + } + arr.push(writeSubPacket(sub.signatureCreationTime, true, util.writeDate(this.created))); + if (this.signatureExpirationTime !== null) { + arr.push(writeSubPacket(sub.signatureExpirationTime, true, util.writeNumber(this.signatureExpirationTime, 4))); + } + if (this.exportable !== null) { + arr.push(writeSubPacket(sub.exportableCertification, true, new Uint8Array([this.exportable ? 1 : 0]))); + } + if (this.trustLevel !== null) { + bytes = new Uint8Array([this.trustLevel, this.trustAmount]); + arr.push(writeSubPacket(sub.trustSignature, true, bytes)); + } + if (this.regularExpression !== null) { + arr.push(writeSubPacket(sub.regularExpression, true, this.regularExpression)); + } + if (this.revocable !== null) { + arr.push(writeSubPacket(sub.revocable, true, new Uint8Array([this.revocable ? 1 : 0]))); + } + if (this.keyExpirationTime !== null) { + arr.push(writeSubPacket(sub.keyExpirationTime, true, util.writeNumber(this.keyExpirationTime, 4))); + } + if (this.preferredSymmetricAlgorithms !== null) { + bytes = util.stringToUint8Array(util.uint8ArrayToString(this.preferredSymmetricAlgorithms)); + arr.push(writeSubPacket(sub.preferredSymmetricAlgorithms, false, bytes)); + } + if (this.revocationKeyClass !== null) { + bytes = new Uint8Array([this.revocationKeyClass, this.revocationKeyAlgorithm]); + bytes = util.concat([bytes, this.revocationKeyFingerprint]); + arr.push(writeSubPacket(sub.revocationKey, false, bytes)); + } + if (!this.issuerKeyID.isNull() && this.issuerKeyVersion !== 5) { + // If the version of [the] key is greater than 4, this subpacket + // MUST NOT be included in the signature. + arr.push(writeSubPacket(sub.issuer, true, this.issuerKeyID.write())); + } + this.rawNotations.forEach(({ name, value, humanReadable, critical }) => { + bytes = [new Uint8Array([humanReadable ? 0x80 : 0, 0, 0, 0])]; + const encodedName = util.encodeUTF8(name); + // 2 octets of name length + bytes.push(util.writeNumber(encodedName.length, 2)); + // 2 octets of value length + bytes.push(util.writeNumber(value.length, 2)); + bytes.push(encodedName); + bytes.push(value); + bytes = util.concat(bytes); + arr.push(writeSubPacket(sub.notationData, critical, bytes)); + }); + if (this.preferredHashAlgorithms !== null) { + bytes = util.stringToUint8Array(util.uint8ArrayToString(this.preferredHashAlgorithms)); + arr.push(writeSubPacket(sub.preferredHashAlgorithms, false, bytes)); + } + if (this.preferredCompressionAlgorithms !== null) { + bytes = util.stringToUint8Array(util.uint8ArrayToString(this.preferredCompressionAlgorithms)); + arr.push(writeSubPacket(sub.preferredCompressionAlgorithms, false, bytes)); + } + if (this.keyServerPreferences !== null) { + bytes = util.stringToUint8Array(util.uint8ArrayToString(this.keyServerPreferences)); + arr.push(writeSubPacket(sub.keyServerPreferences, false, bytes)); + } + if (this.preferredKeyServer !== null) { + arr.push(writeSubPacket(sub.preferredKeyServer, false, util.encodeUTF8(this.preferredKeyServer))); + } + if (this.isPrimaryUserID !== null) { + arr.push(writeSubPacket(sub.primaryUserID, false, new Uint8Array([this.isPrimaryUserID ? 1 : 0]))); + } + if (this.policyURI !== null) { + arr.push(writeSubPacket(sub.policyURI, false, util.encodeUTF8(this.policyURI))); + } + if (this.keyFlags !== null) { + bytes = util.stringToUint8Array(util.uint8ArrayToString(this.keyFlags)); + arr.push(writeSubPacket(sub.keyFlags, true, bytes)); + } + if (this.signersUserID !== null) { + arr.push(writeSubPacket(sub.signersUserID, false, util.encodeUTF8(this.signersUserID))); + } + if (this.reasonForRevocationFlag !== null) { + bytes = util.stringToUint8Array(String.fromCharCode(this.reasonForRevocationFlag) + this.reasonForRevocationString); + arr.push(writeSubPacket(sub.reasonForRevocation, true, bytes)); + } + if (this.features !== null) { + bytes = util.stringToUint8Array(util.uint8ArrayToString(this.features)); + arr.push(writeSubPacket(sub.features, false, bytes)); + } + if (this.signatureTargetPublicKeyAlgorithm !== null) { + bytes = [new Uint8Array([this.signatureTargetPublicKeyAlgorithm, this.signatureTargetHashAlgorithm])]; + bytes.push(util.stringToUint8Array(this.signatureTargetHash)); + bytes = util.concat(bytes); + arr.push(writeSubPacket(sub.signatureTarget, true, bytes)); + } + if (this.embeddedSignature !== null) { + arr.push(writeSubPacket(sub.embeddedSignature, true, this.embeddedSignature.write())); + } + if (this.issuerFingerprint !== null) { + bytes = [new Uint8Array([this.issuerKeyVersion]), this.issuerFingerprint]; + bytes = util.concat(bytes); + arr.push(writeSubPacket(sub.issuerFingerprint, this.version === 5, bytes)); + } + if (this.preferredAEADAlgorithms !== null) { + bytes = util.stringToUint8Array(util.uint8ArrayToString(this.preferredAEADAlgorithms)); + arr.push(writeSubPacket(sub.preferredAEADAlgorithms, false, bytes)); + } + + const result = util.concat(arr); + const length = util.writeNumber(result.length, 2); + + return util.concat([length, result]); + } + + /** + * Creates an Uint8Array containing the unhashed subpackets + * @returns {Uint8Array} Subpacket data. + */ + writeUnhashedSubPackets() { + const arr = []; + this.unhashedSubpackets.forEach(data => { + arr.push(writeSimpleLength(data.length)); + arr.push(data); + }); + + const result = util.concat(arr); + const length = util.writeNumber(result.length, 2); + + return util.concat([length, result]); + } + + // V4 signature sub packets + readSubPacket(bytes, hashed = true) { + let mypos = 0; + + // The leftmost bit denotes a "critical" packet + const critical = !!(bytes[mypos] & 0x80); + const type = bytes[mypos] & 0x7F; + + if (!hashed) { + this.unhashedSubpackets.push(bytes.subarray(mypos, bytes.length)); + if (!allowedUnhashedSubpackets.has(type)) { + return; + } + } + + mypos++; + + // subpacket type + switch (type) { + case enums.signatureSubpacket.signatureCreationTime: + // Signature Creation Time + this.created = util.readDate(bytes.subarray(mypos, bytes.length)); + break; + case enums.signatureSubpacket.signatureExpirationTime: { + // Signature Expiration Time in seconds + const seconds = util.readNumber(bytes.subarray(mypos, bytes.length)); + + this.signatureNeverExpires = seconds === 0; + this.signatureExpirationTime = seconds; + + break; + } + case enums.signatureSubpacket.exportableCertification: + // Exportable Certification + this.exportable = bytes[mypos++] === 1; + break; + case enums.signatureSubpacket.trustSignature: + // Trust Signature + this.trustLevel = bytes[mypos++]; + this.trustAmount = bytes[mypos++]; + break; + case enums.signatureSubpacket.regularExpression: + // Regular Expression + this.regularExpression = bytes[mypos]; + break; + case enums.signatureSubpacket.revocable: + // Revocable + this.revocable = bytes[mypos++] === 1; + break; + case enums.signatureSubpacket.keyExpirationTime: { + // Key Expiration Time in seconds + const seconds = util.readNumber(bytes.subarray(mypos, bytes.length)); + + this.keyExpirationTime = seconds; + this.keyNeverExpires = seconds === 0; + + break; + } + case enums.signatureSubpacket.preferredSymmetricAlgorithms: + // Preferred Symmetric Algorithms + this.preferredSymmetricAlgorithms = [...bytes.subarray(mypos, bytes.length)]; + break; + case enums.signatureSubpacket.revocationKey: + // Revocation Key + // (1 octet of class, 1 octet of public-key algorithm ID, 20 + // octets of + // fingerprint) + this.revocationKeyClass = bytes[mypos++]; + this.revocationKeyAlgorithm = bytes[mypos++]; + this.revocationKeyFingerprint = bytes.subarray(mypos, mypos + 20); + break; + + case enums.signatureSubpacket.issuer: + // Issuer + this.issuerKeyID.read(bytes.subarray(mypos, bytes.length)); + break; + + case enums.signatureSubpacket.notationData: { + // Notation Data + const humanReadable = !!(bytes[mypos] & 0x80); + + // We extract key/value tuple from the byte stream. + mypos += 4; + const m = util.readNumber(bytes.subarray(mypos, mypos + 2)); + mypos += 2; + const n = util.readNumber(bytes.subarray(mypos, mypos + 2)); + mypos += 2; + + const name = util.decodeUTF8(bytes.subarray(mypos, mypos + m)); + const value = bytes.subarray(mypos + m, mypos + m + n); + + this.rawNotations.push({ name, humanReadable, value, critical }); + + if (humanReadable) { + this.notations[name] = util.decodeUTF8(value); + } + break; + } + case enums.signatureSubpacket.preferredHashAlgorithms: + // Preferred Hash Algorithms + this.preferredHashAlgorithms = [...bytes.subarray(mypos, bytes.length)]; + break; + case enums.signatureSubpacket.preferredCompressionAlgorithms: + // Preferred Compression Algorithms + this.preferredCompressionAlgorithms = [...bytes.subarray(mypos, bytes.length)]; + break; + case enums.signatureSubpacket.keyServerPreferences: + // Key Server Preferences + this.keyServerPreferences = [...bytes.subarray(mypos, bytes.length)]; + break; + case enums.signatureSubpacket.preferredKeyServer: + // Preferred Key Server + this.preferredKeyServer = util.decodeUTF8(bytes.subarray(mypos, bytes.length)); + break; + case enums.signatureSubpacket.primaryUserID: + // Primary User ID + this.isPrimaryUserID = bytes[mypos++] !== 0; + break; + case enums.signatureSubpacket.policyURI: + // Policy URI + this.policyURI = util.decodeUTF8(bytes.subarray(mypos, bytes.length)); + break; + case enums.signatureSubpacket.keyFlags: + // Key Flags + this.keyFlags = [...bytes.subarray(mypos, bytes.length)]; + break; + case enums.signatureSubpacket.signersUserID: + // Signer's User ID + this.signersUserID = util.decodeUTF8(bytes.subarray(mypos, bytes.length)); + break; + case enums.signatureSubpacket.reasonForRevocation: + // Reason for Revocation + this.reasonForRevocationFlag = bytes[mypos++]; + this.reasonForRevocationString = util.decodeUTF8(bytes.subarray(mypos, bytes.length)); + break; + case enums.signatureSubpacket.features: + // Features + this.features = [...bytes.subarray(mypos, bytes.length)]; + break; + case enums.signatureSubpacket.signatureTarget: { + // Signature Target + // (1 octet public-key algorithm, 1 octet hash algorithm, N octets hash) + this.signatureTargetPublicKeyAlgorithm = bytes[mypos++]; + this.signatureTargetHashAlgorithm = bytes[mypos++]; + + const len = mod.getHashByteLength(this.signatureTargetHashAlgorithm); + + this.signatureTargetHash = util.uint8ArrayToString(bytes.subarray(mypos, mypos + len)); + break; + } + case enums.signatureSubpacket.embeddedSignature: + // Embedded Signature + this.embeddedSignature = new SignaturePacket(); + this.embeddedSignature.read(bytes.subarray(mypos, bytes.length)); + break; + case enums.signatureSubpacket.issuerFingerprint: + // Issuer Fingerprint + this.issuerKeyVersion = bytes[mypos++]; + this.issuerFingerprint = bytes.subarray(mypos, bytes.length); + if (this.issuerKeyVersion === 5) { + this.issuerKeyID.read(this.issuerFingerprint); + } else { + this.issuerKeyID.read(this.issuerFingerprint.subarray(-8)); + } + break; + case enums.signatureSubpacket.preferredAEADAlgorithms: + // Preferred AEAD Algorithms + this.preferredAEADAlgorithms = [...bytes.subarray(mypos, bytes.length)]; + break; + default: { + const err = new Error(`Unknown signature subpacket type ${type}`); + if (critical) { + throw err; + } else { + util.printDebug(err); + } + } + } + } + + readSubPackets(bytes, trusted = true, config) { + // Two-octet scalar octet count for following subpacket data. + const subpacketLength = util.readNumber(bytes.subarray(0, 2)); + + let i = 2; + + // subpacket data set (zero or more subpackets) + while (i < 2 + subpacketLength) { + const len = readSimpleLength(bytes.subarray(i, bytes.length)); + i += len.offset; + + this.readSubPacket(bytes.subarray(i, i + len.len), trusted, config); + + i += len.len; + } + + return i; + } + + // Produces data to produce signature on + toSign(type, data) { + const t = enums.signature; + + switch (type) { + case t.binary: + if (data.text !== null) { + return util.encodeUTF8(data.getText(true)); + } + return data.getBytes(true); + + case t.text: { + const bytes = data.getBytes(true); + // normalize EOL to \r\n + return util.canonicalizeEOL(bytes); + } + case t.standalone: + return new Uint8Array(0); + + case t.certGeneric: + case t.certPersona: + case t.certCasual: + case t.certPositive: + case t.certRevocation: { + let packet; + let tag; + + if (data.userID) { + tag = 0xB4; + packet = data.userID; + } else if (data.userAttribute) { + tag = 0xD1; + packet = data.userAttribute; + } else { + throw new Error('Either a userID or userAttribute packet needs to be ' + + 'supplied for certification.'); + } + + const bytes = packet.write(); + + return util.concat([this.toSign(t.key, data), + new Uint8Array([tag]), + util.writeNumber(bytes.length, 4), + bytes]); + } + case t.subkeyBinding: + case t.subkeyRevocation: + case t.keyBinding: + return util.concat([this.toSign(t.key, data), this.toSign(t.key, { + key: data.bind + })]); + + case t.key: + if (data.key === undefined) { + throw new Error('Key packet is required for this signature.'); + } + return data.key.writeForHash(this.version); + + case t.keyRevocation: + return this.toSign(t.key, data); + case t.timestamp: + return new Uint8Array(0); + case t.thirdParty: + throw new Error('Not implemented'); + default: + throw new Error('Unknown signature type.'); + } + } + + calculateTrailer(data, detached) { + let length = 0; + return transform(clone(this.signatureData), value => { + length += value.length; + }, () => { + const arr = []; + if (this.version === 5 && (this.signatureType === enums.signature.binary || this.signatureType === enums.signature.text)) { + if (detached) { + arr.push(new Uint8Array(6)); + } else { + arr.push(data.writeHeader()); + } + } + arr.push(new Uint8Array([this.version, 0xFF])); + if (this.version === 5) { + arr.push(new Uint8Array(4)); + } + arr.push(util.writeNumber(length, 4)); + // For v5, this should really be writeNumber(length, 8) rather than the + // hardcoded 4 zero bytes above + return util.concat(arr); + }); + } + + toHash(signatureType, data, detached = false) { + const bytes = this.toSign(signatureType, data); + + return util.concat([bytes, this.signatureData, this.calculateTrailer(data, detached)]); + } + + async hash(signatureType, data, toHash, detached = false) { + if (!toHash) toHash = this.toHash(signatureType, data, detached); + return mod.hash.digest(this.hashAlgorithm, toHash); + } + + /** + * verifies the signature packet. Note: not all signature types are implemented + * @param {PublicSubkeyPacket|PublicKeyPacket| + * SecretSubkeyPacket|SecretKeyPacket} key - the public key to verify the signature + * @param {module:enums.signature} signatureType - Expected signature type + * @param {Uint8Array|Object} data - Data which on the signature applies + * @param {Date} [date] - Use the given date instead of the current time to check for signature validity and expiration + * @param {Boolean} [detached] - Whether to verify a detached signature + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} if signature validation failed + * @async + */ + async verify(key, signatureType, data, date = new Date(), detached = false, config$1 = config) { + if (!this.issuerKeyID.equals(key.getKeyID())) { + throw new Error('Signature was not issued by the given public key'); + } + if (this.publicKeyAlgorithm !== key.algorithm) { + throw new Error('Public key algorithm used to sign signature does not match issuer key algorithm.'); + } + + const isMessageSignature = signatureType === enums.signature.binary || signatureType === enums.signature.text; + // Cryptographic validity is cached after one successful verification. + // However, for message signatures, we always re-verify, since the passed `data` can change + const skipVerify = this[verified] && !isMessageSignature; + if (!skipVerify) { + let toHash; + let hash; + if (this.hashed) { + hash = await this.hashed; + } else { + toHash = this.toHash(signatureType, data, detached); + hash = await this.hash(signatureType, data, toHash); + } + hash = await readToEnd(hash); + if (this.signedHashValue[0] !== hash[0] || + this.signedHashValue[1] !== hash[1]) { + throw new Error('Signed digest did not match'); + } + + this.params = await this.params; + + this[verified] = await mod.signature.verify( + this.publicKeyAlgorithm, this.hashAlgorithm, this.params, key.publicParams, + toHash, hash + ); + + if (!this[verified]) { + throw new Error('Signature verification failed'); + } + } + + const normDate = util.normalizeDate(date); + if (normDate && this.created > normDate) { + throw new Error('Signature creation time is in the future'); + } + if (normDate && normDate >= this.getExpirationTime()) { + throw new Error('Signature is expired'); + } + if (config$1.rejectHashAlgorithms.has(this.hashAlgorithm)) { + throw new Error('Insecure hash algorithm: ' + enums.read(enums.hash, this.hashAlgorithm).toUpperCase()); + } + if (config$1.rejectMessageHashAlgorithms.has(this.hashAlgorithm) && + [enums.signature.binary, enums.signature.text].includes(this.signatureType)) { + throw new Error('Insecure message hash algorithm: ' + enums.read(enums.hash, this.hashAlgorithm).toUpperCase()); + } + this.rawNotations.forEach(({ name, critical }) => { + if (critical && (config$1.knownNotations.indexOf(name) < 0)) { + throw new Error(`Unknown critical notation: ${name}`); + } + }); + if (this.revocationKeyClass !== null) { + throw new Error('This key is intended to be revoked with an authorized key, which OpenPGP.js does not support.'); + } + } + + /** + * Verifies signature expiration date + * @param {Date} [date] - Use the given date for verification instead of the current time + * @returns {Boolean} True if expired. + */ + isExpired(date = new Date()) { + const normDate = util.normalizeDate(date); + if (normDate !== null) { + return !(this.created <= normDate && normDate < this.getExpirationTime()); + } + return false; + } + + /** + * Returns the expiration time of the signature or Infinity if signature does not expire + * @returns {Date | Infinity} Expiration time. + */ + getExpirationTime() { + return this.signatureNeverExpires ? Infinity : new Date(this.created.getTime() + this.signatureExpirationTime * 1000); + } + } + + /** + * Creates a Uint8Array representation of a sub signature packet + * @see {@link https://tools.ietf.org/html/rfc4880#section-5.2.3.1|RFC4880 5.2.3.1} + * @see {@link https://tools.ietf.org/html/rfc4880#section-5.2.3.2|RFC4880 5.2.3.2} + * @param {Integer} type - Subpacket signature type. + * @param {Boolean} critical - Whether the subpacket should be critical. + * @param {String} data - Data to be included + * @returns {Uint8Array} The signature subpacket. + * @private + */ + function writeSubPacket(type, critical, data) { + const arr = []; + arr.push(writeSimpleLength(data.length + 1)); + arr.push(new Uint8Array([(critical ? 0x80 : 0) | type])); + arr.push(data); + return util.concat(arr); + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + const VERSION = 3; + + /** + * Implementation of the One-Pass Signature Packets (Tag 4) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.4|RFC4880 5.4}: + * The One-Pass Signature packet precedes the signed data and contains + * enough information to allow the receiver to begin calculating any + * hashes needed to verify the signature. It allows the Signature + * packet to be placed at the end of the message, so that the signer + * can compute the entire signed message in one pass. + */ + class OnePassSignaturePacket { + static get tag() { + return enums.packet.onePassSignature; + } + + constructor() { + /** A one-octet version number. The current version is 3. */ + this.version = null; + /** + * A one-octet signature type. + * Signature types are described in + * {@link https://tools.ietf.org/html/rfc4880#section-5.2.1|RFC4880 Section 5.2.1}. + * @type {enums.signature} + + */ + this.signatureType = null; + /** + * A one-octet number describing the hash algorithm used. + * @see {@link https://tools.ietf.org/html/rfc4880#section-9.4|RFC4880 9.4} + * @type {enums.hash} + */ + this.hashAlgorithm = null; + /** + * A one-octet number describing the public-key algorithm used. + * @see {@link https://tools.ietf.org/html/rfc4880#section-9.1|RFC4880 9.1} + * @type {enums.publicKey} + */ + this.publicKeyAlgorithm = null; + /** An eight-octet number holding the Key ID of the signing key. */ + this.issuerKeyID = null; + /** + * A one-octet number holding a flag showing whether the signature is nested. + * A zero value indicates that the next packet is another One-Pass Signature packet + * that describes another signature to be applied to the same message data. + */ + this.flags = null; + } + + /** + * parsing function for a one-pass signature packet (tag 4). + * @param {Uint8Array} bytes - Payload of a tag 4 packet + * @returns {OnePassSignaturePacket} Object representation. + */ + read(bytes) { + let mypos = 0; + // A one-octet version number. The current version is 3. + this.version = bytes[mypos++]; + if (this.version !== VERSION) { + throw new UnsupportedError(`Version ${this.version} of the one-pass signature packet is unsupported.`); + } + + // A one-octet signature type. Signature types are described in + // Section 5.2.1. + this.signatureType = bytes[mypos++]; + + // A one-octet number describing the hash algorithm used. + this.hashAlgorithm = bytes[mypos++]; + + // A one-octet number describing the public-key algorithm used. + this.publicKeyAlgorithm = bytes[mypos++]; + + // An eight-octet number holding the Key ID of the signing key. + this.issuerKeyID = new KeyID(); + this.issuerKeyID.read(bytes.subarray(mypos, mypos + 8)); + mypos += 8; + + // A one-octet number holding a flag showing whether the signature + // is nested. A zero value indicates that the next packet is + // another One-Pass Signature packet that describes another + // signature to be applied to the same message data. + this.flags = bytes[mypos++]; + return this; + } + + /** + * creates a string representation of a one-pass signature packet + * @returns {Uint8Array} A Uint8Array representation of a one-pass signature packet. + */ + write() { + const start = new Uint8Array([VERSION, this.signatureType, this.hashAlgorithm, this.publicKeyAlgorithm]); + + const end = new Uint8Array([this.flags]); + + return util.concatUint8Array([start, this.issuerKeyID.write(), end]); + } + + calculateTrailer(...args) { + return fromAsync(async () => SignaturePacket.prototype.calculateTrailer.apply(await this.correspondingSig, args)); + } + + async verify() { + const correspondingSig = await this.correspondingSig; + if (!correspondingSig || correspondingSig.constructor.tag !== enums.packet.signature) { + throw new Error('Corresponding signature packet missing'); + } + if ( + correspondingSig.signatureType !== this.signatureType || + correspondingSig.hashAlgorithm !== this.hashAlgorithm || + correspondingSig.publicKeyAlgorithm !== this.publicKeyAlgorithm || + !correspondingSig.issuerKeyID.equals(this.issuerKeyID) + ) { + throw new Error('Corresponding signature packet does not match one-pass signature packet'); + } + correspondingSig.hashed = this.hashed; + return correspondingSig.verify.apply(correspondingSig, arguments); + } + } + + OnePassSignaturePacket.prototype.hash = SignaturePacket.prototype.hash; + OnePassSignaturePacket.prototype.toHash = SignaturePacket.prototype.toHash; + OnePassSignaturePacket.prototype.toSign = SignaturePacket.prototype.toSign; + + /** + * Instantiate a new packet given its tag + * @function newPacketFromTag + * @param {module:enums.packet} tag - Property value from {@link module:enums.packet} + * @param {Object} allowedPackets - mapping where keys are allowed packet tags, pointing to their Packet class + * @returns {Object} New packet object with type based on tag + * @throws {Error|UnsupportedError} for disallowed or unknown packets + */ + function newPacketFromTag(tag, allowedPackets) { + if (!allowedPackets[tag]) { + // distinguish between disallowed packets and unknown ones + let packetType; + try { + packetType = enums.read(enums.packet, tag); + } catch (e) { + throw new UnsupportedError(`Unknown packet type with tag: ${tag}`); + } + throw new Error(`Packet not allowed in this context: ${packetType}`); + } + return new allowedPackets[tag](); + } + + /** + * This class represents a list of openpgp packets. + * Take care when iterating over it - the packets themselves + * are stored as numerical indices. + * @extends Array + */ + class PacketList extends Array { + /** + * Parses the given binary data and returns a list of packets. + * Equivalent to calling `read` on an empty PacketList instance. + * @param {Uint8Array | ReadableStream} bytes - binary data to parse + * @param {Object} allowedPackets - mapping where keys are allowed packet tags, pointing to their Packet class + * @param {Object} [config] - full configuration, defaults to openpgp.config + * @returns {PacketList} parsed list of packets + * @throws on parsing errors + * @async + */ + static async fromBinary(bytes, allowedPackets, config$1 = config) { + const packets = new PacketList(); + await packets.read(bytes, allowedPackets, config$1); + return packets; + } + + /** + * Reads a stream of binary data and interprets it as a list of packets. + * @param {Uint8Array | ReadableStream} bytes - binary data to parse + * @param {Object} allowedPackets - mapping where keys are allowed packet tags, pointing to their Packet class + * @param {Object} [config] - full configuration, defaults to openpgp.config + * @throws on parsing errors + * @async + */ + async read(bytes, allowedPackets, config$1 = config) { + if (config$1.additionalAllowedPackets.length) { + allowedPackets = { ...allowedPackets, ...util.constructAllowedPackets(config$1.additionalAllowedPackets) }; + } + this.stream = transformPair(bytes, async (readable, writable) => { + const writer = getWriter(writable); + try { + while (true) { + await writer.ready; + const done = await readPackets(readable, async parsed => { + try { + if (parsed.tag === enums.packet.marker || parsed.tag === enums.packet.trust) { + // According to the spec, these packet types should be ignored and not cause parsing errors, even if not esplicitly allowed: + // - Marker packets MUST be ignored when received: https://github.com/openpgpjs/openpgpjs/issues/1145 + // - Trust packets SHOULD be ignored outside of keyrings (unsupported): https://datatracker.ietf.org/doc/html/rfc4880#section-5.10 + return; + } + const packet = newPacketFromTag(parsed.tag, allowedPackets); + packet.packets = new PacketList(); + packet.fromStream = util.isStream(parsed.packet); + await packet.read(parsed.packet, config$1); + await writer.write(packet); + } catch (e) { + const throwUnsupportedError = !config$1.ignoreUnsupportedPackets && e instanceof UnsupportedError; + const throwMalformedError = !config$1.ignoreMalformedPackets && !(e instanceof UnsupportedError); + if (throwUnsupportedError || throwMalformedError || supportsStreaming(parsed.tag)) { + // The packets that support streaming are the ones that contain message data. + // Those are also the ones we want to be more strict about and throw on parse errors + // (since we likely cannot process the message without these packets anyway). + await writer.abort(e); + } else { + const unparsedPacket = new UnparseablePacket(parsed.tag, parsed.packet); + await writer.write(unparsedPacket); + } + util.printDebugError(e); + } + }); + if (done) { + await writer.ready; + await writer.close(); + return; + } + } + } catch (e) { + await writer.abort(e); + } + }); + + // Wait until first few packets have been read + const reader = getReader(this.stream); + while (true) { + const { done, value } = await reader.read(); + if (!done) { + this.push(value); + } else { + this.stream = null; + } + if (done || supportsStreaming(value.constructor.tag)) { + break; + } + } + reader.releaseLock(); + } + + /** + * Creates a binary representation of openpgp objects contained within the + * class instance. + * @returns {Uint8Array} A Uint8Array containing valid openpgp packets. + */ + write() { + const arr = []; + + for (let i = 0; i < this.length; i++) { + const tag = this[i] instanceof UnparseablePacket ? this[i].tag : this[i].constructor.tag; + const packetbytes = this[i].write(); + if (util.isStream(packetbytes) && supportsStreaming(this[i].constructor.tag)) { + let buffer = []; + let bufferLength = 0; + const minLength = 512; + arr.push(writeTag(tag)); + arr.push(transform(packetbytes, value => { + buffer.push(value); + bufferLength += value.length; + if (bufferLength >= minLength) { + const powerOf2 = Math.min(Math.log(bufferLength) / Math.LN2 | 0, 30); + const chunkSize = 2 ** powerOf2; + const bufferConcat = util.concat([writePartialLength(powerOf2)].concat(buffer)); + buffer = [bufferConcat.subarray(1 + chunkSize)]; + bufferLength = buffer[0].length; + return bufferConcat.subarray(0, 1 + chunkSize); + } + }, () => util.concat([writeSimpleLength(bufferLength)].concat(buffer)))); + } else { + if (util.isStream(packetbytes)) { + let length = 0; + arr.push(transform(clone(packetbytes), value => { + length += value.length; + }, () => writeHeader(tag, length))); + } else { + arr.push(writeHeader(tag, packetbytes.length)); + } + arr.push(packetbytes); + } + } + + return util.concat(arr); + } + + /** + * Creates a new PacketList with all packets matching the given tag(s) + * @param {...module:enums.packet} tags - packet tags to look for + * @returns {PacketList} + */ + filterByTag(...tags) { + const filtered = new PacketList(); + + const handle = tag => packetType => tag === packetType; + + for (let i = 0; i < this.length; i++) { + if (tags.some(handle(this[i].constructor.tag))) { + filtered.push(this[i]); + } + } + + return filtered; + } + + /** + * Traverses packet list and returns first packet with matching tag + * @param {module:enums.packet} tag - The packet tag + * @returns {Packet|undefined} + */ + findPacket(tag) { + return this.find(packet => packet.constructor.tag === tag); + } + + /** + * Find indices of packets with the given tag(s) + * @param {...module:enums.packet} tags - packet tags to look for + * @returns {Integer[]} packet indices + */ + indexOfTag(...tags) { + const tagIndex = []; + const that = this; + + const handle = tag => packetType => tag === packetType; + + for (let i = 0; i < this.length; i++) { + if (tags.some(handle(that[i].constructor.tag))) { + tagIndex.push(i); + } + } + return tagIndex; + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + // A Compressed Data packet can contain the following packet types + const allowedPackets = /*#__PURE__*/ util.constructAllowedPackets([ + LiteralDataPacket, + OnePassSignaturePacket, + SignaturePacket + ]); + + /** + * Implementation of the Compressed Data Packet (Tag 8) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.6|RFC4880 5.6}: + * The Compressed Data packet contains compressed data. Typically, + * this packet is found as the contents of an encrypted packet, or following + * a Signature or One-Pass Signature packet, and contains a literal data packet. + */ + class CompressedDataPacket { + static get tag() { + return enums.packet.compressedData; + } + + /** + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + constructor(config$1 = config) { + /** + * List of packets + * @type {PacketList} + */ + this.packets = null; + /** + * Compression algorithm + * @type {enums.compression} + */ + this.algorithm = config$1.preferredCompressionAlgorithm; + + /** + * Compressed packet data + * @type {Uint8Array | ReadableStream} + */ + this.compressed = null; + + /** + * zip/zlib compression level, between 1 and 9 + */ + this.deflateLevel = config$1.deflateLevel; + } + + /** + * Parsing function for the packet. + * @param {Uint8Array | ReadableStream} bytes - Payload of a tag 8 packet + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + async read(bytes, config$1 = config) { + await parse(bytes, async reader => { + + // One octet that gives the algorithm used to compress the packet. + this.algorithm = await reader.readByte(); + + // Compressed data, which makes up the remainder of the packet. + this.compressed = reader.remainder(); + + await this.decompress(config$1); + }); + } + + + /** + * Return the compressed packet. + * @returns {Uint8Array | ReadableStream} Binary compressed packet. + */ + write() { + if (this.compressed === null) { + this.compress(); + } + + return util.concat([new Uint8Array([this.algorithm]), this.compressed]); + } + + + /** + * Decompression method for decompressing the compressed data + * read by read_packet + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + async decompress(config$1 = config) { + const compressionName = enums.read(enums.compression, this.algorithm); + const decompressionFn = decompress_fns[compressionName]; + if (!decompressionFn) { + throw new Error(`${compressionName} decompression not supported`); + } + + this.packets = await PacketList.fromBinary(decompressionFn(this.compressed), allowedPackets, config$1); + } + + /** + * Compress the packet data (member decompressedData) + */ + compress() { + const compressionName = enums.read(enums.compression, this.algorithm); + const compressionFn = compress_fns[compressionName]; + if (!compressionFn) { + throw new Error(`${compressionName} compression not supported`); + } + + this.compressed = compressionFn(this.packets.write(), this.deflateLevel); + } + } + + ////////////////////////// + // // + // Helper functions // + // // + ////////////////////////// + + + const nodeZlib = util.getNodeZlib(); + + function uncompressed(data) { + return data; + } + + function node_zlib(func, create, options = {}) { + return function (data) { + if (!util.isStream(data) || isArrayStream(data)) { + return fromAsync(() => readToEnd(data).then(data => { + return new Promise((resolve, reject) => { + func(data, options, (err, result) => { + if (err) return reject(err); + resolve(result); + }); + }); + })); + } + return nodeToWeb(webToNode(data).pipe(create(options))); + }; + } + + function pako_zlib(constructor, options = {}) { + return function(data) { + const obj = new constructor(options); + return transform(data, value => { + if (value.length) { + obj.push(value, Z_SYNC_FLUSH); + return obj.result; + } + }, () => { + if (constructor === Deflate) { + obj.push([], Z_FINISH); + return obj.result; + } + }); + }; + } + + function bzip2(func) { + return function(data) { + return fromAsync(async () => func(await readToEnd(data))); + }; + } + + const compress_fns = nodeZlib ? { + zip: /*#__PURE__*/ (compressed, level) => node_zlib(nodeZlib.deflateRaw, nodeZlib.createDeflateRaw, { level })(compressed), + zlib: /*#__PURE__*/ (compressed, level) => node_zlib(nodeZlib.deflate, nodeZlib.createDeflate, { level })(compressed) + } : { + zip: /*#__PURE__*/ (compressed, level) => pako_zlib(Deflate, { raw: true, level })(compressed), + zlib: /*#__PURE__*/ (compressed, level) => pako_zlib(Deflate, { level })(compressed) + }; + + const decompress_fns = nodeZlib ? { + uncompressed: uncompressed, + zip: /*#__PURE__*/ node_zlib(nodeZlib.inflateRaw, nodeZlib.createInflateRaw), + zlib: /*#__PURE__*/ node_zlib(nodeZlib.inflate, nodeZlib.createInflate), + bzip2: /*#__PURE__*/ bzip2(lib_4) + } : { + uncompressed: uncompressed, + zip: /*#__PURE__*/ pako_zlib(Inflate, { raw: true }), + zlib: /*#__PURE__*/ pako_zlib(Inflate), + bzip2: /*#__PURE__*/ bzip2(lib_4) + }; + + // GPG4Browsers - An OpenPGP implementation in javascript + + // A SEIP packet can contain the following packet types + const allowedPackets$1 = /*#__PURE__*/ util.constructAllowedPackets([ + LiteralDataPacket, + CompressedDataPacket, + OnePassSignaturePacket, + SignaturePacket + ]); + + const VERSION$1 = 1; // A one-octet version number of the data packet. + + /** + * Implementation of the Sym. Encrypted Integrity Protected Data Packet (Tag 18) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.13|RFC4880 5.13}: + * The Symmetrically Encrypted Integrity Protected Data packet is + * a variant of the Symmetrically Encrypted Data packet. It is a new feature + * created for OpenPGP that addresses the problem of detecting a modification to + * encrypted data. It is used in combination with a Modification Detection Code + * packet. + */ + class SymEncryptedIntegrityProtectedDataPacket { + static get tag() { + return enums.packet.symEncryptedIntegrityProtectedData; + } + + constructor() { + this.version = VERSION$1; + this.encrypted = null; + this.packets = null; + } + + async read(bytes) { + await parse(bytes, async reader => { + const version = await reader.readByte(); + // - A one-octet version number. The only currently defined value is 1. + if (version !== VERSION$1) { + throw new UnsupportedError(`Version ${version} of the SEIP packet is unsupported.`); + } + + // - Encrypted data, the output of the selected symmetric-key cipher + // operating in Cipher Feedback mode with shift amount equal to the + // block size of the cipher (CFB-n where n is the block size). + this.encrypted = reader.remainder(); + }); + } + + write() { + return util.concat([new Uint8Array([VERSION$1]), this.encrypted]); + } + + /** + * Encrypt the payload in the packet. + * @param {enums.symmetric} sessionKeyAlgorithm - The symmetric encryption algorithm to use + * @param {Uint8Array} key - The key of cipher blocksize length to be used + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} + * @throws {Error} on encryption failure + * @async + */ + async encrypt(sessionKeyAlgorithm, key, config$1 = config) { + const { blockSize } = mod.getCipher(sessionKeyAlgorithm); + + let bytes = this.packets.write(); + if (isArrayStream(bytes)) bytes = await readToEnd(bytes); + const prefix = await mod.getPrefixRandom(sessionKeyAlgorithm); + const mdc = new Uint8Array([0xD3, 0x14]); // modification detection code packet + + const tohash = util.concat([prefix, bytes, mdc]); + const hash = await mod.hash.sha1(passiveClone(tohash)); + const plaintext = util.concat([tohash, hash]); + + this.encrypted = await mod.mode.cfb.encrypt(sessionKeyAlgorithm, key, plaintext, new Uint8Array(blockSize), config$1); + return true; + } + + /** + * Decrypts the encrypted data contained in the packet. + * @param {enums.symmetric} sessionKeyAlgorithm - The selected symmetric encryption algorithm to be used + * @param {Uint8Array} key - The key of cipher blocksize length to be used + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} + * @throws {Error} on decryption failure + * @async + */ + async decrypt(sessionKeyAlgorithm, key, config$1 = config) { + const { blockSize } = mod.getCipher(sessionKeyAlgorithm); + let encrypted = clone(this.encrypted); + if (isArrayStream(encrypted)) encrypted = await readToEnd(encrypted); + const decrypted = await mod.mode.cfb.decrypt(sessionKeyAlgorithm, key, encrypted, new Uint8Array(blockSize)); + + // there must be a modification detection code packet as the + // last packet and everything gets hashed except the hash itself + const realHash = slice(passiveClone(decrypted), -20); + const tohash = slice(decrypted, 0, -20); + const verifyHash = Promise.all([ + readToEnd(await mod.hash.sha1(passiveClone(tohash))), + readToEnd(realHash) + ]).then(([hash, mdc]) => { + if (!util.equalsUint8Array(hash, mdc)) { + throw new Error('Modification detected.'); + } + return new Uint8Array(); + }); + const bytes = slice(tohash, blockSize + 2); // Remove random prefix + let packetbytes = slice(bytes, 0, -2); // Remove MDC packet + packetbytes = concat([packetbytes, fromAsync(() => verifyHash)]); + if (!util.isStream(encrypted) || !config$1.allowUnauthenticatedStream) { + packetbytes = await readToEnd(packetbytes); + } + this.packets = await PacketList.fromBinary(packetbytes, allowedPackets$1, config$1); + return true; + } + } + + // OpenPGP.js - An OpenPGP implementation in javascript + + // An AEAD-encrypted Data packet can contain the following packet types + const allowedPackets$2 = /*#__PURE__*/ util.constructAllowedPackets([ + LiteralDataPacket, + CompressedDataPacket, + OnePassSignaturePacket, + SignaturePacket + ]); + + const VERSION$2 = 1; // A one-octet version number of the data packet. + + /** + * Implementation of the Symmetrically Encrypted Authenticated Encryption with + * Additional Data (AEAD) Protected Data Packet + * + * {@link https://tools.ietf.org/html/draft-ford-openpgp-format-00#section-2.1}: + * AEAD Protected Data Packet + */ + class AEADEncryptedDataPacket { + static get tag() { + return enums.packet.aeadEncryptedData; + } + + constructor() { + this.version = VERSION$2; + /** @type {enums.symmetric} */ + this.cipherAlgorithm = null; + /** @type {enums.aead} */ + this.aeadAlgorithm = enums.aead.eax; + this.chunkSizeByte = null; + this.iv = null; + this.encrypted = null; + this.packets = null; + } + + /** + * Parse an encrypted payload of bytes in the order: version, IV, ciphertext (see specification) + * @param {Uint8Array | ReadableStream} bytes + * @throws {Error} on parsing failure + */ + async read(bytes) { + await parse(bytes, async reader => { + const version = await reader.readByte(); + if (version !== VERSION$2) { // The only currently defined value is 1. + throw new UnsupportedError(`Version ${version} of the AEAD-encrypted data packet is not supported.`); + } + this.cipherAlgorithm = await reader.readByte(); + this.aeadAlgorithm = await reader.readByte(); + this.chunkSizeByte = await reader.readByte(); + + const mode = mod.getAEADMode(this.aeadAlgorithm); + this.iv = await reader.readBytes(mode.ivLength); + this.encrypted = reader.remainder(); + }); + } + + /** + * Write the encrypted payload of bytes in the order: version, IV, ciphertext (see specification) + * @returns {Uint8Array | ReadableStream} The encrypted payload. + */ + write() { + return util.concat([new Uint8Array([this.version, this.cipherAlgorithm, this.aeadAlgorithm, this.chunkSizeByte]), this.iv, this.encrypted]); + } + + /** + * Decrypt the encrypted payload. + * @param {enums.symmetric} sessionKeyAlgorithm - The session key's cipher algorithm + * @param {Uint8Array} key - The session key used to encrypt the payload + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} if decryption was not successful + * @async + */ + async decrypt(sessionKeyAlgorithm, key, config$1 = config) { + this.packets = await PacketList.fromBinary( + await this.crypt('decrypt', key, clone(this.encrypted)), + allowedPackets$2, + config$1 + ); + } + + /** + * Encrypt the packet payload. + * @param {enums.symmetric} sessionKeyAlgorithm - The session key's cipher algorithm + * @param {Uint8Array} key - The session key used to encrypt the payload + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} if encryption was not successful + * @async + */ + async encrypt(sessionKeyAlgorithm, key, config$1 = config) { + this.cipherAlgorithm = sessionKeyAlgorithm; + + const { ivLength } = mod.getAEADMode(this.aeadAlgorithm); + this.iv = mod.random.getRandomBytes(ivLength); // generate new random IV + this.chunkSizeByte = config$1.aeadChunkSizeByte; + const data = this.packets.write(); + this.encrypted = await this.crypt('encrypt', key, data); + } + + /** + * En/decrypt the payload. + * @param {encrypt|decrypt} fn - Whether to encrypt or decrypt + * @param {Uint8Array} key - The session key used to en/decrypt the payload + * @param {Uint8Array | ReadableStream} data - The data to en/decrypt + * @returns {Promise>} + * @async + */ + async crypt(fn, key, data) { + const mode = mod.getAEADMode(this.aeadAlgorithm); + const modeInstance = await mode(this.cipherAlgorithm, key); + const tagLengthIfDecrypting = fn === 'decrypt' ? mode.tagLength : 0; + const tagLengthIfEncrypting = fn === 'encrypt' ? mode.tagLength : 0; + const chunkSize = 2 ** (this.chunkSizeByte + 6) + tagLengthIfDecrypting; // ((uint64_t)1 << (c + 6)) + const adataBuffer = new ArrayBuffer(21); + const adataArray = new Uint8Array(adataBuffer, 0, 13); + const adataTagArray = new Uint8Array(adataBuffer); + const adataView = new DataView(adataBuffer); + const chunkIndexArray = new Uint8Array(adataBuffer, 5, 8); + adataArray.set([0xC0 | AEADEncryptedDataPacket.tag, this.version, this.cipherAlgorithm, this.aeadAlgorithm, this.chunkSizeByte], 0); + let chunkIndex = 0; + let latestPromise = Promise.resolve(); + let cryptedBytes = 0; + let queuedBytes = 0; + const iv = this.iv; + return transformPair(data, async (readable, writable) => { + if (util.isStream(readable) !== 'array') { + const buffer = new TransformStream({}, { + highWaterMark: util.getHardwareConcurrency() * 2 ** (this.chunkSizeByte + 6), + size: array => array.length + }); + pipe(buffer.readable, writable); + writable = buffer.writable; + } + const reader = getReader(readable); + const writer = getWriter(writable); + try { + while (true) { + let chunk = await reader.readBytes(chunkSize + tagLengthIfDecrypting) || new Uint8Array(); + const finalChunk = chunk.subarray(chunk.length - tagLengthIfDecrypting); + chunk = chunk.subarray(0, chunk.length - tagLengthIfDecrypting); + let cryptedPromise; + let done; + if (!chunkIndex || chunk.length) { + reader.unshift(finalChunk); + cryptedPromise = modeInstance[fn](chunk, mode.getNonce(iv, chunkIndexArray), adataArray); + queuedBytes += chunk.length - tagLengthIfDecrypting + tagLengthIfEncrypting; + } else { + // After the last chunk, we either encrypt a final, empty + // data chunk to get the final authentication tag or + // validate that final authentication tag. + adataView.setInt32(13 + 4, cryptedBytes); // Should be setInt64(13, ...) + cryptedPromise = modeInstance[fn](finalChunk, mode.getNonce(iv, chunkIndexArray), adataTagArray); + queuedBytes += tagLengthIfEncrypting; + done = true; + } + cryptedBytes += chunk.length - tagLengthIfDecrypting; + // eslint-disable-next-line no-loop-func + latestPromise = latestPromise.then(() => cryptedPromise).then(async crypted => { + await writer.ready; + await writer.write(crypted); + queuedBytes -= crypted.length; + }).catch(err => writer.abort(err)); + if (done || queuedBytes > writer.desiredSize) { + await latestPromise; // Respect backpressure + } + if (!done) { + adataView.setInt32(5 + 4, ++chunkIndex); // Should be setInt64(5, ...) + } else { + await writer.close(); + break; + } + } + } catch (e) { + await writer.abort(e); + } + }); + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + const VERSION$3 = 3; + + /** + * Public-Key Encrypted Session Key Packets (Tag 1) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.1|RFC4880 5.1}: + * A Public-Key Encrypted Session Key packet holds the session key + * used to encrypt a message. Zero or more Public-Key Encrypted Session Key + * packets and/or Symmetric-Key Encrypted Session Key packets may precede a + * Symmetrically Encrypted Data Packet, which holds an encrypted message. The + * message is encrypted with the session key, and the session key is itself + * encrypted and stored in the Encrypted Session Key packet(s). The + * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted + * Session Key packet for each OpenPGP key to which the message is encrypted. + * The recipient of the message finds a session key that is encrypted to their + * public key, decrypts the session key, and then uses the session key to + * decrypt the message. + */ + class PublicKeyEncryptedSessionKeyPacket { + static get tag() { + return enums.packet.publicKeyEncryptedSessionKey; + } + + constructor() { + this.version = 3; + + this.publicKeyID = new KeyID(); + this.publicKeyAlgorithm = null; + + this.sessionKey = null; + /** + * Algorithm to encrypt the message with + * @type {enums.symmetric} + */ + this.sessionKeyAlgorithm = null; + + /** @type {Object} */ + this.encrypted = {}; + } + + /** + * Parsing function for a publickey encrypted session key packet (tag 1). + * + * @param {Uint8Array} bytes - Payload of a tag 1 packet + */ + read(bytes) { + this.version = bytes[0]; + if (this.version !== VERSION$3) { + throw new UnsupportedError(`Version ${this.version} of the PKESK packet is unsupported.`); + } + this.publicKeyID.read(bytes.subarray(1, bytes.length)); + this.publicKeyAlgorithm = bytes[9]; + this.encrypted = mod.parseEncSessionKeyParams(this.publicKeyAlgorithm, bytes.subarray(10)); + } + + /** + * Create a binary representation of a tag 1 packet + * + * @returns {Uint8Array} The Uint8Array representation. + */ + write() { + const arr = [ + new Uint8Array([this.version]), + this.publicKeyID.write(), + new Uint8Array([this.publicKeyAlgorithm]), + mod.serializeParams(this.publicKeyAlgorithm, this.encrypted) + ]; + + return util.concatUint8Array(arr); + } + + /** + * Encrypt session key packet + * @param {PublicKeyPacket} key - Public key + * @throws {Error} if encryption failed + * @async + */ + async encrypt(key) { + const data = util.concatUint8Array([ + new Uint8Array([enums.write(enums.symmetric, this.sessionKeyAlgorithm)]), + this.sessionKey, + util.writeChecksum(this.sessionKey) + ]); + const algo = enums.write(enums.publicKey, this.publicKeyAlgorithm); + this.encrypted = await mod.publicKeyEncrypt( + algo, key.publicParams, data, key.getFingerprintBytes()); + } + + /** + * Decrypts the session key (only for public key encrypted session key packets (tag 1) + * @param {SecretKeyPacket} key - decrypted private key + * @param {Object} [randomSessionKey] - Bogus session key to use in case of sensitive decryption error, or if the decrypted session key is of a different type/size. + * This is needed for constant-time processing. Expected object of the form: { sessionKey: Uint8Array, sessionKeyAlgorithm: enums.symmetric } + * @throws {Error} if decryption failed, unless `randomSessionKey` is given + * @async + */ + async decrypt(key, randomSessionKey) { + // check that session key algo matches the secret key algo + if (this.publicKeyAlgorithm !== key.algorithm) { + throw new Error('Decryption error'); + } + + const randomPayload = randomSessionKey ? util.concatUint8Array([ + new Uint8Array([randomSessionKey.sessionKeyAlgorithm]), + randomSessionKey.sessionKey, + util.writeChecksum(randomSessionKey.sessionKey) + ]) : null; + const decoded = await mod.publicKeyDecrypt(this.publicKeyAlgorithm, key.publicParams, key.privateParams, this.encrypted, key.getFingerprintBytes(), randomPayload); + const symmetricAlgoByte = decoded[0]; + const sessionKey = decoded.subarray(1, decoded.length - 2); + const checksum = decoded.subarray(decoded.length - 2); + const computedChecksum = util.writeChecksum(sessionKey); + const isValidChecksum = computedChecksum[0] === checksum[0] & computedChecksum[1] === checksum[1]; + + if (randomSessionKey) { + // We must not leak info about the validity of the decrypted checksum or cipher algo. + // The decrypted session key must be of the same algo and size as the random session key, otherwise we discard it and use the random data. + const isValidPayload = isValidChecksum & symmetricAlgoByte === randomSessionKey.sessionKeyAlgorithm & sessionKey.length === randomSessionKey.sessionKey.length; + this.sessionKeyAlgorithm = util.selectUint8(isValidPayload, symmetricAlgoByte, randomSessionKey.sessionKeyAlgorithm); + this.sessionKey = util.selectUint8Array(isValidPayload, sessionKey, randomSessionKey.sessionKey); + + } else { + const isValidPayload = isValidChecksum && enums.read(enums.symmetric, symmetricAlgoByte); + if (isValidPayload) { + this.sessionKey = sessionKey; + this.sessionKeyAlgorithm = symmetricAlgoByte; + } else { + throw new Error('Decryption error'); + } + } + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + class S2K { + /** + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + constructor(config$1 = config) { + /** + * Hash function identifier, or 0 for gnu-dummy keys + * @type {module:enums.hash | 0} + */ + this.algorithm = enums.hash.sha256; + /** + * enums.s2k identifier or 'gnu-dummy' + * @type {String} + */ + this.type = 'iterated'; + /** @type {Integer} */ + this.c = config$1.s2kIterationCountByte; + /** Eight bytes of salt in a binary string. + * @type {Uint8Array} + */ + this.salt = null; + } + + getCount() { + // Exponent bias, defined in RFC4880 + const expbias = 6; + + return (16 + (this.c & 15)) << ((this.c >> 4) + expbias); + } + + /** + * Parsing function for a string-to-key specifier ({@link https://tools.ietf.org/html/rfc4880#section-3.7|RFC 4880 3.7}). + * @param {Uint8Array} bytes - Payload of string-to-key specifier + * @returns {Integer} Actual length of the object. + */ + read(bytes) { + let i = 0; + this.type = enums.read(enums.s2k, bytes[i++]); + this.algorithm = bytes[i++]; + + switch (this.type) { + case 'simple': + break; + + case 'salted': + this.salt = bytes.subarray(i, i + 8); + i += 8; + break; + + case 'iterated': + this.salt = bytes.subarray(i, i + 8); + i += 8; + + // Octet 10: count, a one-octet, coded value + this.c = bytes[i++]; + break; + + case 'gnu': + if (util.uint8ArrayToString(bytes.subarray(i, i + 3)) === 'GNU') { + i += 3; // GNU + const gnuExtType = 1000 + bytes[i++]; + if (gnuExtType === 1001) { + this.type = 'gnu-dummy'; + // GnuPG extension mode 1001 -- don't write secret key at all + } else { + throw new Error('Unknown s2k gnu protection mode.'); + } + } else { + throw new Error('Unknown s2k type.'); + } + break; + + default: + throw new Error('Unknown s2k type.'); + } + + return i; + } + + /** + * Serializes s2k information + * @returns {Uint8Array} Binary representation of s2k. + */ + write() { + if (this.type === 'gnu-dummy') { + return new Uint8Array([101, 0, ...util.stringToUint8Array('GNU'), 1]); + } + const arr = [new Uint8Array([enums.write(enums.s2k, this.type), this.algorithm])]; + + switch (this.type) { + case 'simple': + break; + case 'salted': + arr.push(this.salt); + break; + case 'iterated': + arr.push(this.salt); + arr.push(new Uint8Array([this.c])); + break; + case 'gnu': + throw new Error('GNU s2k type not supported.'); + default: + throw new Error('Unknown s2k type.'); + } + + return util.concatUint8Array(arr); + } + + /** + * Produces a key using the specified passphrase and the defined + * hashAlgorithm + * @param {String} passphrase - Passphrase containing user input + * @returns {Promise} Produced key with a length corresponding to. + * hashAlgorithm hash length + * @async + */ + async produceKey(passphrase, numBytes) { + passphrase = util.encodeUTF8(passphrase); + + const arr = []; + let rlength = 0; + + let prefixlen = 0; + while (rlength < numBytes) { + let toHash; + switch (this.type) { + case 'simple': + toHash = util.concatUint8Array([new Uint8Array(prefixlen), passphrase]); + break; + case 'salted': + toHash = util.concatUint8Array([new Uint8Array(prefixlen), this.salt, passphrase]); + break; + case 'iterated': { + const data = util.concatUint8Array([this.salt, passphrase]); + let datalen = data.length; + const count = Math.max(this.getCount(), datalen); + toHash = new Uint8Array(prefixlen + count); + toHash.set(data, prefixlen); + for (let pos = prefixlen + datalen; pos < count; pos += datalen, datalen *= 2) { + toHash.copyWithin(pos, prefixlen, pos); + } + break; + } + case 'gnu': + throw new Error('GNU s2k type not supported.'); + default: + throw new Error('Unknown s2k type.'); + } + const result = await mod.hash.digest(this.algorithm, toHash); + arr.push(result); + rlength += result.length; + prefixlen++; + } + + return util.concatUint8Array(arr).subarray(0, numBytes); + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Symmetric-Key Encrypted Session Key Packets (Tag 3) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.3|RFC4880 5.3}: + * The Symmetric-Key Encrypted Session Key packet holds the + * symmetric-key encryption of a session key used to encrypt a message. + * Zero or more Public-Key Encrypted Session Key packets and/or + * Symmetric-Key Encrypted Session Key packets may precede a + * Symmetrically Encrypted Data packet that holds an encrypted message. + * The message is encrypted with a session key, and the session key is + * itself encrypted and stored in the Encrypted Session Key packet or + * the Symmetric-Key Encrypted Session Key packet. + */ + class SymEncryptedSessionKeyPacket { + static get tag() { + return enums.packet.symEncryptedSessionKey; + } + + /** + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + constructor(config$1 = config) { + this.version = config$1.aeadProtect ? 5 : 4; + this.sessionKey = null; + /** + * Algorithm to encrypt the session key with + * @type {enums.symmetric} + */ + this.sessionKeyEncryptionAlgorithm = null; + /** + * Algorithm to encrypt the message with + * @type {enums.symmetric} + */ + this.sessionKeyAlgorithm = enums.symmetric.aes256; + /** + * AEAD mode to encrypt the session key with (if AEAD protection is enabled) + * @type {enums.aead} + */ + this.aeadAlgorithm = enums.write(enums.aead, config$1.preferredAEADAlgorithm); + this.encrypted = null; + this.s2k = null; + this.iv = null; + } + + /** + * Parsing function for a symmetric encrypted session key packet (tag 3). + * + * @param {Uint8Array} bytes - Payload of a tag 3 packet + */ + read(bytes) { + let offset = 0; + + // A one-octet version number. The only currently defined version is 4. + this.version = bytes[offset++]; + if (this.version !== 4 && this.version !== 5) { + throw new UnsupportedError(`Version ${this.version} of the SKESK packet is unsupported.`); + } + + // A one-octet number describing the symmetric algorithm used. + const algo = bytes[offset++]; + + if (this.version === 5) { + // A one-octet AEAD algorithm. + this.aeadAlgorithm = bytes[offset++]; + } + + // A string-to-key (S2K) specifier, length as defined above. + this.s2k = new S2K(); + offset += this.s2k.read(bytes.subarray(offset, bytes.length)); + + if (this.version === 5) { + const mode = mod.getAEADMode(this.aeadAlgorithm); + + // A starting initialization vector of size specified by the AEAD + // algorithm. + this.iv = bytes.subarray(offset, offset += mode.ivLength); + } + + // The encrypted session key itself, which is decrypted with the + // string-to-key object. This is optional in version 4. + if (this.version === 5 || offset < bytes.length) { + this.encrypted = bytes.subarray(offset, bytes.length); + this.sessionKeyEncryptionAlgorithm = algo; + } else { + this.sessionKeyAlgorithm = algo; + } + } + + /** + * Create a binary representation of a tag 3 packet + * + * @returns {Uint8Array} The Uint8Array representation. + */ + write() { + const algo = this.encrypted === null ? + this.sessionKeyAlgorithm : + this.sessionKeyEncryptionAlgorithm; + + let bytes; + + if (this.version === 5) { + bytes = util.concatUint8Array([new Uint8Array([this.version, algo, this.aeadAlgorithm]), this.s2k.write(), this.iv, this.encrypted]); + } else { + bytes = util.concatUint8Array([new Uint8Array([this.version, algo]), this.s2k.write()]); + + if (this.encrypted !== null) { + bytes = util.concatUint8Array([bytes, this.encrypted]); + } + } + + return bytes; + } + + /** + * Decrypts the session key with the given passphrase + * @param {String} passphrase - The passphrase in string form + * @throws {Error} if decryption was not successful + * @async + */ + async decrypt(passphrase) { + const algo = this.sessionKeyEncryptionAlgorithm !== null ? + this.sessionKeyEncryptionAlgorithm : + this.sessionKeyAlgorithm; + + const { blockSize, keySize } = mod.getCipher(algo); + const key = await this.s2k.produceKey(passphrase, keySize); + + if (this.version === 5) { + const mode = mod.getAEADMode(this.aeadAlgorithm); + const adata = new Uint8Array([0xC0 | SymEncryptedSessionKeyPacket.tag, this.version, this.sessionKeyEncryptionAlgorithm, this.aeadAlgorithm]); + const modeInstance = await mode(algo, key); + this.sessionKey = await modeInstance.decrypt(this.encrypted, this.iv, adata); + } else if (this.encrypted !== null) { + const decrypted = await mod.mode.cfb.decrypt(algo, key, this.encrypted, new Uint8Array(blockSize)); + + this.sessionKeyAlgorithm = enums.write(enums.symmetric, decrypted[0]); + this.sessionKey = decrypted.subarray(1, decrypted.length); + } else { + this.sessionKey = key; + } + } + + /** + * Encrypts the session key with the given passphrase + * @param {String} passphrase - The passphrase in string form + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} if encryption was not successful + * @async + */ + async encrypt(passphrase, config$1 = config) { + const algo = this.sessionKeyEncryptionAlgorithm !== null ? + this.sessionKeyEncryptionAlgorithm : + this.sessionKeyAlgorithm; + + this.sessionKeyEncryptionAlgorithm = algo; + + this.s2k = new S2K(config$1); + this.s2k.salt = mod.random.getRandomBytes(8); + + const { blockSize, keySize } = mod.getCipher(algo); + const encryptionKey = await this.s2k.produceKey(passphrase, keySize); + + if (this.sessionKey === null) { + this.sessionKey = mod.generateSessionKey(this.sessionKeyAlgorithm); + } + + if (this.version === 5) { + const mode = mod.getAEADMode(this.aeadAlgorithm); + this.iv = mod.random.getRandomBytes(mode.ivLength); // generate new random IV + const associatedData = new Uint8Array([0xC0 | SymEncryptedSessionKeyPacket.tag, this.version, this.sessionKeyEncryptionAlgorithm, this.aeadAlgorithm]); + const modeInstance = await mode(algo, encryptionKey); + this.encrypted = await modeInstance.encrypt(this.sessionKey, this.iv, associatedData); + } else { + const toEncrypt = util.concatUint8Array([ + new Uint8Array([this.sessionKeyAlgorithm]), + this.sessionKey + ]); + this.encrypted = await mod.mode.cfb.encrypt(algo, encryptionKey, toEncrypt, new Uint8Array(blockSize), config$1); + } + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Implementation of the Key Material Packet (Tag 5,6,7,14) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.5|RFC4480 5.5}: + * A key material packet contains all the information about a public or + * private key. There are four variants of this packet type, and two + * major versions. + * + * A Public-Key packet starts a series of packets that forms an OpenPGP + * key (sometimes called an OpenPGP certificate). + */ + class PublicKeyPacket { + static get tag() { + return enums.packet.publicKey; + } + + /** + * @param {Date} [date] - Creation date + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + constructor(date = new Date(), config$1 = config) { + /** + * Packet version + * @type {Integer} + */ + this.version = config$1.v5Keys ? 5 : 4; + /** + * Key creation date. + * @type {Date} + */ + this.created = util.normalizeDate(date); + /** + * Public key algorithm. + * @type {enums.publicKey} + */ + this.algorithm = null; + /** + * Algorithm specific public params + * @type {Object} + */ + this.publicParams = null; + /** + * Time until expiration in days (V3 only) + * @type {Integer} + */ + this.expirationTimeV3 = 0; + /** + * Fingerprint bytes + * @type {Uint8Array} + */ + this.fingerprint = null; + /** + * KeyID + * @type {module:type/keyid~KeyID} + */ + this.keyID = null; + } + + /** + * Create a PublicKeyPacket from a SecretKeyPacket + * @param {SecretKeyPacket} secretKeyPacket - key packet to convert + * @returns {PublicKeyPacket} public key packet + * @static + */ + static fromSecretKeyPacket(secretKeyPacket) { + const keyPacket = new PublicKeyPacket(); + const { version, created, algorithm, publicParams, keyID, fingerprint } = secretKeyPacket; + keyPacket.version = version; + keyPacket.created = created; + keyPacket.algorithm = algorithm; + keyPacket.publicParams = publicParams; + keyPacket.keyID = keyID; + keyPacket.fingerprint = fingerprint; + return keyPacket; + } + + /** + * Internal Parser for public keys as specified in {@link https://tools.ietf.org/html/rfc4880#section-5.5.2|RFC 4880 section 5.5.2 Public-Key Packet Formats} + * @param {Uint8Array} bytes - Input array to read the packet from + * @returns {Object} This object with attributes set by the parser + * @async + */ + async read(bytes) { + let pos = 0; + // A one-octet version number (3, 4 or 5). + this.version = bytes[pos++]; + + if (this.version === 4 || this.version === 5) { + // - A four-octet number denoting the time that the key was created. + this.created = util.readDate(bytes.subarray(pos, pos + 4)); + pos += 4; + + // - A one-octet number denoting the public-key algorithm of this key. + this.algorithm = bytes[pos++]; + + if (this.version === 5) { + // - A four-octet scalar octet count for the following key material. + pos += 4; + } + + // - A series of values comprising the key material. + const { read, publicParams } = mod.parsePublicKeyParams(this.algorithm, bytes.subarray(pos)); + this.publicParams = publicParams; + pos += read; + + // we set the fingerprint and keyID already to make it possible to put together the key packets directly in the Key constructor + await this.computeFingerprintAndKeyID(); + return pos; + } + throw new UnsupportedError(`Version ${this.version} of the key packet is unsupported.`); + } + + /** + * Creates an OpenPGP public key packet for the given key. + * @returns {Uint8Array} Bytes encoding the public key OpenPGP packet. + */ + write() { + const arr = []; + // Version + arr.push(new Uint8Array([this.version])); + arr.push(util.writeDate(this.created)); + // A one-octet number denoting the public-key algorithm of this key + arr.push(new Uint8Array([this.algorithm])); + + const params = mod.serializeParams(this.algorithm, this.publicParams); + if (this.version === 5) { + // A four-octet scalar octet count for the following key material + arr.push(util.writeNumber(params.length, 4)); + } + // Algorithm-specific params + arr.push(params); + return util.concatUint8Array(arr); + } + + /** + * Write packet in order to be hashed; either for a signature or a fingerprint + * @param {Integer} version - target version of signature or key + */ + writeForHash(version) { + const bytes = this.writePublicKey(); + + if (version === 5) { + return util.concatUint8Array([new Uint8Array([0x9A]), util.writeNumber(bytes.length, 4), bytes]); + } + return util.concatUint8Array([new Uint8Array([0x99]), util.writeNumber(bytes.length, 2), bytes]); + } + + /** + * Check whether secret-key data is available in decrypted form. Returns null for public keys. + * @returns {Boolean|null} + */ + isDecrypted() { + return null; + } + + /** + * Returns the creation time of the key + * @returns {Date} + */ + getCreationTime() { + return this.created; + } + + /** + * Return the key ID of the key + * @returns {module:type/keyid~KeyID} The 8-byte key ID + */ + getKeyID() { + return this.keyID; + } + + /** + * Computes and set the key ID and fingerprint of the key + * @async + */ + async computeFingerprintAndKeyID() { + await this.computeFingerprint(); + this.keyID = new KeyID(); + + if (this.version === 5) { + this.keyID.read(this.fingerprint.subarray(0, 8)); + } else if (this.version === 4) { + this.keyID.read(this.fingerprint.subarray(12, 20)); + } else { + throw new Error('Unsupported key version'); + } + } + + /** + * Computes and set the fingerprint of the key + */ + async computeFingerprint() { + const toHash = this.writeForHash(this.version); + + if (this.version === 5) { + this.fingerprint = await mod.hash.sha256(toHash); + } else if (this.version === 4) { + this.fingerprint = await mod.hash.sha1(toHash); + } else { + throw new Error('Unsupported key version'); + } + } + + /** + * Returns the fingerprint of the key, as an array of bytes + * @returns {Uint8Array} A Uint8Array containing the fingerprint + */ + getFingerprintBytes() { + return this.fingerprint; + } + + /** + * Calculates and returns the fingerprint of the key, as a string + * @returns {String} A string containing the fingerprint in lowercase hex + */ + getFingerprint() { + return util.uint8ArrayToHex(this.getFingerprintBytes()); + } + + /** + * Calculates whether two keys have the same fingerprint without actually calculating the fingerprint + * @returns {Boolean} Whether the two keys have the same version and public key data. + */ + hasSameFingerprintAs(other) { + return this.version === other.version && util.equalsUint8Array(this.writePublicKey(), other.writePublicKey()); + } + + /** + * Returns algorithm information + * @returns {Object} An object of the form {algorithm: String, bits:int, curve:String}. + */ + getAlgorithmInfo() { + const result = {}; + result.algorithm = enums.read(enums.publicKey, this.algorithm); + // RSA, DSA or ElGamal public modulo + const modulo = this.publicParams.n || this.publicParams.p; + if (modulo) { + result.bits = util.uint8ArrayBitLength(modulo); + } else { + result.curve = this.publicParams.oid.getName(); + } + return result; + } + } + + /** + * Alias of read() + * @see PublicKeyPacket#read + */ + PublicKeyPacket.prototype.readPublicKey = PublicKeyPacket.prototype.read; + + /** + * Alias of write() + * @see PublicKeyPacket#write + */ + PublicKeyPacket.prototype.writePublicKey = PublicKeyPacket.prototype.write; + + // GPG4Browsers - An OpenPGP implementation in javascript + + // A SE packet can contain the following packet types + const allowedPackets$3 = /*#__PURE__*/ util.constructAllowedPackets([ + LiteralDataPacket, + CompressedDataPacket, + OnePassSignaturePacket, + SignaturePacket + ]); + + /** + * Implementation of the Symmetrically Encrypted Data Packet (Tag 9) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.7|RFC4880 5.7}: + * The Symmetrically Encrypted Data packet contains data encrypted with a + * symmetric-key algorithm. When it has been decrypted, it contains other + * packets (usually a literal data packet or compressed data packet, but in + * theory other Symmetrically Encrypted Data packets or sequences of packets + * that form whole OpenPGP messages). + */ + class SymmetricallyEncryptedDataPacket { + static get tag() { + return enums.packet.symmetricallyEncryptedData; + } + + constructor() { + /** + * Encrypted secret-key data + */ + this.encrypted = null; + /** + * Decrypted packets contained within. + * @type {PacketList} + */ + this.packets = null; + } + + read(bytes) { + this.encrypted = bytes; + } + + write() { + return this.encrypted; + } + + /** + * Decrypt the symmetrically-encrypted packet data + * See {@link https://tools.ietf.org/html/rfc4880#section-9.2|RFC 4880 9.2} for algorithms. + * @param {module:enums.symmetric} sessionKeyAlgorithm - Symmetric key algorithm to use + * @param {Uint8Array} key - The key of cipher blocksize length to be used + * @param {Object} [config] - Full configuration, defaults to openpgp.config + + * @throws {Error} if decryption was not successful + * @async + */ + async decrypt(sessionKeyAlgorithm, key, config$1 = config) { + // If MDC errors are not being ignored, all missing MDC packets in symmetrically encrypted data should throw an error + if (!config$1.allowUnauthenticatedMessages) { + throw new Error('Message is not authenticated.'); + } + + const { blockSize } = mod.getCipher(sessionKeyAlgorithm); + const encrypted = await readToEnd(clone(this.encrypted)); + const decrypted = await mod.mode.cfb.decrypt(sessionKeyAlgorithm, key, + encrypted.subarray(blockSize + 2), + encrypted.subarray(2, blockSize + 2) + ); + + this.packets = await PacketList.fromBinary(decrypted, allowedPackets$3, config$1); + } + + /** + * Encrypt the symmetrically-encrypted packet data + * See {@link https://tools.ietf.org/html/rfc4880#section-9.2|RFC 4880 9.2} for algorithms. + * @param {module:enums.symmetric} sessionKeyAlgorithm - Symmetric key algorithm to use + * @param {Uint8Array} key - The key of cipher blocksize length to be used + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} if encryption was not successful + * @async + */ + async encrypt(sessionKeyAlgorithm, key, config$1 = config) { + const data = this.packets.write(); + const { blockSize } = mod.getCipher(sessionKeyAlgorithm); + + const prefix = await mod.getPrefixRandom(sessionKeyAlgorithm); + const FRE = await mod.mode.cfb.encrypt(sessionKeyAlgorithm, key, prefix, new Uint8Array(blockSize), config$1); + const ciphertext = await mod.mode.cfb.encrypt(sessionKeyAlgorithm, key, data, FRE.subarray(2), config$1); + this.encrypted = util.concat([FRE, ciphertext]); + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Implementation of the strange "Marker packet" (Tag 10) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.8|RFC4880 5.8}: + * An experimental version of PGP used this packet as the Literal + * packet, but no released version of PGP generated Literal packets with this + * tag. With PGP 5.x, this packet has been reassigned and is reserved for use as + * the Marker packet. + * + * The body of this packet consists of: + * The three octets 0x50, 0x47, 0x50 (which spell "PGP" in UTF-8). + * + * Such a packet MUST be ignored when received. It may be placed at the + * beginning of a message that uses features not available in PGP + * version 2.6 in order to cause that version to report that newer + * software is necessary to process the message. + */ + class MarkerPacket { + static get tag() { + return enums.packet.marker; + } + + /** + * Parsing function for a marker data packet (tag 10). + * @param {Uint8Array} bytes - Payload of a tag 10 packet + * @returns {Boolean} whether the packet payload contains "PGP" + */ + read(bytes) { + if (bytes[0] === 0x50 && // P + bytes[1] === 0x47 && // G + bytes[2] === 0x50) { // P + return true; + } + return false; + } + + write() { + return new Uint8Array([0x50, 0x47, 0x50]); + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * A Public-Subkey packet (tag 14) has exactly the same format as a + * Public-Key packet, but denotes a subkey. One or more subkeys may be + * associated with a top-level key. By convention, the top-level key + * provides signature services, and the subkeys provide encryption + * services. + * @extends PublicKeyPacket + */ + class PublicSubkeyPacket extends PublicKeyPacket { + static get tag() { + return enums.packet.publicSubkey; + } + + /** + * @param {Date} [date] - Creation date + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + // eslint-disable-next-line no-useless-constructor + constructor(date, config) { + super(date, config); + } + + /** + * Create a PublicSubkeyPacket from a SecretSubkeyPacket + * @param {SecretSubkeyPacket} secretSubkeyPacket - subkey packet to convert + * @returns {SecretSubkeyPacket} public key packet + * @static + */ + static fromSecretSubkeyPacket(secretSubkeyPacket) { + const keyPacket = new PublicSubkeyPacket(); + const { version, created, algorithm, publicParams, keyID, fingerprint } = secretSubkeyPacket; + keyPacket.version = version; + keyPacket.created = created; + keyPacket.algorithm = algorithm; + keyPacket.publicParams = publicParams; + keyPacket.keyID = keyID; + keyPacket.fingerprint = fingerprint; + return keyPacket; + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Implementation of the User Attribute Packet (Tag 17) + * + * The User Attribute packet is a variation of the User ID packet. It + * is capable of storing more types of data than the User ID packet, + * which is limited to text. Like the User ID packet, a User Attribute + * packet may be certified by the key owner ("self-signed") or any other + * key owner who cares to certify it. Except as noted, a User Attribute + * packet may be used anywhere that a User ID packet may be used. + * + * While User Attribute packets are not a required part of the OpenPGP + * standard, implementations SHOULD provide at least enough + * compatibility to properly handle a certification signature on the + * User Attribute packet. A simple way to do this is by treating the + * User Attribute packet as a User ID packet with opaque contents, but + * an implementation may use any method desired. + */ + class UserAttributePacket { + static get tag() { + return enums.packet.userAttribute; + } + + constructor() { + this.attributes = []; + } + + /** + * parsing function for a user attribute packet (tag 17). + * @param {Uint8Array} input - Payload of a tag 17 packet + */ + read(bytes) { + let i = 0; + while (i < bytes.length) { + const len = readSimpleLength(bytes.subarray(i, bytes.length)); + i += len.offset; + + this.attributes.push(util.uint8ArrayToString(bytes.subarray(i, i + len.len))); + i += len.len; + } + } + + /** + * Creates a binary representation of the user attribute packet + * @returns {Uint8Array} String representation. + */ + write() { + const arr = []; + for (let i = 0; i < this.attributes.length; i++) { + arr.push(writeSimpleLength(this.attributes[i].length)); + arr.push(util.stringToUint8Array(this.attributes[i])); + } + return util.concatUint8Array(arr); + } + + /** + * Compare for equality + * @param {UserAttributePacket} usrAttr + * @returns {Boolean} True if equal. + */ + equals(usrAttr) { + if (!usrAttr || !(usrAttr instanceof UserAttributePacket)) { + return false; + } + return this.attributes.every(function(attr, index) { + return attr === usrAttr.attributes[index]; + }); + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * A Secret-Key packet contains all the information that is found in a + * Public-Key packet, including the public-key material, but also + * includes the secret-key material after all the public-key fields. + * @extends PublicKeyPacket + */ + class SecretKeyPacket extends PublicKeyPacket { + static get tag() { + return enums.packet.secretKey; + } + + /** + * @param {Date} [date] - Creation date + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + constructor(date = new Date(), config$1 = config) { + super(date, config$1); + /** + * Secret-key data + */ + this.keyMaterial = null; + /** + * Indicates whether secret-key data is encrypted. `this.isEncrypted === false` means data is available in decrypted form. + */ + this.isEncrypted = null; + /** + * S2K usage + * @type {enums.symmetric} + */ + this.s2kUsage = 0; + /** + * S2K object + * @type {type/s2k} + */ + this.s2k = null; + /** + * Symmetric algorithm to encrypt the key with + * @type {enums.symmetric} + */ + this.symmetric = null; + /** + * AEAD algorithm to encrypt the key with (if AEAD protection is enabled) + * @type {enums.aead} + */ + this.aead = null; + /** + * Decrypted private parameters, referenced by name + * @type {Object} + */ + this.privateParams = null; + } + + // 5.5.3. Secret-Key Packet Formats + + /** + * Internal parser for private keys as specified in + * {@link https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-04#section-5.5.3|RFC4880bis-04 section 5.5.3} + * @param {Uint8Array} bytes - Input string to read the packet from + * @async + */ + async read(bytes) { + // - A Public-Key or Public-Subkey packet, as described above. + let i = await this.readPublicKey(bytes); + + // - One octet indicating string-to-key usage conventions. Zero + // indicates that the secret-key data is not encrypted. 255 or 254 + // indicates that a string-to-key specifier is being given. Any + // other value is a symmetric-key encryption algorithm identifier. + this.s2kUsage = bytes[i++]; + + // - Only for a version 5 packet, a one-octet scalar octet count of + // the next 4 optional fields. + if (this.version === 5) { + i++; + } + + // - [Optional] If string-to-key usage octet was 255, 254, or 253, a + // one-octet symmetric encryption algorithm. + if (this.s2kUsage === 255 || this.s2kUsage === 254 || this.s2kUsage === 253) { + this.symmetric = bytes[i++]; + + // - [Optional] If string-to-key usage octet was 253, a one-octet + // AEAD algorithm. + if (this.s2kUsage === 253) { + this.aead = bytes[i++]; + } + + // - [Optional] If string-to-key usage octet was 255, 254, or 253, a + // string-to-key specifier. The length of the string-to-key + // specifier is implied by its type, as described above. + this.s2k = new S2K(); + i += this.s2k.read(bytes.subarray(i, bytes.length)); + + if (this.s2k.type === 'gnu-dummy') { + return; + } + } else if (this.s2kUsage) { + this.symmetric = this.s2kUsage; + } + + // - [Optional] If secret data is encrypted (string-to-key usage octet + // not zero), an Initial Vector (IV) of the same length as the + // cipher's block size. + if (this.s2kUsage) { + this.iv = bytes.subarray( + i, + i + mod.getCipher(this.symmetric).blockSize + ); + + i += this.iv.length; + } + + // - Only for a version 5 packet, a four-octet scalar octet count for + // the following key material. + if (this.version === 5) { + i += 4; + } + + // - Plain or encrypted multiprecision integers comprising the secret + // key data. These algorithm-specific fields are as described + // below. + this.keyMaterial = bytes.subarray(i); + this.isEncrypted = !!this.s2kUsage; + + if (!this.isEncrypted) { + const cleartext = this.keyMaterial.subarray(0, -2); + if (!util.equalsUint8Array(util.writeChecksum(cleartext), this.keyMaterial.subarray(-2))) { + throw new Error('Key checksum mismatch'); + } + try { + const { privateParams } = mod.parsePrivateKeyParams(this.algorithm, cleartext, this.publicParams); + this.privateParams = privateParams; + } catch (err) { + if (err instanceof UnsupportedError) throw err; + // avoid throwing potentially sensitive errors + throw new Error('Error reading MPIs'); + } + } + } + + /** + * Creates an OpenPGP key packet for the given key. + * @returns {Uint8Array} A string of bytes containing the secret key OpenPGP packet. + */ + write() { + const arr = [this.writePublicKey()]; + + arr.push(new Uint8Array([this.s2kUsage])); + + const optionalFieldsArr = []; + // - [Optional] If string-to-key usage octet was 255, 254, or 253, a + // one- octet symmetric encryption algorithm. + if (this.s2kUsage === 255 || this.s2kUsage === 254 || this.s2kUsage === 253) { + optionalFieldsArr.push(this.symmetric); + + // - [Optional] If string-to-key usage octet was 253, a one-octet + // AEAD algorithm. + if (this.s2kUsage === 253) { + optionalFieldsArr.push(this.aead); + } + + // - [Optional] If string-to-key usage octet was 255, 254, or 253, a + // string-to-key specifier. The length of the string-to-key + // specifier is implied by its type, as described above. + optionalFieldsArr.push(...this.s2k.write()); + } + + // - [Optional] If secret data is encrypted (string-to-key usage octet + // not zero), an Initial Vector (IV) of the same length as the + // cipher's block size. + if (this.s2kUsage && this.s2k.type !== 'gnu-dummy') { + optionalFieldsArr.push(...this.iv); + } + + if (this.version === 5) { + arr.push(new Uint8Array([optionalFieldsArr.length])); + } + arr.push(new Uint8Array(optionalFieldsArr)); + + if (!this.isDummy()) { + if (!this.s2kUsage) { + this.keyMaterial = mod.serializeParams(this.algorithm, this.privateParams); + } + + if (this.version === 5) { + arr.push(util.writeNumber(this.keyMaterial.length, 4)); + } + arr.push(this.keyMaterial); + + if (!this.s2kUsage) { + arr.push(util.writeChecksum(this.keyMaterial)); + } + } + + return util.concatUint8Array(arr); + } + + /** + * Check whether secret-key data is available in decrypted form. + * Returns false for gnu-dummy keys and null for public keys. + * @returns {Boolean|null} + */ + isDecrypted() { + return this.isEncrypted === false; + } + + /** + * Check whether this is a gnu-dummy key + * @returns {Boolean} + */ + isDummy() { + return !!(this.s2k && this.s2k.type === 'gnu-dummy'); + } + + /** + * Remove private key material, converting the key to a dummy one. + * The resulting key cannot be used for signing/decrypting but can still verify signatures. + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + makeDummy(config$1 = config) { + if (this.isDummy()) { + return; + } + if (this.isDecrypted()) { + this.clearPrivateParams(); + } + this.isEncrypted = null; + this.keyMaterial = null; + this.s2k = new S2K(config$1); + this.s2k.algorithm = 0; + this.s2k.c = 0; + this.s2k.type = 'gnu-dummy'; + this.s2kUsage = 254; + this.symmetric = enums.symmetric.aes256; + } + + /** + * Encrypt the payload. By default, we use aes256 and iterated, salted string + * to key specifier. If the key is in a decrypted state (isEncrypted === false) + * and the passphrase is empty or undefined, the key will be set as not encrypted. + * This can be used to remove passphrase protection after calling decrypt(). + * @param {String} passphrase + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} if encryption was not successful + * @async + */ + async encrypt(passphrase, config$1 = config) { + if (this.isDummy()) { + return; + } + + if (!this.isDecrypted()) { + throw new Error('Key packet is already encrypted'); + } + + if (!passphrase) { + throw new Error('A non-empty passphrase is required for key encryption.'); + } + + this.s2k = new S2K(config$1); + this.s2k.salt = mod.random.getRandomBytes(8); + const cleartext = mod.serializeParams(this.algorithm, this.privateParams); + this.symmetric = enums.symmetric.aes256; + const key = await produceEncryptionKey(this.s2k, passphrase, this.symmetric); + + const { blockSize } = mod.getCipher(this.symmetric); + this.iv = mod.random.getRandomBytes(blockSize); + + if (config$1.aeadProtect) { + this.s2kUsage = 253; + this.aead = enums.aead.eax; + const mode = mod.getAEADMode(this.aead); + const modeInstance = await mode(this.symmetric, key); + this.keyMaterial = await modeInstance.encrypt(cleartext, this.iv.subarray(0, mode.ivLength), new Uint8Array()); + } else { + this.s2kUsage = 254; + this.keyMaterial = await mod.mode.cfb.encrypt(this.symmetric, key, util.concatUint8Array([ + cleartext, + await mod.hash.sha1(cleartext, config$1) + ]), this.iv, config$1); + } + } + + /** + * Decrypts the private key params which are needed to use the key. + * Successful decryption does not imply key integrity, call validate() to confirm that. + * {@link SecretKeyPacket.isDecrypted} should be false, as + * otherwise calls to this function will throw an error. + * @param {String} passphrase - The passphrase for this private key as string + * @throws {Error} if the key is already decrypted, or if decryption was not successful + * @async + */ + async decrypt(passphrase) { + if (this.isDummy()) { + return false; + } + + if (this.isDecrypted()) { + throw new Error('Key packet is already decrypted.'); + } + + let key; + if (this.s2kUsage === 254 || this.s2kUsage === 253) { + key = await produceEncryptionKey(this.s2k, passphrase, this.symmetric); + } else if (this.s2kUsage === 255) { + throw new Error('Encrypted private key is authenticated using an insecure two-byte hash'); + } else { + throw new Error('Private key is encrypted using an insecure S2K function: unsalted MD5'); + } + + let cleartext; + if (this.s2kUsage === 253) { + const mode = mod.getAEADMode(this.aead); + const modeInstance = await mode(this.symmetric, key); + try { + cleartext = await modeInstance.decrypt(this.keyMaterial, this.iv.subarray(0, mode.ivLength), new Uint8Array()); + } catch (err) { + if (err.message === 'Authentication tag mismatch') { + throw new Error('Incorrect key passphrase: ' + err.message); + } + throw err; + } + } else { + const cleartextWithHash = await mod.mode.cfb.decrypt(this.symmetric, key, this.keyMaterial, this.iv); + + cleartext = cleartextWithHash.subarray(0, -20); + const hash = await mod.hash.sha1(cleartext); + + if (!util.equalsUint8Array(hash, cleartextWithHash.subarray(-20))) { + throw new Error('Incorrect key passphrase'); + } + } + + try { + const { privateParams } = mod.parsePrivateKeyParams(this.algorithm, cleartext, this.publicParams); + this.privateParams = privateParams; + } catch (err) { + throw new Error('Error reading MPIs'); + } + this.isEncrypted = false; + this.keyMaterial = null; + this.s2kUsage = 0; + } + + /** + * Checks that the key parameters are consistent + * @throws {Error} if validation was not successful + * @async + */ + async validate() { + if (this.isDummy()) { + return; + } + + if (!this.isDecrypted()) { + throw new Error('Key is not decrypted'); + } + + let validParams; + try { + // this can throw if some parameters are undefined + validParams = await mod.validateParams(this.algorithm, this.publicParams, this.privateParams); + } catch (_) { + validParams = false; + } + if (!validParams) { + throw new Error('Key is invalid'); + } + } + + async generate(bits, curve) { + const { privateParams, publicParams } = await mod.generateParams(this.algorithm, bits, curve); + this.privateParams = privateParams; + this.publicParams = publicParams; + this.isEncrypted = false; + } + + /** + * Clear private key parameters + */ + clearPrivateParams() { + if (this.isDummy()) { + return; + } + + Object.keys(this.privateParams).forEach(name => { + const param = this.privateParams[name]; + param.fill(0); + delete this.privateParams[name]; + }); + this.privateParams = null; + this.isEncrypted = true; + } + } + + async function produceEncryptionKey(s2k, passphrase, algorithm) { + const { keySize } = mod.getCipher(algorithm); + return s2k.produceKey(passphrase, keySize); + } + + var emailAddresses = createCommonjsModule(function (module) { + // email-addresses.js - RFC 5322 email address parser + // v 3.1.0 + // + // http://tools.ietf.org/html/rfc5322 + // + // This library does not validate email addresses. + // emailAddresses attempts to parse addresses using the (fairly liberal) + // grammar specified in RFC 5322. + // + // email-addresses returns { + // ast: , + // addresses: [{ + // node: , + // name: , + // address: , + // local: , + // domain: + // }, ...] + // } + // + // emailAddresses.parseOneAddress and emailAddresses.parseAddressList + // work as you might expect. Try it out. + // + // Many thanks to Dominic Sayers and his documentation on the is_email function, + // http://code.google.com/p/isemail/ , which helped greatly in writing this parser. + + (function (global) { + + function parse5322(opts) { + + // tokenizing functions + + function inStr() { return pos < len; } + function curTok() { return parseString[pos]; } + function getPos() { return pos; } + function setPos(i) { pos = i; } + function nextTok() { pos += 1; } + function initialize() { + pos = 0; + len = parseString.length; + } + + // parser helper functions + + function o(name, value) { + return { + name: name, + tokens: value || "", + semantic: value || "", + children: [] + }; + } + + function wrap(name, ast) { + var n; + if (ast === null) { return null; } + n = o(name); + n.tokens = ast.tokens; + n.semantic = ast.semantic; + n.children.push(ast); + return n; + } + + function add(parent, child) { + if (child !== null) { + parent.tokens += child.tokens; + parent.semantic += child.semantic; + } + parent.children.push(child); + return parent; + } + + function compareToken(fxnCompare) { + var tok; + if (!inStr()) { return null; } + tok = curTok(); + if (fxnCompare(tok)) { + nextTok(); + return o('token', tok); + } + return null; + } + + function literal(lit) { + return function literalFunc() { + return wrap('literal', compareToken(function (tok) { + return tok === lit; + })); + }; + } + + function and() { + var args = arguments; + return function andFunc() { + var i, s, result, start; + start = getPos(); + s = o('and'); + for (i = 0; i < args.length; i += 1) { + result = args[i](); + if (result === null) { + setPos(start); + return null; + } + add(s, result); + } + return s; + }; + } + + function or() { + var args = arguments; + return function orFunc() { + var i, result, start; + start = getPos(); + for (i = 0; i < args.length; i += 1) { + result = args[i](); + if (result !== null) { + return result; + } + setPos(start); + } + return null; + }; + } + + function opt(prod) { + return function optFunc() { + var result, start; + start = getPos(); + result = prod(); + if (result !== null) { + return result; + } + else { + setPos(start); + return o('opt'); + } + }; + } + + function invis(prod) { + return function invisFunc() { + var result = prod(); + if (result !== null) { + result.semantic = ""; + } + return result; + }; + } + + function colwsp(prod) { + return function collapseSemanticWhitespace() { + var result = prod(); + if (result !== null && result.semantic.length > 0) { + result.semantic = " "; + } + return result; + }; + } + + function star(prod, minimum) { + return function starFunc() { + var s, result, count, start, min; + start = getPos(); + s = o('star'); + count = 0; + min = minimum === undefined ? 0 : minimum; + while ((result = prod()) !== null) { + count = count + 1; + add(s, result); + } + if (count >= min) { + return s; + } + else { + setPos(start); + return null; + } + }; + } + + // One expects names to get normalized like this: + // " First Last " -> "First Last" + // "First Last" -> "First Last" + // "First Last" -> "First Last" + function collapseWhitespace(s) { + return s.replace(/([ \t]|\r\n)+/g, ' ').replace(/^\s*/, '').replace(/\s*$/, ''); + } + + // UTF-8 pseudo-production (RFC 6532) + // RFC 6532 extends RFC 5322 productions to include UTF-8 + // using the following productions: + // UTF8-non-ascii = UTF8-2 / UTF8-3 / UTF8-4 + // UTF8-2 = + // UTF8-3 = + // UTF8-4 = + // + // For reference, the extended RFC 5322 productions are: + // VCHAR =/ UTF8-non-ascii + // ctext =/ UTF8-non-ascii + // atext =/ UTF8-non-ascii + // qtext =/ UTF8-non-ascii + // dtext =/ UTF8-non-ascii + function isUTF8NonAscii(tok) { + // In JavaScript, we just deal directly with Unicode code points, + // so we aren't checking individual bytes for UTF-8 encoding. + // Just check that the character is non-ascii. + return tok.charCodeAt(0) >= 128; + } + + + // common productions (RFC 5234) + // http://tools.ietf.org/html/rfc5234 + // B.1. Core Rules + + // CR = %x0D + // ; carriage return + function cr() { return wrap('cr', literal('\r')()); } + + // CRLF = CR LF + // ; Internet standard newline + function crlf() { return wrap('crlf', and(cr, lf)()); } + + // DQUOTE = %x22 + // ; " (Double Quote) + function dquote() { return wrap('dquote', literal('"')()); } + + // HTAB = %x09 + // ; horizontal tab + function htab() { return wrap('htab', literal('\t')()); } + + // LF = %x0A + // ; linefeed + function lf() { return wrap('lf', literal('\n')()); } + + // SP = %x20 + function sp() { return wrap('sp', literal(' ')()); } + + // VCHAR = %x21-7E + // ; visible (printing) characters + function vchar() { + return wrap('vchar', compareToken(function vcharFunc(tok) { + var code = tok.charCodeAt(0); + var accept = (0x21 <= code && code <= 0x7E); + if (opts.rfc6532) { + accept = accept || isUTF8NonAscii(tok); + } + return accept; + })); + } + + // WSP = SP / HTAB + // ; white space + function wsp() { return wrap('wsp', or(sp, htab)()); } + + + // email productions (RFC 5322) + // http://tools.ietf.org/html/rfc5322 + // 3.2.1. Quoted characters + + // quoted-pair = ("\" (VCHAR / WSP)) / obs-qp + function quotedPair() { + var qp = wrap('quoted-pair', + or( + and(literal('\\'), or(vchar, wsp)), + obsQP + )()); + if (qp === null) { return null; } + // a quoted pair will be two characters, and the "\" character + // should be semantically "invisible" (RFC 5322 3.2.1) + qp.semantic = qp.semantic[1]; + return qp; + } + + // 3.2.2. Folding White Space and Comments + + // FWS = ([*WSP CRLF] 1*WSP) / obs-FWS + function fws() { + return wrap('fws', or( + obsFws, + and( + opt(and( + star(wsp), + invis(crlf) + )), + star(wsp, 1) + ) + )()); + } + + // ctext = %d33-39 / ; Printable US-ASCII + // %d42-91 / ; characters not including + // %d93-126 / ; "(", ")", or "\" + // obs-ctext + function ctext() { + return wrap('ctext', or( + function ctextFunc1() { + return compareToken(function ctextFunc2(tok) { + var code = tok.charCodeAt(0); + var accept = + (33 <= code && code <= 39) || + (42 <= code && code <= 91) || + (93 <= code && code <= 126); + if (opts.rfc6532) { + accept = accept || isUTF8NonAscii(tok); + } + return accept; + }); + }, + obsCtext + )()); + } + + // ccontent = ctext / quoted-pair / comment + function ccontent() { + return wrap('ccontent', or(ctext, quotedPair, comment)()); + } + + // comment = "(" *([FWS] ccontent) [FWS] ")" + function comment() { + return wrap('comment', and( + literal('('), + star(and(opt(fws), ccontent)), + opt(fws), + literal(')') + )()); + } + + // CFWS = (1*([FWS] comment) [FWS]) / FWS + function cfws() { + return wrap('cfws', or( + and( + star( + and(opt(fws), comment), + 1 + ), + opt(fws) + ), + fws + )()); + } + + // 3.2.3. Atom + + //atext = ALPHA / DIGIT / ; Printable US-ASCII + // "!" / "#" / ; characters not including + // "$" / "%" / ; specials. Used for atoms. + // "&" / "'" / + // "*" / "+" / + // "-" / "/" / + // "=" / "?" / + // "^" / "_" / + // "`" / "{" / + // "|" / "}" / + // "~" + function atext() { + return wrap('atext', compareToken(function atextFunc(tok) { + var accept = + ('a' <= tok && tok <= 'z') || + ('A' <= tok && tok <= 'Z') || + ('0' <= tok && tok <= '9') || + (['!', '#', '$', '%', '&', '\'', '*', '+', '-', '/', + '=', '?', '^', '_', '`', '{', '|', '}', '~'].indexOf(tok) >= 0); + if (opts.rfc6532) { + accept = accept || isUTF8NonAscii(tok); + } + return accept; + })); + } + + // atom = [CFWS] 1*atext [CFWS] + function atom() { + return wrap('atom', and(colwsp(opt(cfws)), star(atext, 1), colwsp(opt(cfws)))()); + } + + // dot-atom-text = 1*atext *("." 1*atext) + function dotAtomText() { + var s, maybeText; + s = wrap('dot-atom-text', star(atext, 1)()); + if (s === null) { return s; } + maybeText = star(and(literal('.'), star(atext, 1)))(); + if (maybeText !== null) { + add(s, maybeText); + } + return s; + } + + // dot-atom = [CFWS] dot-atom-text [CFWS] + function dotAtom() { + return wrap('dot-atom', and(invis(opt(cfws)), dotAtomText, invis(opt(cfws)))()); + } + + // 3.2.4. Quoted Strings + + // qtext = %d33 / ; Printable US-ASCII + // %d35-91 / ; characters not including + // %d93-126 / ; "\" or the quote character + // obs-qtext + function qtext() { + return wrap('qtext', or( + function qtextFunc1() { + return compareToken(function qtextFunc2(tok) { + var code = tok.charCodeAt(0); + var accept = + (33 === code) || + (35 <= code && code <= 91) || + (93 <= code && code <= 126); + if (opts.rfc6532) { + accept = accept || isUTF8NonAscii(tok); + } + return accept; + }); + }, + obsQtext + )()); + } + + // qcontent = qtext / quoted-pair + function qcontent() { + return wrap('qcontent', or(qtext, quotedPair)()); + } + + // quoted-string = [CFWS] + // DQUOTE *([FWS] qcontent) [FWS] DQUOTE + // [CFWS] + function quotedString() { + return wrap('quoted-string', and( + invis(opt(cfws)), + invis(dquote), star(and(opt(colwsp(fws)), qcontent)), opt(invis(fws)), invis(dquote), + invis(opt(cfws)) + )()); + } + + // 3.2.5 Miscellaneous Tokens + + // word = atom / quoted-string + function word() { + return wrap('word', or(atom, quotedString)()); + } + + // phrase = 1*word / obs-phrase + function phrase() { + return wrap('phrase', or(obsPhrase, star(word, 1))()); + } + + // 3.4. Address Specification + // address = mailbox / group + function address() { + return wrap('address', or(mailbox, group)()); + } + + // mailbox = name-addr / addr-spec + function mailbox() { + return wrap('mailbox', or(nameAddr, addrSpec)()); + } + + // name-addr = [display-name] angle-addr + function nameAddr() { + return wrap('name-addr', and(opt(displayName), angleAddr)()); + } + + // angle-addr = [CFWS] "<" addr-spec ">" [CFWS] / + // obs-angle-addr + function angleAddr() { + return wrap('angle-addr', or( + and( + invis(opt(cfws)), + literal('<'), + addrSpec, + literal('>'), + invis(opt(cfws)) + ), + obsAngleAddr + )()); + } + + // group = display-name ":" [group-list] ";" [CFWS] + function group() { + return wrap('group', and( + displayName, + literal(':'), + opt(groupList), + literal(';'), + invis(opt(cfws)) + )()); + } + + // display-name = phrase + function displayName() { + return wrap('display-name', function phraseFixedSemantic() { + var result = phrase(); + if (result !== null) { + result.semantic = collapseWhitespace(result.semantic); + } + return result; + }()); + } + + // mailbox-list = (mailbox *("," mailbox)) / obs-mbox-list + function mailboxList() { + return wrap('mailbox-list', or( + and( + mailbox, + star(and(literal(','), mailbox)) + ), + obsMboxList + )()); + } + + // address-list = (address *("," address)) / obs-addr-list + function addressList() { + return wrap('address-list', or( + and( + address, + star(and(literal(','), address)) + ), + obsAddrList + )()); + } + + // group-list = mailbox-list / CFWS / obs-group-list + function groupList() { + return wrap('group-list', or( + mailboxList, + invis(cfws), + obsGroupList + )()); + } + + // 3.4.1 Addr-Spec Specification + + // local-part = dot-atom / quoted-string / obs-local-part + function localPart() { + // note: quoted-string, dotAtom are proper subsets of obs-local-part + // so we really just have to look for obsLocalPart, if we don't care about the exact parse tree + return wrap('local-part', or(obsLocalPart, dotAtom, quotedString)()); + } + + // dtext = %d33-90 / ; Printable US-ASCII + // %d94-126 / ; characters not including + // obs-dtext ; "[", "]", or "\" + function dtext() { + return wrap('dtext', or( + function dtextFunc1() { + return compareToken(function dtextFunc2(tok) { + var code = tok.charCodeAt(0); + var accept = + (33 <= code && code <= 90) || + (94 <= code && code <= 126); + if (opts.rfc6532) { + accept = accept || isUTF8NonAscii(tok); + } + return accept; + }); + }, + obsDtext + )() + ); + } + + // domain-literal = [CFWS] "[" *([FWS] dtext) [FWS] "]" [CFWS] + function domainLiteral() { + return wrap('domain-literal', and( + invis(opt(cfws)), + literal('['), + star(and(opt(fws), dtext)), + opt(fws), + literal(']'), + invis(opt(cfws)) + )()); + } + + // domain = dot-atom / domain-literal / obs-domain + function domain() { + return wrap('domain', function domainCheckTLD() { + var result = or(obsDomain, dotAtom, domainLiteral)(); + if (opts.rejectTLD) { + if (result && result.semantic && result.semantic.indexOf('.') < 0) { + return null; + } + } + // strip all whitespace from domains + if (result) { + result.semantic = result.semantic.replace(/\s+/g, ''); + } + return result; + }()); + } + + // addr-spec = local-part "@" domain + function addrSpec() { + return wrap('addr-spec', and( + localPart, literal('@'), domain + )()); + } + + // 3.6.2 Originator Fields + // Below we only parse the field body, not the name of the field + // like "From:", "Sender:", or "Reply-To:". Other libraries that + // parse email headers can parse those and defer to these productions + // for the "RFC 5322" part. + + // RFC 6854 2.1. Replacement of RFC 5322, Section 3.6.2. Originator Fields + // from = "From:" (mailbox-list / address-list) CRLF + function fromSpec() { + return wrap('from', or( + mailboxList, + addressList + )()); + } + + // RFC 6854 2.1. Replacement of RFC 5322, Section 3.6.2. Originator Fields + // sender = "Sender:" (mailbox / address) CRLF + function senderSpec() { + return wrap('sender', or( + mailbox, + address + )()); + } + + // RFC 6854 2.1. Replacement of RFC 5322, Section 3.6.2. Originator Fields + // reply-to = "Reply-To:" address-list CRLF + function replyToSpec() { + return wrap('reply-to', addressList()); + } + + // 4.1. Miscellaneous Obsolete Tokens + + // obs-NO-WS-CTL = %d1-8 / ; US-ASCII control + // %d11 / ; characters that do not + // %d12 / ; include the carriage + // %d14-31 / ; return, line feed, and + // %d127 ; white space characters + function obsNoWsCtl() { + return opts.strict ? null : wrap('obs-NO-WS-CTL', compareToken(function (tok) { + var code = tok.charCodeAt(0); + return ((1 <= code && code <= 8) || + (11 === code || 12 === code) || + (14 <= code && code <= 31) || + (127 === code)); + })); + } + + // obs-ctext = obs-NO-WS-CTL + function obsCtext() { return opts.strict ? null : wrap('obs-ctext', obsNoWsCtl()); } + + // obs-qtext = obs-NO-WS-CTL + function obsQtext() { return opts.strict ? null : wrap('obs-qtext', obsNoWsCtl()); } + + // obs-qp = "\" (%d0 / obs-NO-WS-CTL / LF / CR) + function obsQP() { + return opts.strict ? null : wrap('obs-qp', and( + literal('\\'), + or(literal('\0'), obsNoWsCtl, lf, cr) + )()); + } + + // obs-phrase = word *(word / "." / CFWS) + function obsPhrase() { + if (opts.strict ) return null; + return opts.atInDisplayName ? wrap('obs-phrase', and( + word, + star(or(word, literal('.'), literal('@'), colwsp(cfws))) + )()) : + wrap('obs-phrase', and( + word, + star(or(word, literal('.'), colwsp(cfws))) + )()); + } + + // 4.2. Obsolete Folding White Space + + // NOTE: read the errata http://www.rfc-editor.org/errata_search.php?rfc=5322&eid=1908 + // obs-FWS = 1*([CRLF] WSP) + function obsFws() { + return opts.strict ? null : wrap('obs-FWS', star( + and(invis(opt(crlf)), wsp), + 1 + )()); + } + + // 4.4. Obsolete Addressing + + // obs-angle-addr = [CFWS] "<" obs-route addr-spec ">" [CFWS] + function obsAngleAddr() { + return opts.strict ? null : wrap('obs-angle-addr', and( + invis(opt(cfws)), + literal('<'), + obsRoute, + addrSpec, + literal('>'), + invis(opt(cfws)) + )()); + } + + // obs-route = obs-domain-list ":" + function obsRoute() { + return opts.strict ? null : wrap('obs-route', and( + obsDomainList, + literal(':') + )()); + } + + // obs-domain-list = *(CFWS / ",") "@" domain + // *("," [CFWS] ["@" domain]) + function obsDomainList() { + return opts.strict ? null : wrap('obs-domain-list', and( + star(or(invis(cfws), literal(','))), + literal('@'), + domain, + star(and( + literal(','), + invis(opt(cfws)), + opt(and(literal('@'), domain)) + )) + )()); + } + + // obs-mbox-list = *([CFWS] ",") mailbox *("," [mailbox / CFWS]) + function obsMboxList() { + return opts.strict ? null : wrap('obs-mbox-list', and( + star(and( + invis(opt(cfws)), + literal(',') + )), + mailbox, + star(and( + literal(','), + opt(and( + mailbox, + invis(cfws) + )) + )) + )()); + } + + // obs-addr-list = *([CFWS] ",") address *("," [address / CFWS]) + function obsAddrList() { + return opts.strict ? null : wrap('obs-addr-list', and( + star(and( + invis(opt(cfws)), + literal(',') + )), + address, + star(and( + literal(','), + opt(and( + address, + invis(cfws) + )) + )) + )()); + } + + // obs-group-list = 1*([CFWS] ",") [CFWS] + function obsGroupList() { + return opts.strict ? null : wrap('obs-group-list', and( + star(and( + invis(opt(cfws)), + literal(',') + ), 1), + invis(opt(cfws)) + )()); + } + + // obs-local-part = word *("." word) + function obsLocalPart() { + return opts.strict ? null : wrap('obs-local-part', and(word, star(and(literal('.'), word)))()); + } + + // obs-domain = atom *("." atom) + function obsDomain() { + return opts.strict ? null : wrap('obs-domain', and(atom, star(and(literal('.'), atom)))()); + } + + // obs-dtext = obs-NO-WS-CTL / quoted-pair + function obsDtext() { + return opts.strict ? null : wrap('obs-dtext', or(obsNoWsCtl, quotedPair)()); + } + + ///////////////////////////////////////////////////// + + // ast analysis + + function findNode(name, root) { + var i, stack, node; + if (root === null || root === undefined) { return null; } + stack = [root]; + while (stack.length > 0) { + node = stack.pop(); + if (node.name === name) { + return node; + } + for (i = node.children.length - 1; i >= 0; i -= 1) { + stack.push(node.children[i]); + } + } + return null; + } + + function findAllNodes(name, root) { + var i, stack, node, result; + if (root === null || root === undefined) { return null; } + stack = [root]; + result = []; + while (stack.length > 0) { + node = stack.pop(); + if (node.name === name) { + result.push(node); + } + for (i = node.children.length - 1; i >= 0; i -= 1) { + stack.push(node.children[i]); + } + } + return result; + } + + function findAllNodesNoChildren(names, root) { + var i, stack, node, result, namesLookup; + if (root === null || root === undefined) { return null; } + stack = [root]; + result = []; + namesLookup = {}; + for (i = 0; i < names.length; i += 1) { + namesLookup[names[i]] = true; + } + + while (stack.length > 0) { + node = stack.pop(); + if (node.name in namesLookup) { + result.push(node); + // don't look at children (hence findAllNodesNoChildren) + } else { + for (i = node.children.length - 1; i >= 0; i -= 1) { + stack.push(node.children[i]); + } + } + } + return result; + } + + function giveResult(ast) { + var addresses, groupsAndMailboxes, i, groupOrMailbox, result; + if (ast === null) { + return null; + } + addresses = []; + + // An address is a 'group' (i.e. a list of mailboxes) or a 'mailbox'. + groupsAndMailboxes = findAllNodesNoChildren(['group', 'mailbox'], ast); + for (i = 0; i < groupsAndMailboxes.length; i += 1) { + groupOrMailbox = groupsAndMailboxes[i]; + if (groupOrMailbox.name === 'group') { + addresses.push(giveResultGroup(groupOrMailbox)); + } else if (groupOrMailbox.name === 'mailbox') { + addresses.push(giveResultMailbox(groupOrMailbox)); + } + } + + result = { + ast: ast, + addresses: addresses, + }; + if (opts.simple) { + result = simplifyResult(result); + } + if (opts.oneResult) { + return oneResult(result); + } + if (opts.simple) { + return result && result.addresses; + } else { + return result; + } + } + + function giveResultGroup(group) { + var i; + var groupName = findNode('display-name', group); + var groupResultMailboxes = []; + var mailboxes = findAllNodesNoChildren(['mailbox'], group); + for (i = 0; i < mailboxes.length; i += 1) { + groupResultMailboxes.push(giveResultMailbox(mailboxes[i])); + } + return { + node: group, + parts: { + name: groupName, + }, + type: group.name, // 'group' + name: grabSemantic(groupName), + addresses: groupResultMailboxes, + }; + } + + function giveResultMailbox(mailbox) { + var name = findNode('display-name', mailbox); + var aspec = findNode('addr-spec', mailbox); + var cfws = findAllNodes('cfws', mailbox); + var comments = findAllNodesNoChildren(['comment'], mailbox); + + + var local = findNode('local-part', aspec); + var domain = findNode('domain', aspec); + return { + node: mailbox, + parts: { + name: name, + address: aspec, + local: local, + domain: domain, + comments: cfws + }, + type: mailbox.name, // 'mailbox' + name: grabSemantic(name), + address: grabSemantic(aspec), + local: grabSemantic(local), + domain: grabSemantic(domain), + comments: concatComments(comments), + groupName: grabSemantic(mailbox.groupName), + }; + } + + function grabSemantic(n) { + return n !== null && n !== undefined ? n.semantic : null; + } + + function simplifyResult(result) { + var i; + if (result && result.addresses) { + for (i = 0; i < result.addresses.length; i += 1) { + delete result.addresses[i].node; + } + } + return result; + } + + function concatComments(comments) { + var result = ''; + if (comments) { + for (var i = 0; i < comments.length; i += 1) { + result += grabSemantic(comments[i]); + } + } + return result; + } + + function oneResult(result) { + if (!result) { return null; } + if (!opts.partial && result.addresses.length > 1) { return null; } + return result.addresses && result.addresses[0]; + } + + ///////////////////////////////////////////////////// + + var parseString, pos, len, parsed, startProduction; + + opts = handleOpts(opts, {}); + if (opts === null) { return null; } + + parseString = opts.input; + + startProduction = { + 'address': address, + 'address-list': addressList, + 'angle-addr': angleAddr, + 'from': fromSpec, + 'group': group, + 'mailbox': mailbox, + 'mailbox-list': mailboxList, + 'reply-to': replyToSpec, + 'sender': senderSpec, + }[opts.startAt] || addressList; + + if (!opts.strict) { + initialize(); + opts.strict = true; + parsed = startProduction(parseString); + if (opts.partial || !inStr()) { + return giveResult(parsed); + } + opts.strict = false; + } + + initialize(); + parsed = startProduction(parseString); + if (!opts.partial && inStr()) { return null; } + return giveResult(parsed); + } + + function parseOneAddressSimple(opts) { + return parse5322(handleOpts(opts, { + oneResult: true, + rfc6532: true, + simple: true, + startAt: 'address-list', + })); + } + + function parseAddressListSimple(opts) { + return parse5322(handleOpts(opts, { + rfc6532: true, + simple: true, + startAt: 'address-list', + })); + } + + function parseFromSimple(opts) { + return parse5322(handleOpts(opts, { + rfc6532: true, + simple: true, + startAt: 'from', + })); + } + + function parseSenderSimple(opts) { + return parse5322(handleOpts(opts, { + oneResult: true, + rfc6532: true, + simple: true, + startAt: 'sender', + })); + } + + function parseReplyToSimple(opts) { + return parse5322(handleOpts(opts, { + rfc6532: true, + simple: true, + startAt: 'reply-to', + })); + } + + function handleOpts(opts, defs) { + function isString(str) { + return Object.prototype.toString.call(str) === '[object String]'; + } + + function isObject(o) { + return o === Object(o); + } + + function isNullUndef(o) { + return o === null || o === undefined; + } + + var defaults, o; + + if (isString(opts)) { + opts = { input: opts }; + } else if (!isObject(opts)) { + return null; + } + + if (!isString(opts.input)) { return null; } + if (!defs) { return null; } + + defaults = { + oneResult: false, + partial: false, + rejectTLD: false, + rfc6532: false, + simple: false, + startAt: 'address-list', + strict: false, + atInDisplayName: false + }; + + for (o in defaults) { + if (isNullUndef(opts[o])) { + opts[o] = !isNullUndef(defs[o]) ? defs[o] : defaults[o]; + } + } + return opts; + } + + parse5322.parseOneAddress = parseOneAddressSimple; + parse5322.parseAddressList = parseAddressListSimple; + parse5322.parseFrom = parseFromSimple; + parse5322.parseSender = parseSenderSimple; + parse5322.parseReplyTo = parseReplyToSimple; + + { + module.exports = parse5322; + } + + }()); + }); + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * Implementation of the User ID Packet (Tag 13) + * + * A User ID packet consists of UTF-8 text that is intended to represent + * the name and email address of the key holder. By convention, it + * includes an RFC 2822 [RFC2822] mail name-addr, but there are no + * restrictions on its content. The packet length in the header + * specifies the length of the User ID. + */ + class UserIDPacket { + static get tag() { + return enums.packet.userID; + } + + constructor() { + /** A string containing the user id. Usually in the form + * John Doe + * @type {String} + */ + this.userID = ''; + + this.name = ''; + this.email = ''; + this.comment = ''; + } + + /** + * Create UserIDPacket instance from object + * @param {Object} userID - Object specifying userID name, email and comment + * @returns {UserIDPacket} + * @static + */ + static fromObject(userID) { + if (util.isString(userID) || + (userID.name && !util.isString(userID.name)) || + (userID.email && !util.isEmailAddress(userID.email)) || + (userID.comment && !util.isString(userID.comment))) { + throw new Error('Invalid user ID format'); + } + const packet = new UserIDPacket(); + Object.assign(packet, userID); + const components = []; + if (packet.name) components.push(packet.name); + if (packet.comment) components.push(`(${packet.comment})`); + if (packet.email) components.push(`<${packet.email}>`); + packet.userID = components.join(' '); + return packet; + } + + /** + * Parsing function for a user id packet (tag 13). + * @param {Uint8Array} input - Payload of a tag 13 packet + */ + read(bytes, config$1 = config) { + const userID = util.decodeUTF8(bytes); + if (userID.length > config$1.maxUserIDLength) { + throw new Error('User ID string is too long'); + } + try { + const { name, address: email, comments } = emailAddresses.parseOneAddress({ input: userID, atInDisplayName: true }); + this.comment = comments.replace(/^\(|\)$/g, ''); + this.name = name; + this.email = email; + } catch (e) {} + this.userID = userID; + } + + /** + * Creates a binary representation of the user id packet + * @returns {Uint8Array} Binary representation. + */ + write() { + return util.encodeUTF8(this.userID); + } + + equals(otherUserID) { + return otherUserID && otherUserID.userID === this.userID; + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + /** + * A Secret-Subkey packet (tag 7) is the subkey analog of the Secret + * Key packet and has exactly the same format. + * @extends SecretKeyPacket + */ + class SecretSubkeyPacket extends SecretKeyPacket { + static get tag() { + return enums.packet.secretSubkey; + } + + /** + * @param {Date} [date] - Creation date + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + constructor(date = new Date(), config$1 = config) { + super(date, config$1); + } + } + + /** + * Implementation of the Trust Packet (Tag 12) + * + * {@link https://tools.ietf.org/html/rfc4880#section-5.10|RFC4880 5.10}: + * The Trust packet is used only within keyrings and is not normally + * exported. Trust packets contain data that record the user's + * specifications of which key holders are trustworthy introducers, + * along with other information that implementing software uses for + * trust information. The format of Trust packets is defined by a given + * implementation. + * + * Trust packets SHOULD NOT be emitted to output streams that are + * transferred to other users, and they SHOULD be ignored on any input + * other than local keyring files. + */ + class TrustPacket { + static get tag() { + return enums.packet.trust; + } + + /** + * Parsing function for a trust packet (tag 12). + * Currently not implemented as we ignore trust packets + */ + read() { + throw new UnsupportedError('Trust packets are not supported'); + } + + write() { + throw new UnsupportedError('Trust packets are not supported'); + } + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + // A Signature can contain the following packets + const allowedPackets$4 = /*#__PURE__*/ util.constructAllowedPackets([SignaturePacket]); + + /** + * Class that represents an OpenPGP signature. + */ + class Signature { + /** + * @param {PacketList} packetlist - The signature packets + */ + constructor(packetlist) { + this.packets = packetlist || new PacketList(); + } + + /** + * Returns binary encoded signature + * @returns {ReadableStream} Binary signature. + */ + write() { + return this.packets.write(); + } + + /** + * Returns ASCII armored text of signature + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {ReadableStream} ASCII armor. + */ + armor(config$1 = config) { + return armor(enums.armor.signature, this.write(), undefined, undefined, undefined, config$1); + } + + /** + * Returns an array of KeyIDs of all of the issuers who created this signature + * @returns {Array} The Key IDs of the signing keys + */ + getSigningKeyIDs() { + return this.packets.map(packet => packet.issuerKeyID); + } + } + + /** + * reads an (optionally armored) OpenPGP signature and returns a signature object + * @param {Object} options + * @param {String} [options.armoredSignature] - Armored signature to be parsed + * @param {Uint8Array} [options.binarySignature] - Binary signature to be parsed + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} New signature object. + * @async + * @static + */ + async function readSignature({ armoredSignature, binarySignature, config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; + let input = armoredSignature || binarySignature; + if (!input) { + throw new Error('readSignature: must pass options object containing `armoredSignature` or `binarySignature`'); + } + if (armoredSignature && !util.isString(armoredSignature)) { + throw new Error('readSignature: options.armoredSignature must be a string'); + } + if (binarySignature && !util.isUint8Array(binarySignature)) { + throw new Error('readSignature: options.binarySignature must be a Uint8Array'); + } + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (armoredSignature) { + const { type, data } = await unarmor(input, config$1); + if (type !== enums.armor.signature) { + throw new Error('Armored text not of type signature'); + } + input = data; + } + const packetlist = await PacketList.fromBinary(input, allowedPackets$4, config$1); + return new Signature(packetlist); + } + + /** + * @fileoverview Provides helpers methods for key module + * @module key/helper + * @private + */ + + async function generateSecretSubkey(options, config) { + const secretSubkeyPacket = new SecretSubkeyPacket(options.date, config); + secretSubkeyPacket.packets = null; + secretSubkeyPacket.algorithm = enums.write(enums.publicKey, options.algorithm); + await secretSubkeyPacket.generate(options.rsaBits, options.curve); + await secretSubkeyPacket.computeFingerprintAndKeyID(); + return secretSubkeyPacket; + } + + async function generateSecretKey(options, config) { + const secretKeyPacket = new SecretKeyPacket(options.date, config); + secretKeyPacket.packets = null; + secretKeyPacket.algorithm = enums.write(enums.publicKey, options.algorithm); + await secretKeyPacket.generate(options.rsaBits, options.curve, options.config); + await secretKeyPacket.computeFingerprintAndKeyID(); + return secretKeyPacket; + } + + /** + * Returns the valid and non-expired signature that has the latest creation date, while ignoring signatures created in the future. + * @param {Array} signatures - List of signatures + * @param {PublicKeyPacket|PublicSubkeyPacket} publicKey - Public key packet to verify the signature + * @param {Date} date - Use the given date instead of the current time + * @param {Object} config - full configuration + * @returns {Promise} The latest valid signature. + * @async + */ + async function getLatestValidSignature(signatures, publicKey, signatureType, dataToVerify, date = new Date(), config) { + let latestValid; + let exception; + for (let i = signatures.length - 1; i >= 0; i--) { + try { + if ( + (!latestValid || signatures[i].created >= latestValid.created) + ) { + await signatures[i].verify(publicKey, signatureType, dataToVerify, date, undefined, config); + latestValid = signatures[i]; + } + } catch (e) { + exception = e; + } + } + if (!latestValid) { + throw util.wrapError( + `Could not find valid ${enums.read(enums.signature, signatureType)} signature in key ${publicKey.getKeyID().toHex()}` + .replace('certGeneric ', 'self-') + .replace(/([a-z])([A-Z])/g, (_, $1, $2) => $1 + ' ' + $2.toLowerCase()), + exception); + } + return latestValid; + } + + function isDataExpired(keyPacket, signature, date = new Date()) { + const normDate = util.normalizeDate(date); + if (normDate !== null) { + const expirationTime = getKeyExpirationTime(keyPacket, signature); + return !(keyPacket.created <= normDate && normDate < expirationTime); + } + return false; + } + + /** + * Create Binding signature to the key according to the {@link https://tools.ietf.org/html/rfc4880#section-5.2.1} + * @param {SecretSubkeyPacket} subkey - Subkey key packet + * @param {SecretKeyPacket} primaryKey - Primary key packet + * @param {Object} options + * @param {Object} config - Full configuration + */ + async function createBindingSignature(subkey, primaryKey, options, config) { + const dataToSign = {}; + dataToSign.key = primaryKey; + dataToSign.bind = subkey; + const subkeySignaturePacket = new SignaturePacket(); + subkeySignaturePacket.signatureType = enums.signature.subkeyBinding; + subkeySignaturePacket.publicKeyAlgorithm = primaryKey.algorithm; + subkeySignaturePacket.hashAlgorithm = await getPreferredHashAlgo$1(null, subkey, undefined, undefined, config); + if (options.sign) { + subkeySignaturePacket.keyFlags = [enums.keyFlags.signData]; + subkeySignaturePacket.embeddedSignature = await createSignaturePacket(dataToSign, null, subkey, { + signatureType: enums.signature.keyBinding + }, options.date, undefined, undefined, undefined, config); + } else { + subkeySignaturePacket.keyFlags = [enums.keyFlags.encryptCommunication | enums.keyFlags.encryptStorage]; + } + if (options.keyExpirationTime > 0) { + subkeySignaturePacket.keyExpirationTime = options.keyExpirationTime; + subkeySignaturePacket.keyNeverExpires = false; + } + await subkeySignaturePacket.sign(primaryKey, dataToSign, options.date); + return subkeySignaturePacket; + } + + /** + * Returns the preferred signature hash algorithm of a key + * @param {Key} [key] - The key to get preferences from + * @param {SecretKeyPacket|SecretSubkeyPacket} keyPacket - key packet used for signing + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Object} [userID] - User ID + * @param {Object} config - full configuration + * @returns {Promise} + * @async + */ + async function getPreferredHashAlgo$1(key, keyPacket, date = new Date(), userID = {}, config) { + let hashAlgo = config.preferredHashAlgorithm; + let prefAlgo = hashAlgo; + if (key) { + const primaryUser = await key.getPrimaryUser(date, userID, config); + if (primaryUser.selfCertification.preferredHashAlgorithms) { + [prefAlgo] = primaryUser.selfCertification.preferredHashAlgorithms; + hashAlgo = mod.hash.getHashByteLength(hashAlgo) <= mod.hash.getHashByteLength(prefAlgo) ? + prefAlgo : hashAlgo; + } + } + switch (Object.getPrototypeOf(keyPacket)) { + case SecretKeyPacket.prototype: + case PublicKeyPacket.prototype: + case SecretSubkeyPacket.prototype: + case PublicSubkeyPacket.prototype: + switch (keyPacket.algorithm) { + case enums.publicKey.ecdh: + case enums.publicKey.ecdsa: + case enums.publicKey.eddsa: + prefAlgo = mod.publicKey.elliptic.getPreferredHashAlgo(keyPacket.publicParams.oid); + } + } + return mod.hash.getHashByteLength(hashAlgo) <= mod.hash.getHashByteLength(prefAlgo) ? + prefAlgo : hashAlgo; + } + + /** + * Returns the preferred symmetric/aead/compression algorithm for a set of keys + * @param {'symmetric'|'aead'|'compression'} type - Type of preference to return + * @param {Array} [keys] - Set of keys + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Array} [userIDs] - User IDs + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} Preferred algorithm + * @async + */ + async function getPreferredAlgo(type, keys = [], date = new Date(), userIDs = [], config$1 = config) { + const defaultAlgo = { // these are all must-implement in rfc4880bis + 'symmetric': enums.symmetric.aes128, + 'aead': enums.aead.eax, + 'compression': enums.compression.uncompressed + }[type]; + const preferredSenderAlgo = { + 'symmetric': config$1.preferredSymmetricAlgorithm, + 'aead': config$1.preferredAEADAlgorithm, + 'compression': config$1.preferredCompressionAlgorithm + }[type]; + const prefPropertyName = { + 'symmetric': 'preferredSymmetricAlgorithms', + 'aead': 'preferredAEADAlgorithms', + 'compression': 'preferredCompressionAlgorithms' + }[type]; + + // if preferredSenderAlgo appears in the prefs of all recipients, we pick it + // otherwise we use the default algo + // if no keys are available, preferredSenderAlgo is returned + const senderAlgoSupport = await Promise.all(keys.map(async function(key, i) { + const primaryUser = await key.getPrimaryUser(date, userIDs[i], config$1); + const recipientPrefs = primaryUser.selfCertification[prefPropertyName]; + return !!recipientPrefs && recipientPrefs.indexOf(preferredSenderAlgo) >= 0; + })); + return senderAlgoSupport.every(Boolean) ? preferredSenderAlgo : defaultAlgo; + } + + /** + * Create signature packet + * @param {Object} dataToSign - Contains packets to be signed + * @param {PrivateKey} privateKey - key to get preferences from + * @param {SecretKeyPacket| + * SecretSubkeyPacket} signingKeyPacket secret key packet for signing + * @param {Object} [signatureProperties] - Properties to write on the signature packet before signing + * @param {Date} [date] - Override the creationtime of the signature + * @param {Object} [userID] - User ID + * @param {Array} [notations] - Notation Data to add to the signature, e.g. [{ name: 'test@example.org', value: new TextEncoder().encode('test'), humanReadable: true, critical: false }] + * @param {Object} [detached] - Whether to create a detached signature packet + * @param {Object} config - full configuration + * @returns {Promise} Signature packet. + */ + async function createSignaturePacket(dataToSign, privateKey, signingKeyPacket, signatureProperties, date, userID, notations = [], detached = false, config) { + if (signingKeyPacket.isDummy()) { + throw new Error('Cannot sign with a gnu-dummy key.'); + } + if (!signingKeyPacket.isDecrypted()) { + throw new Error('Signing key is not decrypted.'); + } + const signaturePacket = new SignaturePacket(); + Object.assign(signaturePacket, signatureProperties); + signaturePacket.publicKeyAlgorithm = signingKeyPacket.algorithm; + signaturePacket.hashAlgorithm = await getPreferredHashAlgo$1(privateKey, signingKeyPacket, date, userID, config); + signaturePacket.rawNotations = notations; + await signaturePacket.sign(signingKeyPacket, dataToSign, date, detached); + return signaturePacket; + } + + /** + * Merges signatures from source[attr] to dest[attr] + * @param {Object} source + * @param {Object} dest + * @param {String} attr + * @param {Date} [date] - date to use for signature expiration check, instead of the current time + * @param {Function} [checkFn] - signature only merged if true + */ + async function mergeSignatures(source, dest, attr, date = new Date(), checkFn) { + source = source[attr]; + if (source) { + if (!dest[attr].length) { + dest[attr] = source; + } else { + await Promise.all(source.map(async function(sourceSig) { + if (!sourceSig.isExpired(date) && (!checkFn || await checkFn(sourceSig)) && + !dest[attr].some(function(destSig) { + return util.equalsUint8Array(destSig.writeParams(), sourceSig.writeParams()); + })) { + dest[attr].push(sourceSig); + } + })); + } + } + } + + /** + * Checks if a given certificate or binding signature is revoked + * @param {SecretKeyPacket| + * PublicKeyPacket} primaryKey The primary key packet + * @param {Object} dataToVerify - The data to check + * @param {Array} revocations - The revocation signatures to check + * @param {SignaturePacket} signature - The certificate or signature to check + * @param {PublicSubkeyPacket| + * SecretSubkeyPacket| + * PublicKeyPacket| + * SecretKeyPacket} key, optional The key packet to verify the signature, instead of the primary key + * @param {Date} date - Use the given date instead of the current time + * @param {Object} config - Full configuration + * @returns {Promise} True if the signature revokes the data. + * @async + */ + async function isDataRevoked(primaryKey, signatureType, dataToVerify, revocations, signature, key, date = new Date(), config) { + key = key || primaryKey; + const revocationKeyIDs = []; + await Promise.all(revocations.map(async function(revocationSignature) { + try { + if ( + // Note: a third-party revocation signature could legitimately revoke a + // self-signature if the signature has an authorized revocation key. + // However, we don't support passing authorized revocation keys, nor + // verifying such revocation signatures. Instead, we indicate an error + // when parsing a key with an authorized revocation key, and ignore + // third-party revocation signatures here. (It could also be revoking a + // third-party key certification, which should only affect + // `verifyAllCertifications`.) + !signature || revocationSignature.issuerKeyID.equals(signature.issuerKeyID) + ) { + await revocationSignature.verify( + key, signatureType, dataToVerify, config.revocationsExpire ? date : null, false, config + ); + + // TODO get an identifier of the revoked object instead + revocationKeyIDs.push(revocationSignature.issuerKeyID); + } + } catch (e) {} + })); + // TODO further verify that this is the signature that should be revoked + if (signature) { + signature.revoked = revocationKeyIDs.some(keyID => keyID.equals(signature.issuerKeyID)) ? true : + signature.revoked || false; + return signature.revoked; + } + return revocationKeyIDs.length > 0; + } + + /** + * Returns key expiration time based on the given certification signature. + * The expiration time of the signature is ignored. + * @param {PublicSubkeyPacket|PublicKeyPacket} keyPacket - key to check + * @param {SignaturePacket} signature - signature to process + * @returns {Date|Infinity} expiration time or infinity if the key does not expire + */ + function getKeyExpirationTime(keyPacket, signature) { + let expirationTime; + // check V4 expiration time + if (signature.keyNeverExpires === false) { + expirationTime = keyPacket.created.getTime() + signature.keyExpirationTime * 1000; + } + return expirationTime ? new Date(expirationTime) : Infinity; + } + + /** + * Returns whether aead is supported by all keys in the set + * @param {Array} keys - Set of keys + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Array} [userIDs] - User IDs + * @param {Object} config - full configuration + * @returns {Promise} + * @async + */ + async function isAEADSupported(keys, date = new Date(), userIDs = [], config$1 = config) { + let supported = true; + // TODO replace when Promise.some or Promise.any are implemented + await Promise.all(keys.map(async function(key, i) { + const primaryUser = await key.getPrimaryUser(date, userIDs[i], config$1); + if (!primaryUser.selfCertification.features || + !(primaryUser.selfCertification.features[0] & enums.features.aead)) { + supported = false; + } + })); + return supported; + } + + function sanitizeKeyOptions(options, subkeyDefaults = {}) { + options.type = options.type || subkeyDefaults.type; + options.curve = options.curve || subkeyDefaults.curve; + options.rsaBits = options.rsaBits || subkeyDefaults.rsaBits; + options.keyExpirationTime = options.keyExpirationTime !== undefined ? options.keyExpirationTime : subkeyDefaults.keyExpirationTime; + options.passphrase = util.isString(options.passphrase) ? options.passphrase : subkeyDefaults.passphrase; + options.date = options.date || subkeyDefaults.date; + + options.sign = options.sign || false; + + switch (options.type) { + case 'ecc': + try { + options.curve = enums.write(enums.curve, options.curve); + } catch (e) { + throw new Error('Unknown curve'); + } + if (options.curve === enums.curve.ed25519 || options.curve === enums.curve.curve25519) { + options.curve = options.sign ? enums.curve.ed25519 : enums.curve.curve25519; + } + if (options.sign) { + options.algorithm = options.curve === enums.curve.ed25519 ? enums.publicKey.eddsa : enums.publicKey.ecdsa; + } else { + options.algorithm = enums.publicKey.ecdh; + } + break; + case 'rsa': + options.algorithm = enums.publicKey.rsaEncryptSign; + break; + default: + throw new Error(`Unsupported key type ${options.type}`); + } + return options; + } + + function isValidSigningKeyPacket(keyPacket, signature) { + const keyAlgo = keyPacket.algorithm; + return keyAlgo !== enums.publicKey.rsaEncrypt && + keyAlgo !== enums.publicKey.elgamal && + keyAlgo !== enums.publicKey.ecdh && + (!signature.keyFlags || + (signature.keyFlags[0] & enums.keyFlags.signData) !== 0); + } + + function isValidEncryptionKeyPacket(keyPacket, signature) { + const keyAlgo = keyPacket.algorithm; + return keyAlgo !== enums.publicKey.dsa && + keyAlgo !== enums.publicKey.rsaSign && + keyAlgo !== enums.publicKey.ecdsa && + keyAlgo !== enums.publicKey.eddsa && + (!signature.keyFlags || + (signature.keyFlags[0] & enums.keyFlags.encryptCommunication) !== 0 || + (signature.keyFlags[0] & enums.keyFlags.encryptStorage) !== 0); + } + + function isValidDecryptionKeyPacket(signature, config) { + if (config.allowInsecureDecryptionWithSigningKeys) { + // This is only relevant for RSA keys, all other signing algorithms cannot decrypt + return true; + } + + return !signature.keyFlags || + (signature.keyFlags[0] & enums.keyFlags.encryptCommunication) !== 0 || + (signature.keyFlags[0] & enums.keyFlags.encryptStorage) !== 0; + } + + /** + * Check key against blacklisted algorithms and minimum strength requirements. + * @param {SecretKeyPacket|PublicKeyPacket| + * SecretSubkeyPacket|PublicSubkeyPacket} keyPacket + * @param {Config} config + * @throws {Error} if the key packet does not meet the requirements + */ + function checkKeyRequirements(keyPacket, config) { + const keyAlgo = enums.write(enums.publicKey, keyPacket.algorithm); + const algoInfo = keyPacket.getAlgorithmInfo(); + if (config.rejectPublicKeyAlgorithms.has(keyAlgo)) { + throw new Error(`${algoInfo.algorithm} keys are considered too weak.`); + } + switch (keyAlgo) { + case enums.publicKey.rsaEncryptSign: + case enums.publicKey.rsaSign: + case enums.publicKey.rsaEncrypt: + if (algoInfo.bits < config.minRSABits) { + throw new Error(`RSA keys shorter than ${config.minRSABits} bits are considered too weak.`); + } + break; + case enums.publicKey.ecdsa: + case enums.publicKey.eddsa: + case enums.publicKey.ecdh: + if (config.rejectCurves.has(algoInfo.curve)) { + throw new Error(`Support for ${algoInfo.algorithm} keys using curve ${algoInfo.curve} is disabled.`); + } + break; + } + } + + /** + * @module key/User + * @private + */ + + /** + * Class that represents an user ID or attribute packet and the relevant signatures. + * @param {UserIDPacket|UserAttributePacket} userPacket - packet containing the user info + * @param {Key} mainKey - reference to main Key object containing the primary key and subkeys that the user is associated with + */ + class User { + constructor(userPacket, mainKey) { + this.userID = userPacket.constructor.tag === enums.packet.userID ? userPacket : null; + this.userAttribute = userPacket.constructor.tag === enums.packet.userAttribute ? userPacket : null; + this.selfCertifications = []; + this.otherCertifications = []; + this.revocationSignatures = []; + this.mainKey = mainKey; + } + + /** + * Transforms structured user data to packetlist + * @returns {PacketList} + */ + toPacketList() { + const packetlist = new PacketList(); + packetlist.push(this.userID || this.userAttribute); + packetlist.push(...this.revocationSignatures); + packetlist.push(...this.selfCertifications); + packetlist.push(...this.otherCertifications); + return packetlist; + } + + /** + * Shallow clone + * @returns {User} + */ + clone() { + const user = new User(this.userID || this.userAttribute, this.mainKey); + user.selfCertifications = [...this.selfCertifications]; + user.otherCertifications = [...this.otherCertifications]; + user.revocationSignatures = [...this.revocationSignatures]; + return user; + } + + /** + * Generate third-party certifications over this user and its primary key + * @param {Array} signingKeys - Decrypted private keys for signing + * @param {Date} [date] - Date to use as creation date of the certificate, instead of the current time + * @param {Object} config - Full configuration + * @returns {Promise} New user with new certifications. + * @async + */ + async certify(signingKeys, date, config) { + const primaryKey = this.mainKey.keyPacket; + const dataToSign = { + userID: this.userID, + userAttribute: this.userAttribute, + key: primaryKey + }; + const user = new User(dataToSign.userID || dataToSign.userAttribute, this.mainKey); + user.otherCertifications = await Promise.all(signingKeys.map(async function(privateKey) { + if (!privateKey.isPrivate()) { + throw new Error('Need private key for signing'); + } + if (privateKey.hasSameFingerprintAs(primaryKey)) { + throw new Error("The user's own key can only be used for self-certifications"); + } + const signingKey = await privateKey.getSigningKey(undefined, date, undefined, config); + return createSignaturePacket(dataToSign, privateKey, signingKey.keyPacket, { + // Most OpenPGP implementations use generic certification (0x10) + signatureType: enums.signature.certGeneric, + keyFlags: [enums.keyFlags.certifyKeys | enums.keyFlags.signData] + }, date, undefined, undefined, undefined, config); + })); + await user.update(this, date, config); + return user; + } + + /** + * Checks if a given certificate of the user is revoked + * @param {SignaturePacket} certificate - The certificate to verify + * @param {PublicSubkeyPacket| + * SecretSubkeyPacket| + * PublicKeyPacket| + * SecretKeyPacket} [keyPacket] The key packet to verify the signature, instead of the primary key + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Object} config - Full configuration + * @returns {Promise} True if the certificate is revoked. + * @async + */ + async isRevoked(certificate, keyPacket, date = new Date(), config$1 = config) { + const primaryKey = this.mainKey.keyPacket; + return isDataRevoked(primaryKey, enums.signature.certRevocation, { + key: primaryKey, + userID: this.userID, + userAttribute: this.userAttribute + }, this.revocationSignatures, certificate, keyPacket, date, config$1); + } + + /** + * Verifies the user certificate. + * @param {SignaturePacket} certificate - A certificate of this user + * @param {Array} verificationKeys - Array of keys to verify certificate signatures + * @param {Date} [date] - Use the given date instead of the current time + * @param {Object} config - Full configuration + * @returns {Promise} true if the certificate could be verified, or null if the verification keys do not correspond to the certificate + * @throws if the user certificate is invalid. + * @async + */ + async verifyCertificate(certificate, verificationKeys, date = new Date(), config) { + const that = this; + const primaryKey = this.mainKey.keyPacket; + const dataToVerify = { + userID: this.userID, + userAttribute: this.userAttribute, + key: primaryKey + }; + const { issuerKeyID } = certificate; + const issuerKeys = verificationKeys.filter(key => key.getKeys(issuerKeyID).length > 0); + if (issuerKeys.length === 0) { + return null; + } + await Promise.all(issuerKeys.map(async key => { + const signingKey = await key.getSigningKey(issuerKeyID, certificate.created, undefined, config); + if (certificate.revoked || await that.isRevoked(certificate, signingKey.keyPacket, date, config)) { + throw new Error('User certificate is revoked'); + } + try { + await certificate.verify(signingKey.keyPacket, enums.signature.certGeneric, dataToVerify, date, undefined, config); + } catch (e) { + throw util.wrapError('User certificate is invalid', e); + } + })); + return true; + } + + /** + * Verifies all user certificates + * @param {Array} verificationKeys - Array of keys to verify certificate signatures + * @param {Date} [date] - Use the given date instead of the current time + * @param {Object} config - Full configuration + * @returns {Promise>} List of signer's keyID and validity of signature. + * Signature validity is null if the verification keys do not correspond to the certificate. + * @async + */ + async verifyAllCertifications(verificationKeys, date = new Date(), config) { + const that = this; + const certifications = this.selfCertifications.concat(this.otherCertifications); + return Promise.all(certifications.map(async certification => ({ + keyID: certification.issuerKeyID, + valid: await that.verifyCertificate(certification, verificationKeys, date, config).catch(() => false) + }))); + } + + /** + * Verify User. Checks for existence of self signatures, revocation signatures + * and validity of self signature. + * @param {Date} date - Use the given date instead of the current time + * @param {Object} config - Full configuration + * @returns {Promise} Status of user. + * @throws {Error} if there are no valid self signatures. + * @async + */ + async verify(date = new Date(), config) { + if (!this.selfCertifications.length) { + throw new Error('No self-certifications found'); + } + const that = this; + const primaryKey = this.mainKey.keyPacket; + const dataToVerify = { + userID: this.userID, + userAttribute: this.userAttribute, + key: primaryKey + }; + // TODO replace when Promise.some or Promise.any are implemented + let exception; + for (let i = this.selfCertifications.length - 1; i >= 0; i--) { + try { + const selfCertification = this.selfCertifications[i]; + if (selfCertification.revoked || await that.isRevoked(selfCertification, undefined, date, config)) { + throw new Error('Self-certification is revoked'); + } + try { + await selfCertification.verify(primaryKey, enums.signature.certGeneric, dataToVerify, date, undefined, config); + } catch (e) { + throw util.wrapError('Self-certification is invalid', e); + } + return true; + } catch (e) { + exception = e; + } + } + throw exception; + } + + /** + * Update user with new components from specified user + * @param {User} sourceUser - Source user to merge + * @param {Date} date - Date to verify the validity of signatures + * @param {Object} config - Full configuration + * @returns {Promise} + * @async + */ + async update(sourceUser, date, config) { + const primaryKey = this.mainKey.keyPacket; + const dataToVerify = { + userID: this.userID, + userAttribute: this.userAttribute, + key: primaryKey + }; + // self signatures + await mergeSignatures(sourceUser, this, 'selfCertifications', date, async function(srcSelfSig) { + try { + await srcSelfSig.verify(primaryKey, enums.signature.certGeneric, dataToVerify, date, false, config); + return true; + } catch (e) { + return false; + } + }); + // other signatures + await mergeSignatures(sourceUser, this, 'otherCertifications', date); + // revocation signatures + await mergeSignatures(sourceUser, this, 'revocationSignatures', date, function(srcRevSig) { + return isDataRevoked(primaryKey, enums.signature.certRevocation, dataToVerify, [srcRevSig], undefined, undefined, date, config); + }); + } + + /** + * Revokes the user + * @param {SecretKeyPacket} primaryKey - decrypted private primary key for revocation + * @param {Object} reasonForRevocation - optional, object indicating the reason for revocation + * @param {module:enums.reasonForRevocation} reasonForRevocation.flag optional, flag indicating the reason for revocation + * @param {String} reasonForRevocation.string optional, string explaining the reason for revocation + * @param {Date} date - optional, override the creationtime of the revocation signature + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New user with revocation signature. + * @async + */ + async revoke( + primaryKey, + { + flag: reasonForRevocationFlag = enums.reasonForRevocation.noReason, + string: reasonForRevocationString = '' + } = {}, + date = new Date(), + config$1 = config + ) { + const dataToSign = { + userID: this.userID, + userAttribute: this.userAttribute, + key: primaryKey + }; + const user = new User(dataToSign.userID || dataToSign.userAttribute, this.mainKey); + user.revocationSignatures.push(await createSignaturePacket(dataToSign, null, primaryKey, { + signatureType: enums.signature.certRevocation, + reasonForRevocationFlag: enums.write(enums.reasonForRevocation, reasonForRevocationFlag), + reasonForRevocationString + }, date, undefined, undefined, false, config$1)); + await user.update(this); + return user; + } + } + + /** + * @module key/Subkey + * @private + */ + + /** + * Class that represents a subkey packet and the relevant signatures. + * @borrows PublicSubkeyPacket#getKeyID as Subkey#getKeyID + * @borrows PublicSubkeyPacket#getFingerprint as Subkey#getFingerprint + * @borrows PublicSubkeyPacket#hasSameFingerprintAs as Subkey#hasSameFingerprintAs + * @borrows PublicSubkeyPacket#getAlgorithmInfo as Subkey#getAlgorithmInfo + * @borrows PublicSubkeyPacket#getCreationTime as Subkey#getCreationTime + * @borrows PublicSubkeyPacket#isDecrypted as Subkey#isDecrypted + */ + class Subkey { + /** + * @param {SecretSubkeyPacket|PublicSubkeyPacket} subkeyPacket - subkey packet to hold in the Subkey + * @param {Key} mainKey - reference to main Key object, containing the primary key packet corresponding to the subkey + */ + constructor(subkeyPacket, mainKey) { + this.keyPacket = subkeyPacket; + this.bindingSignatures = []; + this.revocationSignatures = []; + this.mainKey = mainKey; + } + + /** + * Transforms structured subkey data to packetlist + * @returns {PacketList} + */ + toPacketList() { + const packetlist = new PacketList(); + packetlist.push(this.keyPacket); + packetlist.push(...this.revocationSignatures); + packetlist.push(...this.bindingSignatures); + return packetlist; + } + + /** + * Shallow clone + * @return {Subkey} + */ + clone() { + const subkey = new Subkey(this.keyPacket, this.mainKey); + subkey.bindingSignatures = [...this.bindingSignatures]; + subkey.revocationSignatures = [...this.revocationSignatures]; + return subkey; + } + + /** + * Checks if a binding signature of a subkey is revoked + * @param {SignaturePacket} signature - The binding signature to verify + * @param {PublicSubkeyPacket| + * SecretSubkeyPacket| + * PublicKeyPacket| + * SecretKeyPacket} key, optional The key to verify the signature + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} True if the binding signature is revoked. + * @async + */ + async isRevoked(signature, key, date = new Date(), config$1 = config) { + const primaryKey = this.mainKey.keyPacket; + return isDataRevoked( + primaryKey, enums.signature.subkeyRevocation, { + key: primaryKey, + bind: this.keyPacket + }, this.revocationSignatures, signature, key, date, config$1 + ); + } + + /** + * Verify subkey. Checks for revocation signatures, expiration time + * and valid binding signature. + * @param {Date} date - Use the given date instead of the current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} + * @throws {Error} if the subkey is invalid. + * @async + */ + async verify(date = new Date(), config$1 = config) { + const primaryKey = this.mainKey.keyPacket; + const dataToVerify = { key: primaryKey, bind: this.keyPacket }; + // check subkey binding signatures + const bindingSignature = await getLatestValidSignature(this.bindingSignatures, primaryKey, enums.signature.subkeyBinding, dataToVerify, date, config$1); + // check binding signature is not revoked + if (bindingSignature.revoked || await this.isRevoked(bindingSignature, null, date, config$1)) { + throw new Error('Subkey is revoked'); + } + // check for expiration time + if (isDataExpired(this.keyPacket, bindingSignature, date)) { + throw new Error('Subkey is expired'); + } + return bindingSignature; + } + + /** + * Returns the expiration time of the subkey or Infinity if key does not expire. + * Returns null if the subkey is invalid. + * @param {Date} date - Use the given date instead of the current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} + * @async + */ + async getExpirationTime(date = new Date(), config$1 = config) { + const primaryKey = this.mainKey.keyPacket; + const dataToVerify = { key: primaryKey, bind: this.keyPacket }; + let bindingSignature; + try { + bindingSignature = await getLatestValidSignature(this.bindingSignatures, primaryKey, enums.signature.subkeyBinding, dataToVerify, date, config$1); + } catch (e) { + return null; + } + const keyExpiry = getKeyExpirationTime(this.keyPacket, bindingSignature); + const sigExpiry = bindingSignature.getExpirationTime(); + return keyExpiry < sigExpiry ? keyExpiry : sigExpiry; + } + + /** + * Update subkey with new components from specified subkey + * @param {Subkey} subkey - Source subkey to merge + * @param {Date} [date] - Date to verify validity of signatures + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} if update failed + * @async + */ + async update(subkey, date = new Date(), config$1 = config) { + const primaryKey = this.mainKey.keyPacket; + if (!this.hasSameFingerprintAs(subkey)) { + throw new Error('Subkey update method: fingerprints of subkeys not equal'); + } + // key packet + if (this.keyPacket.constructor.tag === enums.packet.publicSubkey && + subkey.keyPacket.constructor.tag === enums.packet.secretSubkey) { + this.keyPacket = subkey.keyPacket; + } + // update missing binding signatures + const that = this; + const dataToVerify = { key: primaryKey, bind: that.keyPacket }; + await mergeSignatures(subkey, this, 'bindingSignatures', date, async function(srcBindSig) { + for (let i = 0; i < that.bindingSignatures.length; i++) { + if (that.bindingSignatures[i].issuerKeyID.equals(srcBindSig.issuerKeyID)) { + if (srcBindSig.created > that.bindingSignatures[i].created) { + that.bindingSignatures[i] = srcBindSig; + } + return false; + } + } + try { + await srcBindSig.verify(primaryKey, enums.signature.subkeyBinding, dataToVerify, date, undefined, config$1); + return true; + } catch (e) { + return false; + } + }); + // revocation signatures + await mergeSignatures(subkey, this, 'revocationSignatures', date, function(srcRevSig) { + return isDataRevoked(primaryKey, enums.signature.subkeyRevocation, dataToVerify, [srcRevSig], undefined, undefined, date, config$1); + }); + } + + /** + * Revokes the subkey + * @param {SecretKeyPacket} primaryKey - decrypted private primary key for revocation + * @param {Object} reasonForRevocation - optional, object indicating the reason for revocation + * @param {module:enums.reasonForRevocation} reasonForRevocation.flag optional, flag indicating the reason for revocation + * @param {String} reasonForRevocation.string optional, string explaining the reason for revocation + * @param {Date} date - optional, override the creationtime of the revocation signature + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New subkey with revocation signature. + * @async + */ + async revoke( + primaryKey, + { + flag: reasonForRevocationFlag = enums.reasonForRevocation.noReason, + string: reasonForRevocationString = '' + } = {}, + date = new Date(), + config$1 = config + ) { + const dataToSign = { key: primaryKey, bind: this.keyPacket }; + const subkey = new Subkey(this.keyPacket, this.mainKey); + subkey.revocationSignatures.push(await createSignaturePacket(dataToSign, null, primaryKey, { + signatureType: enums.signature.subkeyRevocation, + reasonForRevocationFlag: enums.write(enums.reasonForRevocation, reasonForRevocationFlag), + reasonForRevocationString + }, date, undefined, undefined, false, config$1)); + await subkey.update(this); + return subkey; + } + + hasSameFingerprintAs(other) { + return this.keyPacket.hasSameFingerprintAs(other.keyPacket || other); + } + } + + ['getKeyID', 'getFingerprint', 'getAlgorithmInfo', 'getCreationTime', 'isDecrypted'].forEach(name => { + Subkey.prototype[name] = + function() { + return this.keyPacket[name](); + }; + }); + + // GPG4Browsers - An OpenPGP implementation in javascript + + // A key revocation certificate can contain the following packets + const allowedRevocationPackets = /*#__PURE__*/ util.constructAllowedPackets([SignaturePacket]); + const mainKeyPacketTags = new Set([enums.packet.publicKey, enums.packet.privateKey]); + const keyPacketTags = new Set([ + enums.packet.publicKey, enums.packet.privateKey, + enums.packet.publicSubkey, enums.packet.privateSubkey + ]); + + /** + * Abstract class that represents an OpenPGP key. Must contain a primary key. + * Can contain additional subkeys, signatures, user ids, user attributes. + * @borrows PublicKeyPacket#getKeyID as Key#getKeyID + * @borrows PublicKeyPacket#getFingerprint as Key#getFingerprint + * @borrows PublicKeyPacket#hasSameFingerprintAs as Key#hasSameFingerprintAs + * @borrows PublicKeyPacket#getAlgorithmInfo as Key#getAlgorithmInfo + * @borrows PublicKeyPacket#getCreationTime as Key#getCreationTime + */ + class Key { + /** + * Transforms packetlist to structured key data + * @param {PacketList} packetlist - The packets that form a key + * @param {Set} disallowedPackets - disallowed packet tags + */ + packetListToStructure(packetlist, disallowedPackets = new Set()) { + let user; + let primaryKeyID; + let subkey; + let ignoreUntil; + + for (const packet of packetlist) { + + if (packet instanceof UnparseablePacket) { + const isUnparseableKeyPacket = keyPacketTags.has(packet.tag); + if (isUnparseableKeyPacket && !ignoreUntil) { + // Since non-key packets apply to the preceding key packet, if a (sub)key is Unparseable we must + // discard all non-key packets that follow, until another (sub)key packet is found. + if (mainKeyPacketTags.has(packet.tag)) { + ignoreUntil = mainKeyPacketTags; + } else { + ignoreUntil = keyPacketTags; + } + } + continue; + } + + const tag = packet.constructor.tag; + if (ignoreUntil) { + if (!ignoreUntil.has(tag)) continue; + ignoreUntil = null; + } + if (disallowedPackets.has(tag)) { + throw new Error(`Unexpected packet type: ${tag}`); + } + switch (tag) { + case enums.packet.publicKey: + case enums.packet.secretKey: + if (this.keyPacket) { + throw new Error('Key block contains multiple keys'); + } + this.keyPacket = packet; + primaryKeyID = this.getKeyID(); + if (!primaryKeyID) { + throw new Error('Missing Key ID'); + } + break; + case enums.packet.userID: + case enums.packet.userAttribute: + user = new User(packet, this); + this.users.push(user); + break; + case enums.packet.publicSubkey: + case enums.packet.secretSubkey: + user = null; + subkey = new Subkey(packet, this); + this.subkeys.push(subkey); + break; + case enums.packet.signature: + switch (packet.signatureType) { + case enums.signature.certGeneric: + case enums.signature.certPersona: + case enums.signature.certCasual: + case enums.signature.certPositive: + if (!user) { + util.printDebug('Dropping certification signatures without preceding user packet'); + continue; + } + if (packet.issuerKeyID.equals(primaryKeyID)) { + user.selfCertifications.push(packet); + } else { + user.otherCertifications.push(packet); + } + break; + case enums.signature.certRevocation: + if (user) { + user.revocationSignatures.push(packet); + } else { + this.directSignatures.push(packet); + } + break; + case enums.signature.key: + this.directSignatures.push(packet); + break; + case enums.signature.subkeyBinding: + if (!subkey) { + util.printDebug('Dropping subkey binding signature without preceding subkey packet'); + continue; + } + subkey.bindingSignatures.push(packet); + break; + case enums.signature.keyRevocation: + this.revocationSignatures.push(packet); + break; + case enums.signature.subkeyRevocation: + if (!subkey) { + util.printDebug('Dropping subkey revocation signature without preceding subkey packet'); + continue; + } + subkey.revocationSignatures.push(packet); + break; + } + break; + } + } + } + + /** + * Transforms structured key data to packetlist + * @returns {PacketList} The packets that form a key. + */ + toPacketList() { + const packetlist = new PacketList(); + packetlist.push(this.keyPacket); + packetlist.push(...this.revocationSignatures); + packetlist.push(...this.directSignatures); + this.users.map(user => packetlist.push(...user.toPacketList())); + this.subkeys.map(subkey => packetlist.push(...subkey.toPacketList())); + return packetlist; + } + + /** + * Clones the key object. The copy is shallow, as it references the same packet objects as the original. However, if the top-level API is used, the two key instances are effectively independent. + * @param {Boolean} [clonePrivateParams=false] Only relevant for private keys: whether the secret key paramenters should be deeply copied. This is needed if e.g. `encrypt()` is to be called either on the clone or the original key. + * @returns {Promise} Clone of the key. + */ + clone(clonePrivateParams = false) { + const key = new this.constructor(this.toPacketList()); + if (clonePrivateParams) { + key.getKeys().forEach(k => { + // shallow clone the key packets + k.keyPacket = Object.create( + Object.getPrototypeOf(k.keyPacket), + Object.getOwnPropertyDescriptors(k.keyPacket) + ); + if (!k.keyPacket.isDecrypted()) return; + // deep clone the private params, which are cleared during encryption + const privateParams = {}; + Object.keys(k.keyPacket.privateParams).forEach(name => { + privateParams[name] = new Uint8Array(k.keyPacket.privateParams[name]); + }); + k.keyPacket.privateParams = privateParams; + }); + } + return key; + } + + /** + * Returns an array containing all public or private subkeys matching keyID; + * If no keyID is given, returns all subkeys. + * @param {type/keyID} [keyID] - key ID to look for + * @returns {Array} array of subkeys + */ + getSubkeys(keyID = null) { + const subkeys = this.subkeys.filter(subkey => ( + !keyID || subkey.getKeyID().equals(keyID, true) + )); + return subkeys; + } + + /** + * Returns an array containing all public or private keys matching keyID. + * If no keyID is given, returns all keys, starting with the primary key. + * @param {type/keyid~KeyID} [keyID] - key ID to look for + * @returns {Array} array of keys + */ + getKeys(keyID = null) { + const keys = []; + if (!keyID || this.getKeyID().equals(keyID, true)) { + keys.push(this); + } + return keys.concat(this.getSubkeys(keyID)); + } + + /** + * Returns key IDs of all keys + * @returns {Array} + */ + getKeyIDs() { + return this.getKeys().map(key => key.getKeyID()); + } + + /** + * Returns userIDs + * @returns {Array} Array of userIDs. + */ + getUserIDs() { + return this.users.map(user => { + return user.userID ? user.userID.userID : null; + }).filter(userID => userID !== null); + } + + /** + * Returns binary encoded key + * @returns {Uint8Array} Binary key. + */ + write() { + return this.toPacketList().write(); + } + + /** + * Returns last created key or key by given keyID that is available for signing and verification + * @param {module:type/keyid~KeyID} [keyID] - key ID of a specific key to retrieve + * @param {Date} [date] - use the fiven date date to to check key validity instead of the current date + * @param {Object} [userID] - filter keys for the given user ID + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} signing key + * @throws if no valid signing key was found + * @async + */ + async getSigningKey(keyID = null, date = new Date(), userID = {}, config$1 = config) { + await this.verifyPrimaryKey(date, userID, config$1); + const primaryKey = this.keyPacket; + const subkeys = this.subkeys.slice().sort((a, b) => b.keyPacket.created - a.keyPacket.created); + let exception; + for (const subkey of subkeys) { + if (!keyID || subkey.getKeyID().equals(keyID)) { + try { + await subkey.verify(date, config$1); + const dataToVerify = { key: primaryKey, bind: subkey.keyPacket }; + const bindingSignature = await getLatestValidSignature( + subkey.bindingSignatures, primaryKey, enums.signature.subkeyBinding, dataToVerify, date, config$1 + ); + if (!isValidSigningKeyPacket(subkey.keyPacket, bindingSignature)) { + continue; + } + if (!bindingSignature.embeddedSignature) { + throw new Error('Missing embedded signature'); + } + // verify embedded signature + await getLatestValidSignature( + [bindingSignature.embeddedSignature], subkey.keyPacket, enums.signature.keyBinding, dataToVerify, date, config$1 + ); + checkKeyRequirements(subkey.keyPacket, config$1); + return subkey; + } catch (e) { + exception = e; + } + } + } + + try { + const primaryUser = await this.getPrimaryUser(date, userID, config$1); + if ((!keyID || primaryKey.getKeyID().equals(keyID)) && + isValidSigningKeyPacket(primaryKey, primaryUser.selfCertification, config$1)) { + checkKeyRequirements(primaryKey, config$1); + return this; + } + } catch (e) { + exception = e; + } + throw util.wrapError('Could not find valid signing key packet in key ' + this.getKeyID().toHex(), exception); + } + + /** + * Returns last created key or key by given keyID that is available for encryption or decryption + * @param {module:type/keyid~KeyID} [keyID] - key ID of a specific key to retrieve + * @param {Date} [date] - use the fiven date date to to check key validity instead of the current date + * @param {Object} [userID] - filter keys for the given user ID + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} encryption key + * @throws if no valid encryption key was found + * @async + */ + async getEncryptionKey(keyID, date = new Date(), userID = {}, config$1 = config) { + await this.verifyPrimaryKey(date, userID, config$1); + const primaryKey = this.keyPacket; + // V4: by convention subkeys are preferred for encryption service + const subkeys = this.subkeys.slice().sort((a, b) => b.keyPacket.created - a.keyPacket.created); + let exception; + for (const subkey of subkeys) { + if (!keyID || subkey.getKeyID().equals(keyID)) { + try { + await subkey.verify(date, config$1); + const dataToVerify = { key: primaryKey, bind: subkey.keyPacket }; + const bindingSignature = await getLatestValidSignature(subkey.bindingSignatures, primaryKey, enums.signature.subkeyBinding, dataToVerify, date, config$1); + if (isValidEncryptionKeyPacket(subkey.keyPacket, bindingSignature)) { + checkKeyRequirements(subkey.keyPacket, config$1); + return subkey; + } + } catch (e) { + exception = e; + } + } + } + + try { + // if no valid subkey for encryption, evaluate primary key + const primaryUser = await this.getPrimaryUser(date, userID, config$1); + if ((!keyID || primaryKey.getKeyID().equals(keyID)) && + isValidEncryptionKeyPacket(primaryKey, primaryUser.selfCertification)) { + checkKeyRequirements(primaryKey, config$1); + return this; + } + } catch (e) { + exception = e; + } + throw util.wrapError('Could not find valid encryption key packet in key ' + this.getKeyID().toHex(), exception); + } + + /** + * Checks if a signature on a key is revoked + * @param {SignaturePacket} signature - The signature to verify + * @param {PublicSubkeyPacket| + * SecretSubkeyPacket| + * PublicKeyPacket| + * SecretKeyPacket} key, optional The key to verify the signature + * @param {Date} [date] - Use the given date for verification, instead of the current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} True if the certificate is revoked. + * @async + */ + async isRevoked(signature, key, date = new Date(), config$1 = config) { + return isDataRevoked( + this.keyPacket, enums.signature.keyRevocation, { key: this.keyPacket }, this.revocationSignatures, signature, key, date, config$1 + ); + } + + /** + * Verify primary key. Checks for revocation signatures, expiration time + * and valid self signature. Throws if the primary key is invalid. + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Object} [userID] - User ID + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} If key verification failed + * @async + */ + async verifyPrimaryKey(date = new Date(), userID = {}, config$1 = config) { + const primaryKey = this.keyPacket; + // check for key revocation signatures + if (await this.isRevoked(null, null, date, config$1)) { + throw new Error('Primary key is revoked'); + } + // check for valid, unrevoked, unexpired self signature + const { selfCertification } = await this.getPrimaryUser(date, userID, config$1); + // check for expiration time in binding signatures + if (isDataExpired(primaryKey, selfCertification, date)) { + throw new Error('Primary key is expired'); + } + // check for expiration time in direct signatures + const directSignature = await getLatestValidSignature( + this.directSignatures, primaryKey, enums.signature.key, { key: primaryKey }, date, config$1 + ).catch(() => {}); // invalid signatures are discarded, to avoid breaking the key + + if (directSignature && isDataExpired(primaryKey, directSignature, date)) { + throw new Error('Primary key is expired'); + } + } + + /** + * Returns the expiration date of the primary key, considering self-certifications and direct-key signatures. + * Returns `Infinity` if the key doesn't expire, or `null` if the key is revoked or invalid. + * @param {Object} [userID] - User ID to consider instead of the primary user + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} + * @async + */ + async getExpirationTime(userID, config$1 = config) { + let primaryKeyExpiry; + try { + const { selfCertification } = await this.getPrimaryUser(null, userID, config$1); + const selfSigKeyExpiry = getKeyExpirationTime(this.keyPacket, selfCertification); + const selfSigExpiry = selfCertification.getExpirationTime(); + const directSignature = await getLatestValidSignature( + this.directSignatures, this.keyPacket, enums.signature.key, { key: this.keyPacket }, null, config$1 + ).catch(() => {}); + if (directSignature) { + const directSigKeyExpiry = getKeyExpirationTime(this.keyPacket, directSignature); + // We do not support the edge case where the direct signature expires, since it would invalidate the corresponding key expiration, + // causing a discountinous validy period for the key + primaryKeyExpiry = Math.min(selfSigKeyExpiry, selfSigExpiry, directSigKeyExpiry); + } else { + primaryKeyExpiry = selfSigKeyExpiry < selfSigExpiry ? selfSigKeyExpiry : selfSigExpiry; + } + } catch (e) { + primaryKeyExpiry = null; + } + + return util.normalizeDate(primaryKeyExpiry); + } + + + /** + * Returns primary user and most significant (latest valid) self signature + * - if multiple primary users exist, returns the one with the latest self signature + * - otherwise, returns the user with the latest self signature + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Object} [userID] - User ID to get instead of the primary user, if it exists + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise<{ + * user: User, + * selfCertification: SignaturePacket + * }>} The primary user and the self signature + * @async + */ + async getPrimaryUser(date = new Date(), userID = {}, config$1 = config) { + const primaryKey = this.keyPacket; + const users = []; + let exception; + for (let i = 0; i < this.users.length; i++) { + try { + const user = this.users[i]; + if (!user.userID) { + continue; + } + if ( + (userID.name !== undefined && user.userID.name !== userID.name) || + (userID.email !== undefined && user.userID.email !== userID.email) || + (userID.comment !== undefined && user.userID.comment !== userID.comment) + ) { + throw new Error('Could not find user that matches that user ID'); + } + const dataToVerify = { userID: user.userID, key: primaryKey }; + const selfCertification = await getLatestValidSignature(user.selfCertifications, primaryKey, enums.signature.certGeneric, dataToVerify, date, config$1); + users.push({ index: i, user, selfCertification }); + } catch (e) { + exception = e; + } + } + if (!users.length) { + throw exception || new Error('Could not find primary user'); + } + await Promise.all(users.map(async function (a) { + return a.selfCertification.revoked || a.user.isRevoked(a.selfCertification, null, date, config$1); + })); + // sort by primary user flag and signature creation time + const primaryUser = users.sort(function(a, b) { + const A = a.selfCertification; + const B = b.selfCertification; + return B.revoked - A.revoked || A.isPrimaryUserID - B.isPrimaryUserID || A.created - B.created; + }).pop(); + const { user, selfCertification: cert } = primaryUser; + if (cert.revoked || await user.isRevoked(cert, null, date, config$1)) { + throw new Error('Primary user is revoked'); + } + return primaryUser; + } + + /** + * Update key with new components from specified key with same key ID: + * users, subkeys, certificates are merged into the destination key, + * duplicates and expired signatures are ignored. + * + * If the source key is a private key and the destination key is public, + * a private key is returned. + * @param {Key} sourceKey - Source key to merge + * @param {Date} [date] - Date to verify validity of signatures and keys + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} updated key + * @async + */ + async update(sourceKey, date = new Date(), config$1 = config) { + if (!this.hasSameFingerprintAs(sourceKey)) { + throw new Error('Primary key fingerprints must be equal to update the key'); + } + if (!this.isPrivate() && sourceKey.isPrivate()) { + // check for equal subkey packets + const equal = (this.subkeys.length === sourceKey.subkeys.length) && + (this.subkeys.every(destSubkey => { + return sourceKey.subkeys.some(srcSubkey => { + return destSubkey.hasSameFingerprintAs(srcSubkey); + }); + })); + if (!equal) { + throw new Error('Cannot update public key with private key if subkeys mismatch'); + } + + return sourceKey.update(this, config$1); + } + // from here on, either: + // - destination key is private, source key is public + // - the keys are of the same type + // hence we don't need to convert the destination key type + const updatedKey = this.clone(); + // revocation signatures + await mergeSignatures(sourceKey, updatedKey, 'revocationSignatures', date, srcRevSig => { + return isDataRevoked(updatedKey.keyPacket, enums.signature.keyRevocation, updatedKey, [srcRevSig], null, sourceKey.keyPacket, date, config$1); + }); + // direct signatures + await mergeSignatures(sourceKey, updatedKey, 'directSignatures', date); + // update users + await Promise.all(sourceKey.users.map(async srcUser => { + // multiple users with the same ID/attribute are not explicitly disallowed by the spec + // hence we support them, just in case + const usersToUpdate = updatedKey.users.filter(dstUser => ( + (srcUser.userID && srcUser.userID.equals(dstUser.userID)) || + (srcUser.userAttribute && srcUser.userAttribute.equals(dstUser.userAttribute)) + )); + if (usersToUpdate.length > 0) { + await Promise.all( + usersToUpdate.map(userToUpdate => userToUpdate.update(srcUser, date, config$1)) + ); + } else { + const newUser = srcUser.clone(); + newUser.mainKey = updatedKey; + updatedKey.users.push(newUser); + } + })); + // update subkeys + await Promise.all(sourceKey.subkeys.map(async srcSubkey => { + // multiple subkeys with same fingerprint might be preset + const subkeysToUpdate = updatedKey.subkeys.filter(dstSubkey => ( + dstSubkey.hasSameFingerprintAs(srcSubkey) + )); + if (subkeysToUpdate.length > 0) { + await Promise.all( + subkeysToUpdate.map(subkeyToUpdate => subkeyToUpdate.update(srcSubkey, date, config$1)) + ); + } else { + const newSubkey = srcSubkey.clone(); + newSubkey.mainKey = updatedKey; + updatedKey.subkeys.push(newSubkey); + } + })); + + return updatedKey; + } + + /** + * Get revocation certificate from a revoked key. + * (To get a revocation certificate for an unrevoked key, call revoke() first.) + * @param {Date} date - Use the given date instead of the current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} Armored revocation certificate. + * @async + */ + async getRevocationCertificate(date = new Date(), config$1 = config) { + const dataToVerify = { key: this.keyPacket }; + const revocationSignature = await getLatestValidSignature(this.revocationSignatures, this.keyPacket, enums.signature.keyRevocation, dataToVerify, date, config$1); + const packetlist = new PacketList(); + packetlist.push(revocationSignature); + return armor(enums.armor.publicKey, packetlist.write(), null, null, 'This is a revocation certificate'); + } + + /** + * Applies a revocation certificate to a key + * This adds the first signature packet in the armored text to the key, + * if it is a valid revocation signature. + * @param {String} revocationCertificate - armored revocation certificate + * @param {Date} [date] - Date to verify the certificate + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} Revoked key. + * @async + */ + async applyRevocationCertificate(revocationCertificate, date = new Date(), config$1 = config) { + const input = await unarmor(revocationCertificate, config$1); + const packetlist = await PacketList.fromBinary(input.data, allowedRevocationPackets, config$1); + const revocationSignature = packetlist.findPacket(enums.packet.signature); + if (!revocationSignature || revocationSignature.signatureType !== enums.signature.keyRevocation) { + throw new Error('Could not find revocation signature packet'); + } + if (!revocationSignature.issuerKeyID.equals(this.getKeyID())) { + throw new Error('Revocation signature does not match key'); + } + try { + await revocationSignature.verify(this.keyPacket, enums.signature.keyRevocation, { key: this.keyPacket }, date, undefined, config$1); + } catch (e) { + throw util.wrapError('Could not verify revocation signature', e); + } + const key = this.clone(); + key.revocationSignatures.push(revocationSignature); + return key; + } + + /** + * Signs primary user of key + * @param {Array} privateKeys - decrypted private keys for signing + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Object} [userID] - User ID to get instead of the primary user, if it exists + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} Key with new certificate signature. + * @async + */ + async signPrimaryUser(privateKeys, date, userID, config$1 = config) { + const { index, user } = await this.getPrimaryUser(date, userID, config$1); + const userSign = await user.certify(privateKeys, date, config$1); + const key = this.clone(); + key.users[index] = userSign; + return key; + } + + /** + * Signs all users of key + * @param {Array} privateKeys - decrypted private keys for signing + * @param {Date} [date] - Use the given date for signing, instead of the current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} Key with new certificate signature. + * @async + */ + async signAllUsers(privateKeys, date = new Date(), config$1 = config) { + const key = this.clone(); + key.users = await Promise.all(this.users.map(function(user) { + return user.certify(privateKeys, date, config$1); + })); + return key; + } + + /** + * Verifies primary user of key + * - if no arguments are given, verifies the self certificates; + * - otherwise, verifies all certificates signed with given keys. + * @param {Array} [verificationKeys] - array of keys to verify certificate signatures, instead of the primary key + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Object} [userID] - User ID to get instead of the primary user, if it exists + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise>} List of signer's keyID and validity of signature. + * Signature validity is null if the verification keys do not correspond to the certificate. + * @async + */ + async verifyPrimaryUser(verificationKeys, date = new Date(), userID, config$1 = config) { + const primaryKey = this.keyPacket; + const { user } = await this.getPrimaryUser(date, userID, config$1); + const results = verificationKeys ? + await user.verifyAllCertifications(verificationKeys, date, config$1) : + [{ keyID: primaryKey.getKeyID(), valid: await user.verify(date, config$1).catch(() => false) }]; + return results; + } + + /** + * Verifies all users of key + * - if no arguments are given, verifies the self certificates; + * - otherwise, verifies all certificates signed with given keys. + * @param {Array} [verificationKeys] - array of keys to verify certificate signatures + * @param {Date} [date] - Use the given date for verification instead of the current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise>} List of userID, signer's keyID and validity of signature. + * Signature validity is null if the verification keys do not correspond to the certificate. + * @async + */ + async verifyAllUsers(verificationKeys, date = new Date(), config$1 = config) { + const primaryKey = this.keyPacket; + const results = []; + await Promise.all(this.users.map(async user => { + const signatures = verificationKeys ? + await user.verifyAllCertifications(verificationKeys, date, config$1) : + [{ keyID: primaryKey.getKeyID(), valid: await user.verify(date, config$1).catch(() => false) }]; + + results.push(...signatures.map( + signature => ({ + userID: user.userID.userID, + keyID: signature.keyID, + valid: signature.valid + })) + ); + })); + return results; + } + } + + ['getKeyID', 'getFingerprint', 'getAlgorithmInfo', 'getCreationTime', 'hasSameFingerprintAs'].forEach(name => { + Key.prototype[name] = + Subkey.prototype[name]; + }); + + // This library is free software; you can redistribute it and/or + + /** + * Class that represents an OpenPGP Public Key + */ + class PublicKey extends Key { + /** + * @param {PacketList} packetlist - The packets that form this key + */ + constructor(packetlist) { + super(); + this.keyPacket = null; + this.revocationSignatures = []; + this.directSignatures = []; + this.users = []; + this.subkeys = []; + if (packetlist) { + this.packetListToStructure(packetlist, new Set([enums.packet.secretKey, enums.packet.secretSubkey])); + if (!this.keyPacket) { + throw new Error('Invalid key: missing public-key packet'); + } + } + } + + /** + * Returns true if this is a private key + * @returns {false} + */ + isPrivate() { + return false; + } + + /** + * Returns key as public key (shallow copy) + * @returns {PublicKey} New public Key + */ + toPublic() { + return this; + } + + /** + * Returns ASCII armored text of key + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {ReadableStream} ASCII armor. + */ + armor(config$1 = config) { + return armor(enums.armor.publicKey, this.toPacketList().write(), undefined, undefined, undefined, config$1); + } + } + + /** + * Class that represents an OpenPGP Private key + */ + class PrivateKey extends PublicKey { + /** + * @param {PacketList} packetlist - The packets that form this key + */ + constructor(packetlist) { + super(); + this.packetListToStructure(packetlist, new Set([enums.packet.publicKey, enums.packet.publicSubkey])); + if (!this.keyPacket) { + throw new Error('Invalid key: missing private-key packet'); + } + } + + /** + * Returns true if this is a private key + * @returns {Boolean} + */ + isPrivate() { + return true; + } + + /** + * Returns key as public key (shallow copy) + * @returns {PublicKey} New public Key + */ + toPublic() { + const packetlist = new PacketList(); + const keyPackets = this.toPacketList(); + for (const keyPacket of keyPackets) { + switch (keyPacket.constructor.tag) { + case enums.packet.secretKey: { + const pubKeyPacket = PublicKeyPacket.fromSecretKeyPacket(keyPacket); + packetlist.push(pubKeyPacket); + break; + } + case enums.packet.secretSubkey: { + const pubSubkeyPacket = PublicSubkeyPacket.fromSecretSubkeyPacket(keyPacket); + packetlist.push(pubSubkeyPacket); + break; + } + default: + packetlist.push(keyPacket); + } + } + return new PublicKey(packetlist); + } + + /** + * Returns ASCII armored text of key + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {ReadableStream} ASCII armor. + */ + armor(config$1 = config) { + return armor(enums.armor.privateKey, this.toPacketList().write(), undefined, undefined, undefined, config$1); + } + + /** + * Returns all keys that are available for decryption, matching the keyID when given + * This is useful to retrieve keys for session key decryption + * @param {module:type/keyid~KeyID} keyID, optional + * @param {Date} date, optional + * @param {String} userID, optional + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise>} Array of decryption keys. + * @async + */ + async getDecryptionKeys(keyID, date = new Date(), userID = {}, config$1 = config) { + const primaryKey = this.keyPacket; + const keys = []; + for (let i = 0; i < this.subkeys.length; i++) { + if (!keyID || this.subkeys[i].getKeyID().equals(keyID, true)) { + try { + const dataToVerify = { key: primaryKey, bind: this.subkeys[i].keyPacket }; + const bindingSignature = await getLatestValidSignature(this.subkeys[i].bindingSignatures, primaryKey, enums.signature.subkeyBinding, dataToVerify, date, config$1); + if (isValidDecryptionKeyPacket(bindingSignature, config$1)) { + keys.push(this.subkeys[i]); + } + } catch (e) {} + } + } + + // evaluate primary key + const primaryUser = await this.getPrimaryUser(date, userID, config$1); + if ((!keyID || primaryKey.getKeyID().equals(keyID, true)) && + isValidDecryptionKeyPacket(primaryUser.selfCertification, config$1)) { + keys.push(this); + } + + return keys; + } + + /** + * Returns true if the primary key or any subkey is decrypted. + * A dummy key is considered encrypted. + */ + isDecrypted() { + return this.getKeys().some(({ keyPacket }) => keyPacket.isDecrypted()); + } + + /** + * Check whether the private and public primary key parameters correspond + * Together with verification of binding signatures, this guarantees key integrity + * In case of gnu-dummy primary key, it is enough to validate any signing subkeys + * otherwise all encryption subkeys are validated + * If only gnu-dummy keys are found, we cannot properly validate so we throw an error + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @throws {Error} if validation was not successful and the key cannot be trusted + * @async + */ + async validate(config$1 = config) { + if (!this.isPrivate()) { + throw new Error('Cannot validate a public key'); + } + + let signingKeyPacket; + if (!this.keyPacket.isDummy()) { + signingKeyPacket = this.keyPacket; + } else { + /** + * It is enough to validate any signing keys + * since its binding signatures are also checked + */ + const signingKey = await this.getSigningKey(null, null, undefined, { ...config$1, rejectPublicKeyAlgorithms: new Set(), minRSABits: 0 }); + // This could again be a dummy key + if (signingKey && !signingKey.keyPacket.isDummy()) { + signingKeyPacket = signingKey.keyPacket; + } + } + + if (signingKeyPacket) { + return signingKeyPacket.validate(); + } else { + const keys = this.getKeys(); + const allDummies = keys.map(key => key.keyPacket.isDummy()).every(Boolean); + if (allDummies) { + throw new Error('Cannot validate an all-gnu-dummy key'); + } + + return Promise.all(keys.map(async key => key.keyPacket.validate())); + } + } + + /** + * Clear private key parameters + */ + clearPrivateParams() { + this.getKeys().forEach(({ keyPacket }) => { + if (keyPacket.isDecrypted()) { + keyPacket.clearPrivateParams(); + } + }); + } + + /** + * Revokes the key + * @param {Object} reasonForRevocation - optional, object indicating the reason for revocation + * @param {module:enums.reasonForRevocation} reasonForRevocation.flag optional, flag indicating the reason for revocation + * @param {String} reasonForRevocation.string optional, string explaining the reason for revocation + * @param {Date} date - optional, override the creationtime of the revocation signature + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New key with revocation signature. + * @async + */ + async revoke( + { + flag: reasonForRevocationFlag = enums.reasonForRevocation.noReason, + string: reasonForRevocationString = '' + } = {}, + date = new Date(), + config$1 = config + ) { + if (!this.isPrivate()) { + throw new Error('Need private key for revoking'); + } + const dataToSign = { key: this.keyPacket }; + const key = this.clone(); + key.revocationSignatures.push(await createSignaturePacket(dataToSign, null, this.keyPacket, { + signatureType: enums.signature.keyRevocation, + reasonForRevocationFlag: enums.write(enums.reasonForRevocation, reasonForRevocationFlag), + reasonForRevocationString + }, date, undefined, undefined, undefined, config$1)); + return key; + } + + + /** + * Generates a new OpenPGP subkey, and returns a clone of the Key object with the new subkey added. + * Supports RSA and ECC keys. Defaults to the algorithm and bit size/curve of the primary key. DSA primary keys default to RSA subkeys. + * @param {ecc|rsa} options.type The subkey algorithm: ECC or RSA + * @param {String} options.curve (optional) Elliptic curve for ECC keys + * @param {Integer} options.rsaBits (optional) Number of bits for RSA subkeys + * @param {Number} options.keyExpirationTime (optional) Number of seconds from the key creation time after which the key expires + * @param {Date} options.date (optional) Override the creation date of the key and the key signatures + * @param {Boolean} options.sign (optional) Indicates whether the subkey should sign rather than encrypt. Defaults to false + * @param {Object} options.config (optional) custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} + * @async + */ + async addSubkey(options = {}) { + const config$1 = { ...config, ...options.config }; + if (options.passphrase) { + throw new Error('Subkey could not be encrypted here, please encrypt whole key'); + } + if (options.rsaBits < config$1.minRSABits) { + throw new Error(`rsaBits should be at least ${config$1.minRSABits}, got: ${options.rsaBits}`); + } + const secretKeyPacket = this.keyPacket; + if (secretKeyPacket.isDummy()) { + throw new Error('Cannot add subkey to gnu-dummy primary key'); + } + if (!secretKeyPacket.isDecrypted()) { + throw new Error('Key is not decrypted'); + } + const defaultOptions = secretKeyPacket.getAlgorithmInfo(); + defaultOptions.type = defaultOptions.curve ? 'ecc' : 'rsa'; // DSA keys default to RSA + defaultOptions.rsaBits = defaultOptions.bits || 4096; + defaultOptions.curve = defaultOptions.curve || 'curve25519'; + options = sanitizeKeyOptions(options, defaultOptions); + const keyPacket = await generateSecretSubkey(options); + checkKeyRequirements(keyPacket, config$1); + const bindingSignature = await createBindingSignature(keyPacket, secretKeyPacket, options, config$1); + const packetList = this.toPacketList(); + packetList.push(keyPacket, bindingSignature); + return new PrivateKey(packetList); + } + } + + // OpenPGP.js - An OpenPGP implementation in javascript + + // A Key can contain the following packets + const allowedKeyPackets = /*#__PURE__*/ util.constructAllowedPackets([ + PublicKeyPacket, + PublicSubkeyPacket, + SecretKeyPacket, + SecretSubkeyPacket, + UserIDPacket, + UserAttributePacket, + SignaturePacket + ]); + + /** + * Creates a PublicKey or PrivateKey depending on the packetlist in input + * @param {PacketList} - packets to parse + * @return {Key} parsed key + * @throws if no key packet was found + */ + function createKey(packetlist) { + for (const packet of packetlist) { + switch (packet.constructor.tag) { + case enums.packet.secretKey: + return new PrivateKey(packetlist); + case enums.packet.publicKey: + return new PublicKey(packetlist); + } + } + throw new Error('No key packet found'); + } + + + /** + * Generates a new OpenPGP key. Supports RSA and ECC keys. + * By default, primary and subkeys will be of same type. + * @param {ecc|rsa} options.type The primary key algorithm type: ECC or RSA + * @param {String} options.curve Elliptic curve for ECC keys + * @param {Integer} options.rsaBits Number of bits for RSA keys + * @param {Array} options.userIDs User IDs as strings or objects: 'Jo Doe ' or { name:'Jo Doe', email:'info@jo.com' } + * @param {String} options.passphrase Passphrase used to encrypt the resulting private key + * @param {Number} options.keyExpirationTime (optional) Number of seconds from the key creation time after which the key expires + * @param {Date} options.date Creation date of the key and the key signatures + * @param {Object} config - Full configuration + * @param {Array} options.subkeys (optional) options for each subkey, default to main key options. e.g. [{sign: true, passphrase: '123'}] + * sign parameter defaults to false, and indicates whether the subkey should sign rather than encrypt + * @returns {Promise<{{ key: PrivateKey, revocationCertificate: String }}>} + * @async + * @static + * @private + */ + async function generate$2(options, config) { + options.sign = true; // primary key is always a signing key + options = sanitizeKeyOptions(options); + options.subkeys = options.subkeys.map((subkey, index) => sanitizeKeyOptions(options.subkeys[index], options)); + let promises = [generateSecretKey(options, config)]; + promises = promises.concat(options.subkeys.map(options => generateSecretSubkey(options, config))); + const packets = await Promise.all(promises); + + const key = await wrapKeyObject(packets[0], packets.slice(1), options, config); + const revocationCertificate = await key.getRevocationCertificate(options.date, config); + key.revocationSignatures = []; + return { key, revocationCertificate }; + } + + /** + * Reformats and signs an OpenPGP key with a given User ID. Currently only supports RSA keys. + * @param {PrivateKey} options.privateKey The private key to reformat + * @param {Array} options.userIDs User IDs as strings or objects: 'Jo Doe ' or { name:'Jo Doe', email:'info@jo.com' } + * @param {String} options.passphrase Passphrase used to encrypt the resulting private key + * @param {Number} options.keyExpirationTime Number of seconds from the key creation time after which the key expires + * @param {Date} options.date Override the creation date of the key signatures + * @param {Array} options.subkeys (optional) options for each subkey, default to main key options. e.g. [{sign: true, passphrase: '123'}] + * @param {Object} config - Full configuration + * + * @returns {Promise<{{ key: PrivateKey, revocationCertificate: String }}>} + * @async + * @static + * @private + */ + async function reformat(options, config) { + options = sanitize(options); + const { privateKey } = options; + + if (!privateKey.isPrivate()) { + throw new Error('Cannot reformat a public key'); + } + + if (privateKey.keyPacket.isDummy()) { + throw new Error('Cannot reformat a gnu-dummy primary key'); + } + + const isDecrypted = privateKey.getKeys().every(({ keyPacket }) => keyPacket.isDecrypted()); + if (!isDecrypted) { + throw new Error('Key is not decrypted'); + } + + const secretKeyPacket = privateKey.keyPacket; + + if (!options.subkeys) { + options.subkeys = await Promise.all(privateKey.subkeys.map(async subkey => { + const secretSubkeyPacket = subkey.keyPacket; + const dataToVerify = { key: secretKeyPacket, bind: secretSubkeyPacket }; + const bindingSignature = await ( + getLatestValidSignature(subkey.bindingSignatures, secretKeyPacket, enums.signature.subkeyBinding, dataToVerify, null, config) + ).catch(() => ({})); + return { + sign: bindingSignature.keyFlags && (bindingSignature.keyFlags[0] & enums.keyFlags.signData) + }; + })); + } + + const secretSubkeyPackets = privateKey.subkeys.map(subkey => subkey.keyPacket); + if (options.subkeys.length !== secretSubkeyPackets.length) { + throw new Error('Number of subkey options does not match number of subkeys'); + } + + options.subkeys = options.subkeys.map(subkeyOptions => sanitize(subkeyOptions, options)); + + const key = await wrapKeyObject(secretKeyPacket, secretSubkeyPackets, options, config); + const revocationCertificate = await key.getRevocationCertificate(options.date, config); + key.revocationSignatures = []; + return { key, revocationCertificate }; + + function sanitize(options, subkeyDefaults = {}) { + options.keyExpirationTime = options.keyExpirationTime || subkeyDefaults.keyExpirationTime; + options.passphrase = util.isString(options.passphrase) ? options.passphrase : subkeyDefaults.passphrase; + options.date = options.date || subkeyDefaults.date; + + return options; + } + } + + /** + * Construct PrivateKey object from the given key packets, add certification signatures and set passphrase protection + * The new key includes a revocation certificate that must be removed before returning the key, otherwise the key is considered revoked. + * @param {SecretKeyPacket} secretKeyPacket + * @param {SecretSubkeyPacket} secretSubkeyPackets + * @param {Object} options + * @param {Object} config - Full configuration + * @returns {PrivateKey} + */ + async function wrapKeyObject(secretKeyPacket, secretSubkeyPackets, options, config) { + // set passphrase protection + if (options.passphrase) { + await secretKeyPacket.encrypt(options.passphrase, config); + } + + await Promise.all(secretSubkeyPackets.map(async function(secretSubkeyPacket, index) { + const subkeyPassphrase = options.subkeys[index].passphrase; + if (subkeyPassphrase) { + await secretSubkeyPacket.encrypt(subkeyPassphrase, config); + } + })); + + const packetlist = new PacketList(); + packetlist.push(secretKeyPacket); + + await Promise.all(options.userIDs.map(async function(userID, index) { + function createPreferredAlgos(algos, preferredAlgo) { + return [preferredAlgo, ...algos.filter(algo => algo !== preferredAlgo)]; + } + + const userIDPacket = UserIDPacket.fromObject(userID); + const dataToSign = {}; + dataToSign.userID = userIDPacket; + dataToSign.key = secretKeyPacket; + const signaturePacket = new SignaturePacket(); + signaturePacket.signatureType = enums.signature.certGeneric; + signaturePacket.publicKeyAlgorithm = secretKeyPacket.algorithm; + signaturePacket.hashAlgorithm = await getPreferredHashAlgo$1(null, secretKeyPacket, undefined, undefined, config); + signaturePacket.keyFlags = [enums.keyFlags.certifyKeys | enums.keyFlags.signData]; + signaturePacket.preferredSymmetricAlgorithms = createPreferredAlgos([ + // prefer aes256, aes128, then aes192 (no WebCrypto support: https://www.chromium.org/blink/webcrypto#TOC-AES-support) + enums.symmetric.aes256, + enums.symmetric.aes128, + enums.symmetric.aes192 + ], config.preferredSymmetricAlgorithm); + if (config.aeadProtect) { + signaturePacket.preferredAEADAlgorithms = createPreferredAlgos([ + enums.aead.eax, + enums.aead.ocb + ], config.preferredAEADAlgorithm); + } + signaturePacket.preferredHashAlgorithms = createPreferredAlgos([ + // prefer fast asm.js implementations (SHA-256) + enums.hash.sha256, + enums.hash.sha512 + ], config.preferredHashAlgorithm); + signaturePacket.preferredCompressionAlgorithms = createPreferredAlgos([ + enums.compression.zlib, + enums.compression.zip, + enums.compression.uncompressed + ], config.preferredCompressionAlgorithm); + if (index === 0) { + signaturePacket.isPrimaryUserID = true; + } + // integrity protection always enabled + signaturePacket.features = [0]; + signaturePacket.features[0] |= enums.features.modificationDetection; + if (config.aeadProtect) { + signaturePacket.features[0] |= enums.features.aead; + } + if (config.v5Keys) { + signaturePacket.features[0] |= enums.features.v5Keys; + } + if (options.keyExpirationTime > 0) { + signaturePacket.keyExpirationTime = options.keyExpirationTime; + signaturePacket.keyNeverExpires = false; + } + await signaturePacket.sign(secretKeyPacket, dataToSign, options.date); + + return { userIDPacket, signaturePacket }; + })).then(list => { + list.forEach(({ userIDPacket, signaturePacket }) => { + packetlist.push(userIDPacket); + packetlist.push(signaturePacket); + }); + }); + + await Promise.all(secretSubkeyPackets.map(async function(secretSubkeyPacket, index) { + const subkeyOptions = options.subkeys[index]; + const subkeySignaturePacket = await createBindingSignature(secretSubkeyPacket, secretKeyPacket, subkeyOptions, config); + return { secretSubkeyPacket, subkeySignaturePacket }; + })).then(packets => { + packets.forEach(({ secretSubkeyPacket, subkeySignaturePacket }) => { + packetlist.push(secretSubkeyPacket); + packetlist.push(subkeySignaturePacket); + }); + }); + + // Add revocation signature packet for creating a revocation certificate. + // This packet should be removed before returning the key. + const dataToSign = { key: secretKeyPacket }; + packetlist.push(await createSignaturePacket(dataToSign, null, secretKeyPacket, { + signatureType: enums.signature.keyRevocation, + reasonForRevocationFlag: enums.reasonForRevocation.noReason, + reasonForRevocationString: '' + }, options.date, undefined, undefined, undefined, config)); + + if (options.passphrase) { + secretKeyPacket.clearPrivateParams(); + } + + await Promise.all(secretSubkeyPackets.map(async function(secretSubkeyPacket, index) { + const subkeyPassphrase = options.subkeys[index].passphrase; + if (subkeyPassphrase) { + secretSubkeyPacket.clearPrivateParams(); + } + })); + + return new PrivateKey(packetlist); + } + + /** + * Reads an (optionally armored) OpenPGP key and returns a key object + * @param {Object} options + * @param {String} [options.armoredKey] - Armored key to be parsed + * @param {Uint8Array} [options.binaryKey] - Binary key to be parsed + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} Key object. + * @async + * @static + */ + async function readKey({ armoredKey, binaryKey, config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; + if (!armoredKey && !binaryKey) { + throw new Error('readKey: must pass options object containing `armoredKey` or `binaryKey`'); + } + if (armoredKey && !util.isString(armoredKey)) { + throw new Error('readKey: options.armoredKey must be a string'); + } + if (binaryKey && !util.isUint8Array(binaryKey)) { + throw new Error('readKey: options.binaryKey must be a Uint8Array'); + } + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + let input; + if (armoredKey) { + const { type, data } = await unarmor(armoredKey, config$1); + if (!(type === enums.armor.publicKey || type === enums.armor.privateKey)) { + throw new Error('Armored text not of type key'); + } + input = data; + } else { + input = binaryKey; + } + const packetlist = await PacketList.fromBinary(input, allowedKeyPackets, config$1); + return createKey(packetlist); + } + + /** + * Reads an (optionally armored) OpenPGP private key and returns a PrivateKey object + * @param {Object} options + * @param {String} [options.armoredKey] - Armored key to be parsed + * @param {Uint8Array} [options.binaryKey] - Binary key to be parsed + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} Key object. + * @async + * @static + */ + async function readPrivateKey({ armoredKey, binaryKey, config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; + if (!armoredKey && !binaryKey) { + throw new Error('readPrivateKey: must pass options object containing `armoredKey` or `binaryKey`'); + } + if (armoredKey && !util.isString(armoredKey)) { + throw new Error('readPrivateKey: options.armoredKey must be a string'); + } + if (binaryKey && !util.isUint8Array(binaryKey)) { + throw new Error('readPrivateKey: options.binaryKey must be a Uint8Array'); + } + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + let input; + if (armoredKey) { + const { type, data } = await unarmor(armoredKey, config$1); + if (!(type === enums.armor.privateKey)) { + throw new Error('Armored text not of type private key'); + } + input = data; + } else { + input = binaryKey; + } + const packetlist = await PacketList.fromBinary(input, allowedKeyPackets, config$1); + return new PrivateKey(packetlist); + } + + /** + * Reads an (optionally armored) OpenPGP key block and returns a list of key objects + * @param {Object} options + * @param {String} [options.armoredKeys] - Armored keys to be parsed + * @param {Uint8Array} [options.binaryKeys] - Binary keys to be parsed + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise>} Key objects. + * @async + * @static + */ + async function readKeys({ armoredKeys, binaryKeys, config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; + let input = armoredKeys || binaryKeys; + if (!input) { + throw new Error('readKeys: must pass options object containing `armoredKeys` or `binaryKeys`'); + } + if (armoredKeys && !util.isString(armoredKeys)) { + throw new Error('readKeys: options.armoredKeys must be a string'); + } + if (binaryKeys && !util.isUint8Array(binaryKeys)) { + throw new Error('readKeys: options.binaryKeys must be a Uint8Array'); + } + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (armoredKeys) { + const { type, data } = await unarmor(armoredKeys, config$1); + if (type !== enums.armor.publicKey && type !== enums.armor.privateKey) { + throw new Error('Armored text not of type key'); + } + input = data; + } + const keys = []; + const packetlist = await PacketList.fromBinary(input, allowedKeyPackets, config$1); + const keyIndex = packetlist.indexOfTag(enums.packet.publicKey, enums.packet.secretKey); + if (keyIndex.length === 0) { + throw new Error('No key packet found'); + } + for (let i = 0; i < keyIndex.length; i++) { + const oneKeyList = packetlist.slice(keyIndex[i], keyIndex[i + 1]); + const newKey = createKey(oneKeyList); + keys.push(newKey); + } + return keys; + } + + /** + * Reads an (optionally armored) OpenPGP private key block and returns a list of PrivateKey objects + * @param {Object} options + * @param {String} [options.armoredKeys] - Armored keys to be parsed + * @param {Uint8Array} [options.binaryKeys] - Binary keys to be parsed + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise>} Key objects. + * @async + * @static + */ + async function readPrivateKeys({ armoredKeys, binaryKeys, config: config$1 }) { + config$1 = { ...config, ...config$1 }; + let input = armoredKeys || binaryKeys; + if (!input) { + throw new Error('readPrivateKeys: must pass options object containing `armoredKeys` or `binaryKeys`'); + } + if (armoredKeys && !util.isString(armoredKeys)) { + throw new Error('readPrivateKeys: options.armoredKeys must be a string'); + } + if (binaryKeys && !util.isUint8Array(binaryKeys)) { + throw new Error('readPrivateKeys: options.binaryKeys must be a Uint8Array'); + } + if (armoredKeys) { + const { type, data } = await unarmor(armoredKeys, config$1); + if (type !== enums.armor.privateKey) { + throw new Error('Armored text not of type private key'); + } + input = data; + } + const keys = []; + const packetlist = await PacketList.fromBinary(input, allowedKeyPackets, config$1); + const keyIndex = packetlist.indexOfTag(enums.packet.secretKey); + if (keyIndex.length === 0) { + throw new Error('No secret key packet found'); + } + for (let i = 0; i < keyIndex.length; i++) { + const oneKeyList = packetlist.slice(keyIndex[i], keyIndex[i + 1]); + const newKey = new PrivateKey(oneKeyList); + keys.push(newKey); + } + return keys; + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + // A Message can contain the following packets + const allowedMessagePackets = /*#__PURE__*/ util.constructAllowedPackets([ + LiteralDataPacket, + CompressedDataPacket, + AEADEncryptedDataPacket, + SymEncryptedIntegrityProtectedDataPacket, + SymmetricallyEncryptedDataPacket, + PublicKeyEncryptedSessionKeyPacket, + SymEncryptedSessionKeyPacket, + OnePassSignaturePacket, + SignaturePacket + ]); + // A SKESK packet can contain the following packets + const allowedSymSessionKeyPackets = /*#__PURE__*/ util.constructAllowedPackets([SymEncryptedSessionKeyPacket]); + // A detached signature can contain the following packets + const allowedDetachedSignaturePackets = /*#__PURE__*/ util.constructAllowedPackets([SignaturePacket]); + + /** + * Class that represents an OpenPGP message. + * Can be an encrypted message, signed message, compressed message or literal message + * See {@link https://tools.ietf.org/html/rfc4880#section-11.3} + */ + class Message { + /** + * @param {PacketList} packetlist - The packets that form this message + */ + constructor(packetlist) { + this.packets = packetlist || new PacketList(); + } + + /** + * Returns the key IDs of the keys to which the session key is encrypted + * @returns {Array} Array of keyID objects. + */ + getEncryptionKeyIDs() { + const keyIDs = []; + const pkESKeyPacketlist = this.packets.filterByTag(enums.packet.publicKeyEncryptedSessionKey); + pkESKeyPacketlist.forEach(function(packet) { + keyIDs.push(packet.publicKeyID); + }); + return keyIDs; + } + + /** + * Returns the key IDs of the keys that signed the message + * @returns {Array} Array of keyID objects. + */ + getSigningKeyIDs() { + const msg = this.unwrapCompressed(); + // search for one pass signatures + const onePassSigList = msg.packets.filterByTag(enums.packet.onePassSignature); + if (onePassSigList.length > 0) { + return onePassSigList.map(packet => packet.issuerKeyID); + } + // if nothing found look for signature packets + const signatureList = msg.packets.filterByTag(enums.packet.signature); + return signatureList.map(packet => packet.issuerKeyID); + } + + /** + * Decrypt the message. Either a private key, a session key, or a password must be specified. + * @param {Array} [decryptionKeys] - Private keys with decrypted secret data + * @param {Array} [passwords] - Passwords used to decrypt + * @param {Array} [sessionKeys] - Session keys in the form: { data:Uint8Array, algorithm:String, [aeadAlgorithm:String] } + * @param {Date} [date] - Use the given date for key verification instead of the current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New message with decrypted content. + * @async + */ + async decrypt(decryptionKeys, passwords, sessionKeys, date = new Date(), config$1 = config) { + const sessionKeyObjects = sessionKeys || await this.decryptSessionKeys(decryptionKeys, passwords, date, config$1); + + const symEncryptedPacketlist = this.packets.filterByTag( + enums.packet.symmetricallyEncryptedData, + enums.packet.symEncryptedIntegrityProtectedData, + enums.packet.aeadEncryptedData + ); + + if (symEncryptedPacketlist.length === 0) { + throw new Error('No encrypted data found'); + } + + const symEncryptedPacket = symEncryptedPacketlist[0]; + let exception = null; + const decryptedPromise = Promise.all(sessionKeyObjects.map(async ({ algorithm: algorithmName, data }) => { + if (!util.isUint8Array(data) || !util.isString(algorithmName)) { + throw new Error('Invalid session key for decryption.'); + } + + try { + const algo = enums.write(enums.symmetric, algorithmName); + await symEncryptedPacket.decrypt(algo, data, config$1); + } catch (e) { + util.printDebugError(e); + exception = e; + } + })); + // We don't await stream.cancel here because it only returns when the other copy is canceled too. + cancel(symEncryptedPacket.encrypted); // Don't keep copy of encrypted data in memory. + symEncryptedPacket.encrypted = null; + await decryptedPromise; + + if (!symEncryptedPacket.packets || !symEncryptedPacket.packets.length) { + throw exception || new Error('Decryption failed.'); + } + + const resultMsg = new Message(symEncryptedPacket.packets); + symEncryptedPacket.packets = new PacketList(); // remove packets after decryption + + return resultMsg; + } + + /** + * Decrypt encrypted session keys either with private keys or passwords. + * @param {Array} [decryptionKeys] - Private keys with decrypted secret data + * @param {Array} [passwords] - Passwords used to decrypt + * @param {Date} [date] - Use the given date for key verification, instead of current time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise>} array of object with potential sessionKey, algorithm pairs + * @async + */ + async decryptSessionKeys(decryptionKeys, passwords, date = new Date(), config$1 = config) { + let decryptedSessionKeyPackets = []; + + let exception; + if (passwords) { + const skeskPackets = this.packets.filterByTag(enums.packet.symEncryptedSessionKey); + if (skeskPackets.length === 0) { + throw new Error('No symmetrically encrypted session key packet found.'); + } + await Promise.all(passwords.map(async function(password, i) { + let packets; + if (i) { + packets = await PacketList.fromBinary(skeskPackets.write(), allowedSymSessionKeyPackets, config$1); + } else { + packets = skeskPackets; + } + await Promise.all(packets.map(async function(skeskPacket) { + try { + await skeskPacket.decrypt(password); + decryptedSessionKeyPackets.push(skeskPacket); + } catch (err) { + util.printDebugError(err); + } + })); + })); + } else if (decryptionKeys) { + const pkeskPackets = this.packets.filterByTag(enums.packet.publicKeyEncryptedSessionKey); + if (pkeskPackets.length === 0) { + throw new Error('No public key encrypted session key packet found.'); + } + await Promise.all(pkeskPackets.map(async function(pkeskPacket) { + await Promise.all(decryptionKeys.map(async function(decryptionKey) { + let algos = [ + enums.symmetric.aes256, // Old OpenPGP.js default fallback + enums.symmetric.aes128, // RFC4880bis fallback + enums.symmetric.tripledes, // RFC4880 fallback + enums.symmetric.cast5 // Golang OpenPGP fallback + ]; + try { + const primaryUser = await decryptionKey.getPrimaryUser(date, undefined, config$1); // TODO: Pass userID from somewhere. + if (primaryUser.selfCertification.preferredSymmetricAlgorithms) { + algos = algos.concat(primaryUser.selfCertification.preferredSymmetricAlgorithms); + } + } catch (e) {} + + // do not check key expiration to allow decryption of old messages + const decryptionKeyPackets = (await decryptionKey.getDecryptionKeys(pkeskPacket.publicKeyID, null, undefined, config$1)).map(key => key.keyPacket); + await Promise.all(decryptionKeyPackets.map(async function(decryptionKeyPacket) { + if (!decryptionKeyPacket || decryptionKeyPacket.isDummy()) { + return; + } + if (!decryptionKeyPacket.isDecrypted()) { + throw new Error('Decryption key is not decrypted.'); + } + + // To hinder CCA attacks against PKCS1, we carry out a constant-time decryption flow if the `constantTimePKCS1Decryption` config option is set. + const doConstantTimeDecryption = config$1.constantTimePKCS1Decryption && ( + pkeskPacket.publicKeyAlgorithm === enums.publicKey.rsaEncrypt || + pkeskPacket.publicKeyAlgorithm === enums.publicKey.rsaEncryptSign || + pkeskPacket.publicKeyAlgorithm === enums.publicKey.rsaSign || + pkeskPacket.publicKeyAlgorithm === enums.publicKey.elgamal + ); + + if (doConstantTimeDecryption) { + // The goal is to not reveal whether PKESK decryption (specifically the PKCS1 decoding step) failed, hence, we always proceed to decrypt the message, + // either with the successfully decrypted session key, or with a randomly generated one. + // Since the SEIP/AEAD's symmetric algorithm and key size are stored in the encrypted portion of the PKESK, and the execution flow cannot depend on + // the decrypted payload, we always assume the message to be encrypted with one of the symmetric algorithms specified in `config.constantTimePKCS1DecryptionSupportedSymmetricAlgorithms`: + // - If the PKESK decryption succeeds, and the session key cipher is in the supported set, then we try to decrypt the data with the decrypted session key as well as with the + // randomly generated keys of the remaining key types. + // - If the PKESK decryptions fails, or if it succeeds but support for the cipher is not enabled, then we discard the session key and try to decrypt the data using only the randomly + // generated session keys. + // NB: as a result, if the data is encrypted with a non-suported cipher, decryption will always fail. + + const serialisedPKESK = pkeskPacket.write(); // make copies to be able to decrypt the PKESK packet multiple times + await Promise.all(Array.from(config$1.constantTimePKCS1DecryptionSupportedSymmetricAlgorithms).map(async sessionKeyAlgorithm => { + const pkeskPacketCopy = new PublicKeyEncryptedSessionKeyPacket(); + pkeskPacketCopy.read(serialisedPKESK); + const randomSessionKey = { + sessionKeyAlgorithm, + sessionKey: mod.generateSessionKey(sessionKeyAlgorithm) + }; + try { + await pkeskPacketCopy.decrypt(decryptionKeyPacket, randomSessionKey); + decryptedSessionKeyPackets.push(pkeskPacketCopy); + } catch (err) { + // `decrypt` can still throw some non-security-sensitive errors + util.printDebugError(err); + exception = err; + } + })); + + } else { + try { + await pkeskPacket.decrypt(decryptionKeyPacket); + if (!algos.includes(enums.write(enums.symmetric, pkeskPacket.sessionKeyAlgorithm))) { + throw new Error('A non-preferred symmetric algorithm was used.'); + } + decryptedSessionKeyPackets.push(pkeskPacket); + } catch (err) { + util.printDebugError(err); + exception = err; + } + } + })); + })); + cancel(pkeskPacket.encrypted); // Don't keep copy of encrypted data in memory. + pkeskPacket.encrypted = null; + })); + } else { + throw new Error('No key or password specified.'); + } + + if (decryptedSessionKeyPackets.length > 0) { + // Return only unique session keys + if (decryptedSessionKeyPackets.length > 1) { + const seen = new Set(); + decryptedSessionKeyPackets = decryptedSessionKeyPackets.filter(item => { + const k = item.sessionKeyAlgorithm + util.uint8ArrayToString(item.sessionKey); + if (seen.has(k)) { + return false; + } + seen.add(k); + return true; + }); + } + + return decryptedSessionKeyPackets.map(packet => ({ + data: packet.sessionKey, + algorithm: enums.read(enums.symmetric, packet.sessionKeyAlgorithm) + })); + } + throw exception || new Error('Session key decryption failed.'); + } + + /** + * Get literal data that is the body of the message + * @returns {(Uint8Array|null)} Literal body of the message as Uint8Array. + */ + getLiteralData() { + const msg = this.unwrapCompressed(); + const literal = msg.packets.findPacket(enums.packet.literalData); + return (literal && literal.getBytes()) || null; + } + + /** + * Get filename from literal data packet + * @returns {(String|null)} Filename of literal data packet as string. + */ + getFilename() { + const msg = this.unwrapCompressed(); + const literal = msg.packets.findPacket(enums.packet.literalData); + return (literal && literal.getFilename()) || null; + } + + /** + * Get literal data as text + * @returns {(String|null)} Literal body of the message interpreted as text. + */ + getText() { + const msg = this.unwrapCompressed(); + const literal = msg.packets.findPacket(enums.packet.literalData); + if (literal) { + return literal.getText(); + } + return null; + } + + /** + * Generate a new session key object, taking the algorithm preferences of the passed encryption keys into account, if any. + * @param {Array} [encryptionKeys] - Public key(s) to select algorithm preferences for + * @param {Date} [date] - Date to select algorithm preferences at + * @param {Array} [userIDs] - User IDs to select algorithm preferences for + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise<{ data: Uint8Array, algorithm: String, aeadAlgorithm: undefined|String }>} Object with session key data and algorithms. + * @async + */ + static async generateSessionKey(encryptionKeys = [], date = new Date(), userIDs = [], config$1 = config) { + const algo = await getPreferredAlgo('symmetric', encryptionKeys, date, userIDs, config$1); + const algorithmName = enums.read(enums.symmetric, algo); + const aeadAlgorithmName = config$1.aeadProtect && await isAEADSupported(encryptionKeys, date, userIDs, config$1) ? + enums.read(enums.aead, await getPreferredAlgo('aead', encryptionKeys, date, userIDs, config$1)) : + undefined; + + const sessionKeyData = mod.generateSessionKey(algo); + return { data: sessionKeyData, algorithm: algorithmName, aeadAlgorithm: aeadAlgorithmName }; + } + + /** + * Encrypt the message either with public keys, passwords, or both at once. + * @param {Array} [encryptionKeys] - Public key(s) for message encryption + * @param {Array} [passwords] - Password(s) for message encryption + * @param {Object} [sessionKey] - Session key in the form: { data:Uint8Array, algorithm:String, [aeadAlgorithm:String] } + * @param {Boolean} [wildcard] - Use a key ID of 0 instead of the public key IDs + * @param {Array} [encryptionKeyIDs] - Array of key IDs to use for encryption. Each encryptionKeyIDs[i] corresponds to keys[i] + * @param {Date} [date] - Override the creation date of the literal package + * @param {Array} [userIDs] - User IDs to encrypt for, e.g. [{ name:'Robert Receiver', email:'robert@openpgp.org' }] + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New message with encrypted content. + * @async + */ + async encrypt(encryptionKeys, passwords, sessionKey, wildcard = false, encryptionKeyIDs = [], date = new Date(), userIDs = [], config$1 = config) { + if (sessionKey) { + if (!util.isUint8Array(sessionKey.data) || !util.isString(sessionKey.algorithm)) { + throw new Error('Invalid session key for encryption.'); + } + } else if (encryptionKeys && encryptionKeys.length) { + sessionKey = await Message.generateSessionKey(encryptionKeys, date, userIDs, config$1); + } else if (passwords && passwords.length) { + sessionKey = await Message.generateSessionKey(undefined, undefined, undefined, config$1); + } else { + throw new Error('No keys, passwords, or session key provided.'); + } + + const { data: sessionKeyData, algorithm: algorithmName, aeadAlgorithm: aeadAlgorithmName } = sessionKey; + + const msg = await Message.encryptSessionKey(sessionKeyData, algorithmName, aeadAlgorithmName, encryptionKeys, passwords, wildcard, encryptionKeyIDs, date, userIDs, config$1); + + let symEncryptedPacket; + if (aeadAlgorithmName) { + symEncryptedPacket = new AEADEncryptedDataPacket(); + symEncryptedPacket.aeadAlgorithm = enums.write(enums.aead, aeadAlgorithmName); + } else { + symEncryptedPacket = new SymEncryptedIntegrityProtectedDataPacket(); + } + symEncryptedPacket.packets = this.packets; + + const algorithm = enums.write(enums.symmetric, algorithmName); + await symEncryptedPacket.encrypt(algorithm, sessionKeyData, config$1); + + msg.packets.push(symEncryptedPacket); + symEncryptedPacket.packets = new PacketList(); // remove packets after encryption + return msg; + } + + /** + * Encrypt a session key either with public keys, passwords, or both at once. + * @param {Uint8Array} sessionKey - session key for encryption + * @param {String} algorithmName - session key algorithm + * @param {String} [aeadAlgorithmName] - AEAD algorithm, e.g. 'eax' or 'ocb' + * @param {Array} [encryptionKeys] - Public key(s) for message encryption + * @param {Array} [passwords] - For message encryption + * @param {Boolean} [wildcard] - Use a key ID of 0 instead of the public key IDs + * @param {Array} [encryptionKeyIDs] - Array of key IDs to use for encryption. Each encryptionKeyIDs[i] corresponds to encryptionKeys[i] + * @param {Date} [date] - Override the date + * @param {Array} [userIDs] - User IDs to encrypt for, e.g. [{ name:'Robert Receiver', email:'robert@openpgp.org' }] + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New message with encrypted content. + * @async + */ + static async encryptSessionKey(sessionKey, algorithmName, aeadAlgorithmName, encryptionKeys, passwords, wildcard = false, encryptionKeyIDs = [], date = new Date(), userIDs = [], config$1 = config) { + const packetlist = new PacketList(); + const algorithm = enums.write(enums.symmetric, algorithmName); + const aeadAlgorithm = aeadAlgorithmName && enums.write(enums.aead, aeadAlgorithmName); + + if (encryptionKeys) { + const results = await Promise.all(encryptionKeys.map(async function(primaryKey, i) { + const encryptionKey = await primaryKey.getEncryptionKey(encryptionKeyIDs[i], date, userIDs, config$1); + const pkESKeyPacket = new PublicKeyEncryptedSessionKeyPacket(); + pkESKeyPacket.publicKeyID = wildcard ? KeyID.wildcard() : encryptionKey.getKeyID(); + pkESKeyPacket.publicKeyAlgorithm = encryptionKey.keyPacket.algorithm; + pkESKeyPacket.sessionKey = sessionKey; + pkESKeyPacket.sessionKeyAlgorithm = algorithm; + await pkESKeyPacket.encrypt(encryptionKey.keyPacket); + delete pkESKeyPacket.sessionKey; // delete plaintext session key after encryption + return pkESKeyPacket; + })); + packetlist.push(...results); + } + if (passwords) { + const testDecrypt = async function(keyPacket, password) { + try { + await keyPacket.decrypt(password); + return 1; + } catch (e) { + return 0; + } + }; + + const sum = (accumulator, currentValue) => accumulator + currentValue; + + const encryptPassword = async function(sessionKey, algorithm, aeadAlgorithm, password) { + const symEncryptedSessionKeyPacket = new SymEncryptedSessionKeyPacket(config$1); + symEncryptedSessionKeyPacket.sessionKey = sessionKey; + symEncryptedSessionKeyPacket.sessionKeyAlgorithm = algorithm; + if (aeadAlgorithm) { + symEncryptedSessionKeyPacket.aeadAlgorithm = aeadAlgorithm; + } + await symEncryptedSessionKeyPacket.encrypt(password, config$1); + + if (config$1.passwordCollisionCheck) { + const results = await Promise.all(passwords.map(pwd => testDecrypt(symEncryptedSessionKeyPacket, pwd))); + if (results.reduce(sum) !== 1) { + return encryptPassword(sessionKey, algorithm, password); + } + } + + delete symEncryptedSessionKeyPacket.sessionKey; // delete plaintext session key after encryption + return symEncryptedSessionKeyPacket; + }; + + const results = await Promise.all(passwords.map(pwd => encryptPassword(sessionKey, algorithm, aeadAlgorithm, pwd))); + packetlist.push(...results); + } + + return new Message(packetlist); + } + + /** + * Sign the message (the literal data packet of the message) + * @param {Array} signingKeys - private keys with decrypted secret key data for signing + * @param {Signature} [signature] - Any existing detached signature to add to the message + * @param {Array} [signingKeyIDs] - Array of key IDs to use for signing. Each signingKeyIDs[i] corresponds to signingKeys[i] + * @param {Date} [date] - Override the creation time of the signature + * @param {Array} [userIDs] - User IDs to sign with, e.g. [{ name:'Steve Sender', email:'steve@openpgp.org' }] + * @param {Array} [notations] - Notation Data to add to the signatures, e.g. [{ name: 'test@example.org', value: new TextEncoder().encode('test'), humanReadable: true, critical: false }] + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New message with signed content. + * @async + */ + async sign(signingKeys = [], signature = null, signingKeyIDs = [], date = new Date(), userIDs = [], notations = [], config$1 = config) { + const packetlist = new PacketList(); + + const literalDataPacket = this.packets.findPacket(enums.packet.literalData); + if (!literalDataPacket) { + throw new Error('No literal data packet to sign.'); + } + + let i; + let existingSigPacketlist; + // If data packet was created from Uint8Array, use binary, otherwise use text + const signatureType = literalDataPacket.text === null ? + enums.signature.binary : enums.signature.text; + + if (signature) { + existingSigPacketlist = signature.packets.filterByTag(enums.packet.signature); + for (i = existingSigPacketlist.length - 1; i >= 0; i--) { + const signaturePacket = existingSigPacketlist[i]; + const onePassSig = new OnePassSignaturePacket(); + onePassSig.signatureType = signaturePacket.signatureType; + onePassSig.hashAlgorithm = signaturePacket.hashAlgorithm; + onePassSig.publicKeyAlgorithm = signaturePacket.publicKeyAlgorithm; + onePassSig.issuerKeyID = signaturePacket.issuerKeyID; + if (!signingKeys.length && i === 0) { + onePassSig.flags = 1; + } + packetlist.push(onePassSig); + } + } + + await Promise.all(Array.from(signingKeys).reverse().map(async function (primaryKey, i) { + if (!primaryKey.isPrivate()) { + throw new Error('Need private key for signing'); + } + const signingKeyID = signingKeyIDs[signingKeys.length - 1 - i]; + const signingKey = await primaryKey.getSigningKey(signingKeyID, date, userIDs, config$1); + const onePassSig = new OnePassSignaturePacket(); + onePassSig.signatureType = signatureType; + onePassSig.hashAlgorithm = await getPreferredHashAlgo$1(primaryKey, signingKey.keyPacket, date, userIDs, config$1); + onePassSig.publicKeyAlgorithm = signingKey.keyPacket.algorithm; + onePassSig.issuerKeyID = signingKey.getKeyID(); + if (i === signingKeys.length - 1) { + onePassSig.flags = 1; + } + return onePassSig; + })).then(onePassSignatureList => { + onePassSignatureList.forEach(onePassSig => packetlist.push(onePassSig)); + }); + + packetlist.push(literalDataPacket); + packetlist.push(...(await createSignaturePackets(literalDataPacket, signingKeys, signature, signingKeyIDs, date, userIDs, notations, false, config$1))); + + return new Message(packetlist); + } + + /** + * Compresses the message (the literal and -if signed- signature data packets of the message) + * @param {module:enums.compression} algo - compression algorithm + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Message} New message with compressed content. + */ + compress(algo, config$1 = config) { + if (algo === enums.compression.uncompressed) { + return this; + } + + const compressed = new CompressedDataPacket(config$1); + compressed.algorithm = algo; + compressed.packets = this.packets; + + const packetList = new PacketList(); + packetList.push(compressed); + + return new Message(packetList); + } + + /** + * Create a detached signature for the message (the literal data packet of the message) + * @param {Array} signingKeys - private keys with decrypted secret key data for signing + * @param {Signature} [signature] - Any existing detached signature + * @param {Array} [signingKeyIDs] - Array of key IDs to use for signing. Each signingKeyIDs[i] corresponds to signingKeys[i] + * @param {Date} [date] - Override the creation time of the signature + * @param {Array} [userIDs] - User IDs to sign with, e.g. [{ name:'Steve Sender', email:'steve@openpgp.org' }] + * @param {Array} [notations] - Notation Data to add to the signatures, e.g. [{ name: 'test@example.org', value: new TextEncoder().encode('test'), humanReadable: true, critical: false }] + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New detached signature of message content. + * @async + */ + async signDetached(signingKeys = [], signature = null, signingKeyIDs = [], date = new Date(), userIDs = [], notations = [], config$1 = config) { + const literalDataPacket = this.packets.findPacket(enums.packet.literalData); + if (!literalDataPacket) { + throw new Error('No literal data packet to sign.'); + } + return new Signature(await createSignaturePackets(literalDataPacket, signingKeys, signature, signingKeyIDs, date, userIDs, notations, true, config$1)); + } + + /** + * Verify message signatures + * @param {Array} verificationKeys - Array of public keys to verify signatures + * @param {Date} [date] - Verify the signature against the given date, i.e. check signature creation time < date < expiration time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise, + * verified: Promise + * }>>} List of signer's keyID and validity of signatures. + * @async + */ + async verify(verificationKeys, date = new Date(), config$1 = config) { + const msg = this.unwrapCompressed(); + const literalDataList = msg.packets.filterByTag(enums.packet.literalData); + if (literalDataList.length !== 1) { + throw new Error('Can only verify message with one literal data packet.'); + } + if (isArrayStream(msg.packets.stream)) { + msg.packets.push(...await readToEnd(msg.packets.stream, _ => _ || [])); + } + const onePassSigList = msg.packets.filterByTag(enums.packet.onePassSignature).reverse(); + const signatureList = msg.packets.filterByTag(enums.packet.signature); + if (onePassSigList.length && !signatureList.length && util.isStream(msg.packets.stream) && !isArrayStream(msg.packets.stream)) { + await Promise.all(onePassSigList.map(async onePassSig => { + onePassSig.correspondingSig = new Promise((resolve, reject) => { + onePassSig.correspondingSigResolve = resolve; + onePassSig.correspondingSigReject = reject; + }); + onePassSig.signatureData = fromAsync(async () => (await onePassSig.correspondingSig).signatureData); + onePassSig.hashed = readToEnd(await onePassSig.hash(onePassSig.signatureType, literalDataList[0], undefined, false)); + onePassSig.hashed.catch(() => {}); + })); + msg.packets.stream = transformPair(msg.packets.stream, async (readable, writable) => { + const reader = getReader(readable); + const writer = getWriter(writable); + try { + for (let i = 0; i < onePassSigList.length; i++) { + const { value: signature } = await reader.read(); + onePassSigList[i].correspondingSigResolve(signature); + } + await reader.readToEnd(); + await writer.ready; + await writer.close(); + } catch (e) { + onePassSigList.forEach(onePassSig => { + onePassSig.correspondingSigReject(e); + }); + await writer.abort(e); + } + }); + return createVerificationObjects(onePassSigList, literalDataList, verificationKeys, date, false, config$1); + } + return createVerificationObjects(signatureList, literalDataList, verificationKeys, date, false, config$1); + } + + /** + * Verify detached message signature + * @param {Array} verificationKeys - Array of public keys to verify signatures + * @param {Signature} signature + * @param {Date} date - Verify the signature against the given date, i.e. check signature creation time < date < expiration time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise, + * verified: Promise + * }>>} List of signer's keyID and validity of signature. + * @async + */ + verifyDetached(signature, verificationKeys, date = new Date(), config$1 = config) { + const msg = this.unwrapCompressed(); + const literalDataList = msg.packets.filterByTag(enums.packet.literalData); + if (literalDataList.length !== 1) { + throw new Error('Can only verify message with one literal data packet.'); + } + const signatureList = signature.packets; + return createVerificationObjects(signatureList, literalDataList, verificationKeys, date, true, config$1); + } + + /** + * Unwrap compressed message + * @returns {Message} Message Content of compressed message. + */ + unwrapCompressed() { + const compressed = this.packets.filterByTag(enums.packet.compressedData); + if (compressed.length) { + return new Message(compressed[0].packets); + } + return this; + } + + /** + * Append signature to unencrypted message object + * @param {String|Uint8Array} detachedSignature - The detached ASCII-armored or Uint8Array PGP signature + * @param {Object} [config] - Full configuration, defaults to openpgp.config + */ + async appendSignature(detachedSignature, config$1 = config) { + await this.packets.read( + util.isUint8Array(detachedSignature) ? detachedSignature : (await unarmor(detachedSignature)).data, + allowedDetachedSignaturePackets, + config$1 + ); + } + + /** + * Returns binary encoded message + * @returns {ReadableStream} Binary message. + */ + write() { + return this.packets.write(); + } + + /** + * Returns ASCII armored text of message + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {ReadableStream} ASCII armor. + */ + armor(config$1 = config) { + return armor(enums.armor.message, this.write(), null, null, null, config$1); + } + } + + /** + * Create signature packets for the message + * @param {LiteralDataPacket} literalDataPacket - the literal data packet to sign + * @param {Array} [signingKeys] - private keys with decrypted secret key data for signing + * @param {Signature} [signature] - Any existing detached signature to append + * @param {Array} [signingKeyIDs] - Array of key IDs to use for signing. Each signingKeyIDs[i] corresponds to signingKeys[i] + * @param {Date} [date] - Override the creationtime of the signature + * @param {Array} [userIDs] - User IDs to sign with, e.g. [{ name:'Steve Sender', email:'steve@openpgp.org' }] + * @param {Array} [notations] - Notation Data to add to the signatures, e.g. [{ name: 'test@example.org', value: new TextEncoder().encode('test'), humanReadable: true, critical: false }] + * @param {Boolean} [detached] - Whether to create detached signature packets + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} List of signature packets. + * @async + * @private + */ + async function createSignaturePackets(literalDataPacket, signingKeys, signature = null, signingKeyIDs = [], date = new Date(), userIDs = [], notations = [], detached = false, config$1 = config) { + const packetlist = new PacketList(); + + // If data packet was created from Uint8Array, use binary, otherwise use text + const signatureType = literalDataPacket.text === null ? + enums.signature.binary : enums.signature.text; + + await Promise.all(signingKeys.map(async (primaryKey, i) => { + const userID = userIDs[i]; + if (!primaryKey.isPrivate()) { + throw new Error('Need private key for signing'); + } + const signingKey = await primaryKey.getSigningKey(signingKeyIDs[i], date, userID, config$1); + return createSignaturePacket(literalDataPacket, primaryKey, signingKey.keyPacket, { signatureType }, date, userID, notations, detached, config$1); + })).then(signatureList => { + packetlist.push(...signatureList); + }); + + if (signature) { + const existingSigPacketlist = signature.packets.filterByTag(enums.packet.signature); + packetlist.push(...existingSigPacketlist); + } + return packetlist; + } + + /** + * Create object containing signer's keyID and validity of signature + * @param {SignaturePacket} signature - Signature packet + * @param {Array} literalDataList - Array of literal data packets + * @param {Array} verificationKeys - Array of public keys to verify signatures + * @param {Date} [date] - Check signature validity with respect to the given date + * @param {Boolean} [detached] - Whether to verify detached signature packets + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise<{ + * keyID: module:type/keyid~KeyID, + * signature: Promise, + * verified: Promise + * }>} signer's keyID and validity of signature + * @async + * @private + */ + async function createVerificationObject(signature, literalDataList, verificationKeys, date = new Date(), detached = false, config$1 = config) { + let primaryKey; + let unverifiedSigningKey; + + for (const key of verificationKeys) { + const issuerKeys = key.getKeys(signature.issuerKeyID); + if (issuerKeys.length > 0) { + primaryKey = key; + unverifiedSigningKey = issuerKeys[0]; + break; + } + } + + const isOnePassSignature = signature instanceof OnePassSignaturePacket; + const signaturePacketPromise = isOnePassSignature ? signature.correspondingSig : signature; + + const verifiedSig = { + keyID: signature.issuerKeyID, + verified: (async () => { + if (!unverifiedSigningKey) { + throw new Error(`Could not find signing key with key ID ${signature.issuerKeyID.toHex()}`); + } + + await signature.verify(unverifiedSigningKey.keyPacket, signature.signatureType, literalDataList[0], date, detached, config$1); + const signaturePacket = await signaturePacketPromise; + if (unverifiedSigningKey.getCreationTime() > signaturePacket.created) { + throw new Error('Key is newer than the signature'); + } + // We pass the signature creation time to check whether the key was expired at the time of signing. + // We check this after signature verification because for streamed one-pass signatures, the creation time is not available before + try { + await primaryKey.getSigningKey(unverifiedSigningKey.getKeyID(), signaturePacket.created, undefined, config$1); + } catch (e) { + // If a key was reformatted then the self-signatures of the signing key might be in the future compared to the message signature, + // making the key invalid at the time of signing. + // However, if the key is valid at the given `date`, we still allow using it provided the relevant `config` setting is enabled. + // Note: we do not support the edge case of a key that was reformatted and it has expired. + if (config$1.allowInsecureVerificationWithReformattedKeys && e.message.match(/Signature creation time is in the future/)) { + await primaryKey.getSigningKey(unverifiedSigningKey.getKeyID(), date, undefined, config$1); + } else { + throw e; + } + } + return true; + })(), + signature: (async () => { + const signaturePacket = await signaturePacketPromise; + const packetlist = new PacketList(); + signaturePacket && packetlist.push(signaturePacket); + return new Signature(packetlist); + })() + }; + + // Mark potential promise rejections as "handled". This is needed because in + // some cases, we reject them before the user has a reasonable chance to + // handle them (e.g. `await readToEnd(result.data); await result.verified` and + // the data stream errors). + verifiedSig.signature.catch(() => {}); + verifiedSig.verified.catch(() => {}); + + return verifiedSig; + } + + /** + * Create list of objects containing signer's keyID and validity of signature + * @param {Array} signatureList - Array of signature packets + * @param {Array} literalDataList - Array of literal data packets + * @param {Array} verificationKeys - Array of public keys to verify signatures + * @param {Date} date - Verify the signature against the given date, + * i.e. check signature creation time < date < expiration time + * @param {Boolean} [detached] - Whether to verify detached signature packets + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise, + * verified: Promise + * }>>} list of signer's keyID and validity of signatures (one entry per signature packet in input) + * @async + * @private + */ + async function createVerificationObjects(signatureList, literalDataList, verificationKeys, date = new Date(), detached = false, config$1 = config) { + return Promise.all(signatureList.filter(function(signature) { + return ['text', 'binary'].includes(enums.read(enums.signature, signature.signatureType)); + }).map(async function(signature) { + return createVerificationObject(signature, literalDataList, verificationKeys, date, detached, config$1); + })); + } + + /** + * Reads an (optionally armored) OpenPGP message and returns a Message object + * @param {Object} options + * @param {String | ReadableStream} [options.armoredMessage] - Armored message to be parsed + * @param {Uint8Array | ReadableStream} [options.binaryMessage] - Binary to be parsed + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} New message object. + * @async + * @static + */ + async function readMessage({ armoredMessage, binaryMessage, config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; + let input = armoredMessage || binaryMessage; + if (!input) { + throw new Error('readMessage: must pass options object containing `armoredMessage` or `binaryMessage`'); + } + if (armoredMessage && !util.isString(armoredMessage) && !util.isStream(armoredMessage)) { + throw new Error('readMessage: options.armoredMessage must be a string or stream'); + } + if (binaryMessage && !util.isUint8Array(binaryMessage) && !util.isStream(binaryMessage)) { + throw new Error('readMessage: options.binaryMessage must be a Uint8Array or stream'); + } + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + const streamType = util.isStream(input); + if (streamType) { + await loadStreamsPonyfill(); + input = toStream(input); + } + if (armoredMessage) { + const { type, data } = await unarmor(input, config$1); + if (type !== enums.armor.message) { + throw new Error('Armored text not of type message'); + } + input = data; + } + const packetlist = await PacketList.fromBinary(input, allowedMessagePackets, config$1); + const message = new Message(packetlist); + message.fromStream = streamType; + return message; + } + + /** + * Creates new message object from text or binary data. + * @param {Object} options + * @param {String | ReadableStream} [options.text] - The text message contents + * @param {Uint8Array | ReadableStream} [options.binary] - The binary message contents + * @param {String} [options.filename=""] - Name of the file (if any) + * @param {Date} [options.date=current date] - Date of the message, or modification date of the file + * @param {'utf8'|'binary'|'text'|'mime'} [options.format='utf8' if text is passed, 'binary' otherwise] - Data packet type + * @returns {Promise} New message object. + * @async + * @static + */ + async function createMessage({ text, binary, filename, date = new Date(), format = text !== undefined ? 'utf8' : 'binary', ...rest }) { + let input = text !== undefined ? text : binary; + if (input === undefined) { + throw new Error('createMessage: must pass options object containing `text` or `binary`'); + } + if (text && !util.isString(text) && !util.isStream(text)) { + throw new Error('createMessage: options.text must be a string or stream'); + } + if (binary && !util.isUint8Array(binary) && !util.isStream(binary)) { + throw new Error('createMessage: options.binary must be a Uint8Array or stream'); + } + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + const streamType = util.isStream(input); + if (streamType) { + await loadStreamsPonyfill(); + input = toStream(input); + } + const literalDataPacket = new LiteralDataPacket(date); + if (text !== undefined) { + literalDataPacket.setText(input, enums.write(enums.literal, format)); + } else { + literalDataPacket.setBytes(input, enums.write(enums.literal, format)); + } + if (filename !== undefined) { + literalDataPacket.setFilename(filename); + } + const literalDataPacketlist = new PacketList(); + literalDataPacketlist.push(literalDataPacket); + const message = new Message(literalDataPacketlist); + message.fromStream = streamType; + return message; + } + + // GPG4Browsers - An OpenPGP implementation in javascript + + // A Cleartext message can contain the following packets + const allowedPackets$5 = /*#__PURE__*/ util.constructAllowedPackets([SignaturePacket]); + + /** + * Class that represents an OpenPGP cleartext signed message. + * See {@link https://tools.ietf.org/html/rfc4880#section-7} + */ + class CleartextMessage { + /** + * @param {String} text - The cleartext of the signed message + * @param {Signature} signature - The detached signature or an empty signature for unsigned messages + */ + constructor(text, signature) { + // remove trailing whitespace and normalize EOL to canonical form + this.text = util.removeTrailingSpaces(text).replace(/\r?\n/g, '\r\n'); + if (signature && !(signature instanceof Signature)) { + throw new Error('Invalid signature input'); + } + this.signature = signature || new Signature(new PacketList()); + } + + /** + * Returns the key IDs of the keys that signed the cleartext message + * @returns {Array} Array of keyID objects. + */ + getSigningKeyIDs() { + const keyIDs = []; + const signatureList = this.signature.packets; + signatureList.forEach(function(packet) { + keyIDs.push(packet.issuerKeyID); + }); + return keyIDs; + } + + /** + * Sign the cleartext message + * @param {Array} privateKeys - private keys with decrypted secret key data for signing + * @param {Signature} [signature] - Any existing detached signature + * @param {Array} [signingKeyIDs] - Array of key IDs to use for signing. Each signingKeyIDs[i] corresponds to privateKeys[i] + * @param {Date} [date] - The creation time of the signature that should be created + * @param {Array} [userIDs] - User IDs to sign with, e.g. [{ name:'Steve Sender', email:'steve@openpgp.org' }] + * @param {Array} [notations] - Notation Data to add to the signatures, e.g. [{ name: 'test@example.org', value: new TextEncoder().encode('test'), humanReadable: true, critical: false }] + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise} New cleartext message with signed content. + * @async + */ + async sign(privateKeys, signature = null, signingKeyIDs = [], date = new Date(), userIDs = [], notations = [], config$1 = config) { + const literalDataPacket = new LiteralDataPacket(); + literalDataPacket.setText(this.text); + const newSignature = new Signature(await createSignaturePackets(literalDataPacket, privateKeys, signature, signingKeyIDs, date, userIDs, notations, true, config$1)); + return new CleartextMessage(this.text, newSignature); + } + + /** + * Verify signatures of cleartext signed message + * @param {Array} keys - Array of keys to verify signatures + * @param {Date} [date] - Verify the signature against the given date, i.e. check signature creation time < date < expiration time + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {Promise, + * verified: Promise + * }>>} List of signer's keyID and validity of signature. + * @async + */ + verify(keys, date = new Date(), config$1 = config) { + const signatureList = this.signature.packets; + const literalDataPacket = new LiteralDataPacket(); + // we assume that cleartext signature is generated based on UTF8 cleartext + literalDataPacket.setText(this.text); + return createVerificationObjects(signatureList, [literalDataPacket], keys, date, true, config$1); + } + + /** + * Get cleartext + * @returns {String} Cleartext of message. + */ + getText() { + // normalize end of line to \n + return this.text.replace(/\r\n/g, '\n'); + } + + /** + * Returns ASCII armored text of cleartext signed message + * @param {Object} [config] - Full configuration, defaults to openpgp.config + * @returns {String | ReadableStream} ASCII armor. + */ + armor(config$1 = config) { + let hashes = this.signature.packets.map(function(packet) { + return enums.read(enums.hash, packet.hashAlgorithm).toUpperCase(); + }); + hashes = hashes.filter(function(item, i, ar) { return ar.indexOf(item) === i; }); + const body = { + hash: hashes.join(), + text: this.text, + data: this.signature.packets.write() + }; + return armor(enums.armor.signed, body, undefined, undefined, undefined, config$1); + } + } + + /** + * Reads an OpenPGP cleartext signed message and returns a CleartextMessage object + * @param {Object} options + * @param {String} options.cleartextMessage - Text to be parsed + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} New cleartext message object. + * @async + * @static + */ + async function readCleartextMessage({ cleartextMessage, config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; + if (!cleartextMessage) { + throw new Error('readCleartextMessage: must pass options object containing `cleartextMessage`'); + } + if (!util.isString(cleartextMessage)) { + throw new Error('readCleartextMessage: options.cleartextMessage must be a string'); + } + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + const input = await unarmor(cleartextMessage); + if (input.type !== enums.armor.signed) { + throw new Error('No cleartext signed message.'); + } + const packetlist = await PacketList.fromBinary(input.data, allowedPackets$5, config$1); + verifyHeaders$1(input.headers, packetlist); + const signature = new Signature(packetlist); + return new CleartextMessage(input.text, signature); + } + + /** + * Compare hash algorithm specified in the armor header with signatures + * @param {Array} headers - Armor headers + * @param {PacketList} packetlist - The packetlist with signature packets + * @private + */ + function verifyHeaders$1(headers, packetlist) { + const checkHashAlgos = function(hashAlgos) { + const check = packet => algo => packet.hashAlgorithm === algo; + + for (let i = 0; i < packetlist.length; i++) { + if (packetlist[i].constructor.tag === enums.packet.signature && !hashAlgos.some(check(packetlist[i]))) { + return false; + } + } + return true; + }; + + let oneHeader = null; + let hashAlgos = []; + headers.forEach(function(header) { + oneHeader = header.match(/Hash: (.+)/); // get header value + if (oneHeader) { + oneHeader = oneHeader[1].replace(/\s/g, ''); // remove whitespace + oneHeader = oneHeader.split(','); + oneHeader = oneHeader.map(function(hash) { + hash = hash.toLowerCase(); + try { + return enums.write(enums.hash, hash); + } catch (e) { + throw new Error('Unknown hash algorithm in armor header: ' + hash); + } + }); + hashAlgos = hashAlgos.concat(oneHeader); + } else { + throw new Error('Only "Hash" header allowed in cleartext signed message'); + } + }); + + if (!hashAlgos.length && !checkHashAlgos([enums.hash.md5])) { + throw new Error('If no "Hash" header in cleartext signed message, then only MD5 signatures allowed'); + } else if (hashAlgos.length && !checkHashAlgos(hashAlgos)) { + throw new Error('Hash algorithm mismatch in armor header and signature'); + } + } + + /** + * Creates a new CleartextMessage object from text + * @param {Object} options + * @param {String} options.text + * @static + * @async + */ + async function createCleartextMessage({ text, ...rest }) { + if (!text) { + throw new Error('createCleartextMessage: must pass options object containing `text`'); + } + if (!util.isString(text)) { + throw new Error('createCleartextMessage: options.text must be a string'); + } + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + return new CleartextMessage(text); + } + + // OpenPGP.js - An OpenPGP implementation in javascript + + + ////////////////////// + // // + // Key handling // + // // + ////////////////////// + + + /** + * Generates a new OpenPGP key pair. Supports RSA and ECC keys. By default, primary and subkeys will be of same type. + * The generated primary key will have signing capabilities. By default, one subkey with encryption capabilities is also generated. + * @param {Object} options + * @param {Object|Array} options.userIDs - User IDs as objects: `{ name: 'Jo Doe', email: 'info@jo.com' }` + * @param {'ecc'|'rsa'} [options.type='ecc'] - The primary key algorithm type: ECC (default) or RSA + * @param {String} [options.passphrase=(not protected)] - The passphrase used to encrypt the generated private key. If omitted or empty, the key won't be encrypted. + * @param {Number} [options.rsaBits=4096] - Number of bits for RSA keys + * @param {String} [options.curve='curve25519'] - Elliptic curve for ECC keys: + * curve25519 (default), p256, p384, p521, secp256k1, + * brainpoolP256r1, brainpoolP384r1, or brainpoolP512r1 + * @param {Date} [options.date=current date] - Override the creation date of the key and the key signatures + * @param {Number} [options.keyExpirationTime=0 (never expires)] - Number of seconds from the key creation time after which the key expires + * @param {Array} [options.subkeys=a single encryption subkey] - Options for each subkey e.g. `[{sign: true, passphrase: '123'}]` + * default to main key options, except for `sign` parameter that defaults to false, and indicates whether the subkey should sign rather than encrypt + * @param {'armored'|'binary'|'object'} [options.format='armored'] - format of the output keys + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} The generated key object in the form: + * { privateKey:PrivateKey|Uint8Array|String, publicKey:PublicKey|Uint8Array|String, revocationCertificate:String } + * @async + * @static + */ + async function generateKey({ userIDs = [], passphrase, type = 'ecc', rsaBits = 4096, curve = 'curve25519', keyExpirationTime = 0, date = new Date(), subkeys = [{}], format = 'armored', config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + userIDs = toArray$1(userIDs); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (userIDs.length === 0) { + throw new Error('UserIDs are required for key generation'); + } + if (type === 'rsa' && rsaBits < config$1.minRSABits) { + throw new Error(`rsaBits should be at least ${config$1.minRSABits}, got: ${rsaBits}`); + } + + const options = { userIDs, passphrase, type, rsaBits, curve, keyExpirationTime, date, subkeys }; + + try { + const { key, revocationCertificate } = await generate$2(options, config$1); + key.getKeys().forEach(({ keyPacket }) => checkKeyRequirements(keyPacket, config$1)); + + return { + privateKey: formatObject(key, format, config$1), + publicKey: formatObject(key.toPublic(), format, config$1), + revocationCertificate + }; + } catch (err) { + throw util.wrapError('Error generating keypair', err); + } + } + + /** + * Reformats signature packets for a key and rewraps key object. + * @param {Object} options + * @param {PrivateKey} options.privateKey - Private key to reformat + * @param {Object|Array} options.userIDs - User IDs as objects: `{ name: 'Jo Doe', email: 'info@jo.com' }` + * @param {String} [options.passphrase=(not protected)] - The passphrase used to encrypt the reformatted private key. If omitted or empty, the key won't be encrypted. + * @param {Number} [options.keyExpirationTime=0 (never expires)] - Number of seconds from the key creation time after which the key expires + * @param {Date} [options.date] - Override the creation date of the key signatures. If the key was previously used to sign messages, it is recommended + * to set the same date as the key creation time to ensure that old message signatures will still be verifiable using the reformatted key. + * @param {'armored'|'binary'|'object'} [options.format='armored'] - format of the output keys + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} The generated key object in the form: + * { privateKey:PrivateKey|Uint8Array|String, publicKey:PublicKey|Uint8Array|String, revocationCertificate:String } + * @async + * @static + */ + async function reformatKey({ privateKey, userIDs = [], passphrase, keyExpirationTime = 0, date, format = 'armored', config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + userIDs = toArray$1(userIDs); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (userIDs.length === 0) { + throw new Error('UserIDs are required for key reformat'); + } + const options = { privateKey, userIDs, passphrase, keyExpirationTime, date }; + + try { + const { key: reformattedKey, revocationCertificate } = await reformat(options, config$1); + + return { + privateKey: formatObject(reformattedKey, format, config$1), + publicKey: formatObject(reformattedKey.toPublic(), format, config$1), + revocationCertificate + }; + } catch (err) { + throw util.wrapError('Error reformatting keypair', err); + } + } + + /** + * Revokes a key. Requires either a private key or a revocation certificate. + * If a revocation certificate is passed, the reasonForRevocation parameter will be ignored. + * @param {Object} options + * @param {Key} options.key - Public or private key to revoke + * @param {String} [options.revocationCertificate] - Revocation certificate to revoke the key with + * @param {Object} [options.reasonForRevocation] - Object indicating the reason for revocation + * @param {module:enums.reasonForRevocation} [options.reasonForRevocation.flag=[noReason]{@link module:enums.reasonForRevocation}] - Flag indicating the reason for revocation + * @param {String} [options.reasonForRevocation.string=""] - String explaining the reason for revocation + * @param {Date} [options.date] - Use the given date instead of the current time to verify validity of revocation certificate (if provided), or as creation time of the revocation signature + * @param {'armored'|'binary'|'object'} [options.format='armored'] - format of the output key(s) + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} The revoked key in the form: + * { privateKey:PrivateKey|Uint8Array|String, publicKey:PublicKey|Uint8Array|String } if private key is passed, or + * { privateKey: null, publicKey:PublicKey|Uint8Array|String } otherwise + * @async + * @static + */ + async function revokeKey({ key, revocationCertificate, reasonForRevocation, date = new Date(), format = 'armored', config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + try { + const revokedKey = revocationCertificate ? + await key.applyRevocationCertificate(revocationCertificate, date, config$1) : + await key.revoke(reasonForRevocation, date, config$1); + + return revokedKey.isPrivate() ? { + privateKey: formatObject(revokedKey, format, config$1), + publicKey: formatObject(revokedKey.toPublic(), format, config$1) + } : { + privateKey: null, + publicKey: formatObject(revokedKey, format, config$1) + }; + } catch (err) { + throw util.wrapError('Error revoking key', err); + } + } + + /** + * Unlock a private key with the given passphrase. + * This method does not change the original key. + * @param {Object} options + * @param {PrivateKey} options.privateKey - The private key to decrypt + * @param {String|Array} options.passphrase - The user's passphrase(s) + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} The unlocked key object. + * @async + */ + async function decryptKey({ privateKey, passphrase, config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (!privateKey.isPrivate()) { + throw new Error('Cannot decrypt a public key'); + } + const clonedPrivateKey = privateKey.clone(true); + const passphrases = util.isArray(passphrase) ? passphrase : [passphrase]; + + try { + await Promise.all(clonedPrivateKey.getKeys().map(key => ( + // try to decrypt each key with any of the given passphrases + util.anyPromise(passphrases.map(passphrase => key.keyPacket.decrypt(passphrase))) + ))); + + await clonedPrivateKey.validate(config$1); + return clonedPrivateKey; + } catch (err) { + clonedPrivateKey.clearPrivateParams(); + throw util.wrapError('Error decrypting private key', err); + } + } + + /** + * Lock a private key with the given passphrase. + * This method does not change the original key. + * @param {Object} options + * @param {PrivateKey} options.privateKey - The private key to encrypt + * @param {String|Array} options.passphrase - If multiple passphrases, they should be in the same order as the packets each should encrypt + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} The locked key object. + * @async + */ + async function encryptKey({ privateKey, passphrase, config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (!privateKey.isPrivate()) { + throw new Error('Cannot encrypt a public key'); + } + const clonedPrivateKey = privateKey.clone(true); + + const keys = clonedPrivateKey.getKeys(); + const passphrases = util.isArray(passphrase) ? passphrase : new Array(keys.length).fill(passphrase); + if (passphrases.length !== keys.length) { + throw new Error('Invalid number of passphrases given for key encryption'); + } + + try { + await Promise.all(keys.map(async (key, i) => { + const { keyPacket } = key; + await keyPacket.encrypt(passphrases[i], config$1); + keyPacket.clearPrivateParams(); + })); + return clonedPrivateKey; + } catch (err) { + clonedPrivateKey.clearPrivateParams(); + throw util.wrapError('Error encrypting private key', err); + } + } + + + /////////////////////////////////////////// + // // + // Message encryption and decryption // + // // + /////////////////////////////////////////// + + + /** + * Encrypts a message using public keys, passwords or both at once. At least one of `encryptionKeys`, `passwords` or `sessionKeys` + * must be specified. If signing keys are specified, those will be used to sign the message. + * @param {Object} options + * @param {Message} options.message - Message to be encrypted as created by {@link createMessage} + * @param {PublicKey|PublicKey[]} [options.encryptionKeys] - Array of keys or single key, used to encrypt the message + * @param {PrivateKey|PrivateKey[]} [options.signingKeys] - Private keys for signing. If omitted message will not be signed + * @param {String|String[]} [options.passwords] - Array of passwords or a single password to encrypt the message + * @param {Object} [options.sessionKey] - Session key in the form: `{ data:Uint8Array, algorithm:String }` + * @param {'armored'|'binary'|'object'} [options.format='armored'] - Format of the returned message + * @param {Signature} [options.signature] - A detached signature to add to the encrypted message + * @param {Boolean} [options.wildcard=false] - Use a key ID of 0 instead of the public key IDs + * @param {KeyID|KeyID[]} [options.signingKeyIDs=latest-created valid signing (sub)keys] - Array of key IDs to use for signing. Each `signingKeyIDs[i]` corresponds to `signingKeys[i]` + * @param {KeyID|KeyID[]} [options.encryptionKeyIDs=latest-created valid encryption (sub)keys] - Array of key IDs to use for encryption. Each `encryptionKeyIDs[i]` corresponds to `encryptionKeys[i]` + * @param {Date} [options.date=current date] - Override the creation date of the message signature + * @param {Object|Object[]} [options.signingUserIDs=primary user IDs] - Array of user IDs to sign with, one per key in `signingKeys`, e.g. `[{ name: 'Steve Sender', email: 'steve@openpgp.org' }]` + * @param {Object|Object[]} [options.encryptionUserIDs=primary user IDs] - Array of user IDs to encrypt for, one per key in `encryptionKeys`, e.g. `[{ name: 'Robert Receiver', email: 'robert@openpgp.org' }]` + * @param {Object|Object[]} [options.signatureNotations=[]] - Array of notations to add to the signatures, e.g. `[{ name: 'test@example.org', value: new TextEncoder().encode('test'), humanReadable: true, critical: false }]` + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise|MaybeStream>} Encrypted message (string if `armor` was true, the default; Uint8Array if `armor` was false). + * @async + * @static + */ + async function encrypt$4({ message, encryptionKeys, signingKeys, passwords, sessionKey, format = 'armored', signature = null, wildcard = false, signingKeyIDs = [], encryptionKeyIDs = [], date = new Date(), signingUserIDs = [], encryptionUserIDs = [], signatureNotations = [], config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + checkMessage(message); checkOutputMessageFormat(format); + encryptionKeys = toArray$1(encryptionKeys); signingKeys = toArray$1(signingKeys); passwords = toArray$1(passwords); + signingKeyIDs = toArray$1(signingKeyIDs); encryptionKeyIDs = toArray$1(encryptionKeyIDs); signingUserIDs = toArray$1(signingUserIDs); encryptionUserIDs = toArray$1(encryptionUserIDs); signatureNotations = toArray$1(signatureNotations); + if (rest.detached) { + throw new Error("The `detached` option has been removed from openpgp.encrypt, separately call openpgp.sign instead. Don't forget to remove the `privateKeys` option as well."); + } + if (rest.publicKeys) throw new Error('The `publicKeys` option has been removed from openpgp.encrypt, pass `encryptionKeys` instead'); + if (rest.privateKeys) throw new Error('The `privateKeys` option has been removed from openpgp.encrypt, pass `signingKeys` instead'); + if (rest.armor !== undefined) throw new Error('The `armor` option has been removed from openpgp.encrypt, pass `format` instead.'); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (!signingKeys) { + signingKeys = []; + } + const streaming = message.fromStream; + try { + if (signingKeys.length || signature) { // sign the message only if signing keys or signature is specified + message = await message.sign(signingKeys, signature, signingKeyIDs, date, signingUserIDs, signatureNotations, config$1); + } + message = message.compress( + await getPreferredAlgo('compression', encryptionKeys, date, encryptionUserIDs, config$1), + config$1 + ); + message = await message.encrypt(encryptionKeys, passwords, sessionKey, wildcard, encryptionKeyIDs, date, encryptionUserIDs, config$1); + if (format === 'object') return message; + // serialize data + const armor = format === 'armored'; + const data = armor ? message.armor(config$1) : message.write(); + return convertStream(data, streaming, armor ? 'utf8' : 'binary'); + } catch (err) { + throw util.wrapError('Error encrypting message', err); + } + } + + /** + * Decrypts a message with the user's private key, a session key or a password. + * One of `decryptionKeys`, `sessionkeys` or `passwords` must be specified (passing a combination of these options is not supported). + * @param {Object} options + * @param {Message} options.message - The message object with the encrypted data + * @param {PrivateKey|PrivateKey[]} [options.decryptionKeys] - Private keys with decrypted secret key data or session key + * @param {String|String[]} [options.passwords] - Passwords to decrypt the message + * @param {Object|Object[]} [options.sessionKeys] - Session keys in the form: { data:Uint8Array, algorithm:String } + * @param {PublicKey|PublicKey[]} [options.verificationKeys] - Array of public keys or single key, to verify signatures + * @param {Boolean} [options.expectSigned=false] - If true, data decryption fails if the message is not signed with the provided publicKeys + * @param {'utf8'|'binary'} [options.format='utf8'] - Whether to return data as a string(Stream) or Uint8Array(Stream). If 'utf8' (the default), also normalize newlines. + * @param {Signature} [options.signature] - Detached signature for verification + * @param {Date} [options.date=current date] - Use the given date for verification instead of the current time + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} Object containing decrypted and verified message in the form: + * + * { + * data: MaybeStream, (if format was 'utf8', the default) + * data: MaybeStream, (if format was 'binary') + * filename: String, + * signatures: [ + * { + * keyID: module:type/keyid~KeyID, + * verified: Promise, + * signature: Promise + * }, ... + * ] + * } + * + * where `signatures` contains a separate entry for each signature packet found in the input message. + * @async + * @static + */ + async function decrypt$4({ message, decryptionKeys, passwords, sessionKeys, verificationKeys, expectSigned = false, format = 'utf8', signature = null, date = new Date(), config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + checkMessage(message); verificationKeys = toArray$1(verificationKeys); decryptionKeys = toArray$1(decryptionKeys); passwords = toArray$1(passwords); sessionKeys = toArray$1(sessionKeys); + if (rest.privateKeys) throw new Error('The `privateKeys` option has been removed from openpgp.decrypt, pass `decryptionKeys` instead'); + if (rest.publicKeys) throw new Error('The `publicKeys` option has been removed from openpgp.decrypt, pass `verificationKeys` instead'); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + try { + const decrypted = await message.decrypt(decryptionKeys, passwords, sessionKeys, date, config$1); + if (!verificationKeys) { + verificationKeys = []; + } + + const result = {}; + result.signatures = signature ? await decrypted.verifyDetached(signature, verificationKeys, date, config$1) : await decrypted.verify(verificationKeys, date, config$1); + result.data = format === 'binary' ? decrypted.getLiteralData() : decrypted.getText(); + result.filename = decrypted.getFilename(); + linkStreams(result, message); + if (expectSigned) { + if (verificationKeys.length === 0) { + throw new Error('Verification keys are required to verify message signatures'); + } + if (result.signatures.length === 0) { + throw new Error('Message is not signed'); + } + result.data = concat([ + result.data, + fromAsync(async () => { + await util.anyPromise(result.signatures.map(sig => sig.verified)); + }) + ]); + } + result.data = await convertStream(result.data, message.fromStream, format); + return result; + } catch (err) { + throw util.wrapError('Error decrypting message', err); + } + } + + + ////////////////////////////////////////// + // // + // Message signing and verification // + // // + ////////////////////////////////////////// + + + /** + * Signs a message. + * @param {Object} options + * @param {CleartextMessage|Message} options.message - (cleartext) message to be signed + * @param {PrivateKey|PrivateKey[]} options.signingKeys - Array of keys or single key with decrypted secret key data to sign cleartext + * @param {'armored'|'binary'|'object'} [options.format='armored'] - Format of the returned message + * @param {Boolean} [options.detached=false] - If the return value should contain a detached signature + * @param {KeyID|KeyID[]} [options.signingKeyIDs=latest-created valid signing (sub)keys] - Array of key IDs to use for signing. Each signingKeyIDs[i] corresponds to signingKeys[i] + * @param {Date} [options.date=current date] - Override the creation date of the signature + * @param {Object|Object[]} [options.signingUserIDs=primary user IDs] - Array of user IDs to sign with, one per key in `signingKeys`, e.g. `[{ name: 'Steve Sender', email: 'steve@openpgp.org' }]` + * @param {Object|Object[]} [options.signatureNotations=[]] - Array of notations to add to the signatures, e.g. `[{ name: 'test@example.org', value: new TextEncoder().encode('test'), humanReadable: true, critical: false }]` + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise>} Signed message (string if `armor` was true, the default; Uint8Array if `armor` was false). + * @async + * @static + */ + async function sign$5({ message, signingKeys, format = 'armored', detached = false, signingKeyIDs = [], date = new Date(), signingUserIDs = [], signatureNotations = [], config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + checkCleartextOrMessage(message); checkOutputMessageFormat(format); + signingKeys = toArray$1(signingKeys); signingKeyIDs = toArray$1(signingKeyIDs); signingUserIDs = toArray$1(signingUserIDs); signatureNotations = toArray$1(signatureNotations); + + if (rest.privateKeys) throw new Error('The `privateKeys` option has been removed from openpgp.sign, pass `signingKeys` instead'); + if (rest.armor !== undefined) throw new Error('The `armor` option has been removed from openpgp.sign, pass `format` instead.'); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (message instanceof CleartextMessage && format === 'binary') throw new Error('Cannot return signed cleartext message in binary format'); + if (message instanceof CleartextMessage && detached) throw new Error('Cannot detach-sign a cleartext message'); + + if (!signingKeys || signingKeys.length === 0) { + throw new Error('No signing keys provided'); + } + + try { + let signature; + if (detached) { + signature = await message.signDetached(signingKeys, undefined, signingKeyIDs, date, signingUserIDs, signatureNotations, config$1); + } else { + signature = await message.sign(signingKeys, undefined, signingKeyIDs, date, signingUserIDs, signatureNotations, config$1); + } + if (format === 'object') return signature; + + const armor = format === 'armored'; + signature = armor ? signature.armor(config$1) : signature.write(); + if (detached) { + signature = transformPair(message.packets.write(), async (readable, writable) => { + await Promise.all([ + pipe(signature, writable), + readToEnd(readable).catch(() => {}) + ]); + }); + } + return convertStream(signature, message.fromStream, armor ? 'utf8' : 'binary'); + } catch (err) { + throw util.wrapError('Error signing message', err); + } + } + + /** + * Verifies signatures of cleartext signed message + * @param {Object} options + * @param {CleartextMessage|Message} options.message - (cleartext) message object with signatures + * @param {PublicKey|PublicKey[]} options.verificationKeys - Array of publicKeys or single key, to verify signatures + * @param {Boolean} [options.expectSigned=false] - If true, verification throws if the message is not signed with the provided publicKeys + * @param {'utf8'|'binary'} [options.format='utf8'] - Whether to return data as a string(Stream) or Uint8Array(Stream). If 'utf8' (the default), also normalize newlines. + * @param {Signature} [options.signature] - Detached signature for verification + * @param {Date} [options.date=current date] - Use the given date for verification instead of the current time + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} Object containing verified message in the form: + * + * { + * data: MaybeStream, (if `message` was a CleartextMessage) + * data: MaybeStream, (if `message` was a Message) + * signatures: [ + * { + * keyID: module:type/keyid~KeyID, + * verified: Promise, + * signature: Promise + * }, ... + * ] + * } + * + * where `signatures` contains a separate entry for each signature packet found in the input message. + * @async + * @static + */ + async function verify$5({ message, verificationKeys, expectSigned = false, format = 'utf8', signature = null, date = new Date(), config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + checkCleartextOrMessage(message); verificationKeys = toArray$1(verificationKeys); + if (rest.publicKeys) throw new Error('The `publicKeys` option has been removed from openpgp.verify, pass `verificationKeys` instead'); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if (message instanceof CleartextMessage && format === 'binary') throw new Error("Can't return cleartext message data as binary"); + if (message instanceof CleartextMessage && signature) throw new Error("Can't verify detached cleartext signature"); + + try { + const result = {}; + if (signature) { + result.signatures = await message.verifyDetached(signature, verificationKeys, date, config$1); + } else { + result.signatures = await message.verify(verificationKeys, date, config$1); + } + result.data = format === 'binary' ? message.getLiteralData() : message.getText(); + if (message.fromStream) linkStreams(result, message); + if (expectSigned) { + if (result.signatures.length === 0) { + throw new Error('Message is not signed'); + } + result.data = concat([ + result.data, + fromAsync(async () => { + await util.anyPromise(result.signatures.map(sig => sig.verified)); + }) + ]); + } + result.data = await convertStream(result.data, message.fromStream, format); + return result; + } catch (err) { + throw util.wrapError('Error verifying signed message', err); + } + } + + + /////////////////////////////////////////////// + // // + // Session key encryption and decryption // + // // + /////////////////////////////////////////////// + + /** + * Generate a new session key object, taking the algorithm preferences of the passed public keys into account, if any. + * @param {Object} options + * @param {PublicKey|PublicKey[]} [options.encryptionKeys] - Array of public keys or single key used to select algorithm preferences for. If no keys are given, the algorithm will be [config.preferredSymmetricAlgorithm]{@link module:config.preferredSymmetricAlgorithm} + * @param {Date} [options.date=current date] - Date to select algorithm preferences at + * @param {Object|Object[]} [options.encryptionUserIDs=primary user IDs] - User IDs to select algorithm preferences for + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise<{ data: Uint8Array, algorithm: String }>} Object with session key data and algorithm. + * @async + * @static + */ + async function generateSessionKey$1({ encryptionKeys, date = new Date(), encryptionUserIDs = [], config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + encryptionKeys = toArray$1(encryptionKeys); encryptionUserIDs = toArray$1(encryptionUserIDs); + if (rest.publicKeys) throw new Error('The `publicKeys` option has been removed from openpgp.generateSessionKey, pass `encryptionKeys` instead'); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + try { + const sessionKeys = await Message.generateSessionKey(encryptionKeys, date, encryptionUserIDs, config$1); + return sessionKeys; + } catch (err) { + throw util.wrapError('Error generating session key', err); + } + } + + /** + * Encrypt a symmetric session key with public keys, passwords, or both at once. + * At least one of `encryptionKeys` or `passwords` must be specified. + * @param {Object} options + * @param {Uint8Array} options.data - The session key to be encrypted e.g. 16 random bytes (for aes128) + * @param {String} options.algorithm - Algorithm of the symmetric session key e.g. 'aes128' or 'aes256' + * @param {String} [options.aeadAlgorithm] - AEAD algorithm, e.g. 'eax' or 'ocb' + * @param {PublicKey|PublicKey[]} [options.encryptionKeys] - Array of public keys or single key, used to encrypt the key + * @param {String|String[]} [options.passwords] - Passwords for the message + * @param {'armored'|'binary'} [options.format='armored'] - Format of the returned value + * @param {Boolean} [options.wildcard=false] - Use a key ID of 0 instead of the public key IDs + * @param {KeyID|KeyID[]} [options.encryptionKeyIDs=latest-created valid encryption (sub)keys] - Array of key IDs to use for encryption. Each encryptionKeyIDs[i] corresponds to encryptionKeys[i] + * @param {Date} [options.date=current date] - Override the date + * @param {Object|Object[]} [options.encryptionUserIDs=primary user IDs] - Array of user IDs to encrypt for, one per key in `encryptionKeys`, e.g. `[{ name: 'Phil Zimmermann', email: 'phil@openpgp.org' }]` + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} Encrypted session keys (string if `armor` was true, the default; Uint8Array if `armor` was false). + * @async + * @static + */ + async function encryptSessionKey({ data, algorithm, aeadAlgorithm, encryptionKeys, passwords, format = 'armored', wildcard = false, encryptionKeyIDs = [], date = new Date(), encryptionUserIDs = [], config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + checkBinary(data); checkString(algorithm, 'algorithm'); checkOutputMessageFormat(format); + encryptionKeys = toArray$1(encryptionKeys); passwords = toArray$1(passwords); encryptionKeyIDs = toArray$1(encryptionKeyIDs); encryptionUserIDs = toArray$1(encryptionUserIDs); + if (rest.publicKeys) throw new Error('The `publicKeys` option has been removed from openpgp.encryptSessionKey, pass `encryptionKeys` instead'); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + if ((!encryptionKeys || encryptionKeys.length === 0) && (!passwords || passwords.length === 0)) { + throw new Error('No encryption keys or passwords provided.'); + } + + try { + const message = await Message.encryptSessionKey(data, algorithm, aeadAlgorithm, encryptionKeys, passwords, wildcard, encryptionKeyIDs, date, encryptionUserIDs, config$1); + return formatObject(message, format, config$1); + } catch (err) { + throw util.wrapError('Error encrypting session key', err); + } + } + + /** + * Decrypt symmetric session keys using private keys or passwords (not both). + * One of `decryptionKeys` or `passwords` must be specified. + * @param {Object} options + * @param {Message} options.message - A message object containing the encrypted session key packets + * @param {PrivateKey|PrivateKey[]} [options.decryptionKeys] - Private keys with decrypted secret key data + * @param {String|String[]} [options.passwords] - Passwords to decrypt the session key + * @param {Date} [options.date] - Date to use for key verification instead of the current time + * @param {Object} [options.config] - Custom configuration settings to overwrite those in [config]{@link module:config} + * @returns {Promise} Array of decrypted session key, algorithm pairs in the form: + * { data:Uint8Array, algorithm:String } + * @throws if no session key could be found or decrypted + * @async + * @static + */ + async function decryptSessionKeys({ message, decryptionKeys, passwords, date = new Date(), config: config$1, ...rest }) { + config$1 = { ...config, ...config$1 }; checkConfig(config$1); + checkMessage(message); decryptionKeys = toArray$1(decryptionKeys); passwords = toArray$1(passwords); + if (rest.privateKeys) throw new Error('The `privateKeys` option has been removed from openpgp.decryptSessionKeys, pass `decryptionKeys` instead'); + const unknownOptions = Object.keys(rest); if (unknownOptions.length > 0) throw new Error(`Unknown option: ${unknownOptions.join(', ')}`); + + try { + const sessionKeys = await message.decryptSessionKeys(decryptionKeys, passwords, date, config$1); + return sessionKeys; + } catch (err) { + throw util.wrapError('Error decrypting session keys', err); + } + } + + + ////////////////////////// + // // + // Helper functions // + // // + ////////////////////////// + + + /** + * Input validation + * @private + */ + function checkString(data, name) { + if (!util.isString(data)) { + throw new Error('Parameter [' + (name || 'data') + '] must be of type String'); + } + } + function checkBinary(data, name) { + if (!util.isUint8Array(data)) { + throw new Error('Parameter [' + (name || 'data') + '] must be of type Uint8Array'); + } + } + function checkMessage(message) { + if (!(message instanceof Message)) { + throw new Error('Parameter [message] needs to be of type Message'); + } + } + function checkCleartextOrMessage(message) { + if (!(message instanceof CleartextMessage) && !(message instanceof Message)) { + throw new Error('Parameter [message] needs to be of type Message or CleartextMessage'); + } + } + function checkOutputMessageFormat(format) { + if (format !== 'armored' && format !== 'binary' && format !== 'object') { + throw new Error(`Unsupported format ${format}`); + } + } + const defaultConfigPropsCount = Object.keys(config).length; + function checkConfig(config$1) { + const inputConfigProps = Object.keys(config$1); + if (inputConfigProps.length !== defaultConfigPropsCount) { + for (const inputProp of inputConfigProps) { + if (config[inputProp] === undefined) { + throw new Error(`Unknown config property: ${inputProp}`); + } + } + } + } + + /** + * Normalize parameter to an array if it is not undefined. + * @param {Object} param - the parameter to be normalized + * @returns {Array|undefined} The resulting array or undefined. + * @private + */ + function toArray$1(param) { + if (param && !util.isArray(param)) { + param = [param]; + } + return param; + } + + /** + * Convert data to or from Stream + * @param {Object} data - the data to convert + * @param {'web'|'ponyfill'|'node'|false} streaming - Whether to return a ReadableStream, and of what type + * @param {'utf8'|'binary'} [encoding] - How to return data in Node Readable streams + * @returns {Promise} The data in the respective format. + * @async + * @private + */ + async function convertStream(data, streaming, encoding = 'utf8') { + const streamType = util.isStream(data); + if (streamType === 'array') { + return readToEnd(data); + } + if (streaming === 'node') { + data = webToNode(data); + if (encoding !== 'binary') data.setEncoding(encoding); + return data; + } + if (streaming === 'web' && streamType === 'ponyfill') { + return toNativeReadable(data); + } + return data; + } + + /** + * Link result.data to the message stream for cancellation. + * Also, forward errors in the message to result.data. + * @param {Object} result - the data to convert + * @param {Message} message - message object + * @returns {Object} + * @private + */ + function linkStreams(result, message) { + result.data = transformPair(message.packets.stream, async (readable, writable) => { + await pipe(result.data, writable, { + preventClose: true + }); + const writer = getWriter(writable); + try { + // Forward errors in the message stream to result.data. + await readToEnd(readable, _ => _); + await writer.close(); + } catch (e) { + await writer.abort(e); + } + }); + } + + /** + * Convert the object to the given format + * @param {Key|Message} object + * @param {'armored'|'binary'|'object'} format + * @param {Object} config - Full configuration + * @returns {String|Uint8Array|Object} + */ + function formatObject(object, format, config) { + switch (format) { + case 'object': + return object; + case 'armored': + return object.armor(config); + case 'binary': + return object.write(); + default: + throw new Error(`Unsupported format ${format}`); + } + } + + /** + * web-streams-polyfill v3.0.3 + */ + /// + const SymbolPolyfill = typeof Symbol === 'function' && typeof Symbol.iterator === 'symbol' ? + Symbol : + description => `Symbol(${description})`; + + /// + function noop() { + return undefined; + } + function getGlobals() { + if (typeof self !== 'undefined') { + return self; + } + else if (typeof window !== 'undefined') { + return window; + } + else if (typeof global !== 'undefined') { + return global; + } + return undefined; + } + const globals = getGlobals(); + + function typeIsObject(x) { + return (typeof x === 'object' && x !== null) || typeof x === 'function'; + } + const rethrowAssertionErrorRejection = noop; + + const originalPromise = Promise; + const originalPromiseThen = Promise.prototype.then; + const originalPromiseResolve = Promise.resolve.bind(originalPromise); + const originalPromiseReject = Promise.reject.bind(originalPromise); + function newPromise(executor) { + return new originalPromise(executor); + } + function promiseResolvedWith(value) { + return originalPromiseResolve(value); + } + function promiseRejectedWith(reason) { + return originalPromiseReject(reason); + } + function PerformPromiseThen(promise, onFulfilled, onRejected) { + // There doesn't appear to be any way to correctly emulate the behaviour from JavaScript, so this is just an + // approximation. + return originalPromiseThen.call(promise, onFulfilled, onRejected); + } + function uponPromise(promise, onFulfilled, onRejected) { + PerformPromiseThen(PerformPromiseThen(promise, onFulfilled, onRejected), undefined, rethrowAssertionErrorRejection); + } + function uponFulfillment(promise, onFulfilled) { + uponPromise(promise, onFulfilled); + } + function uponRejection(promise, onRejected) { + uponPromise(promise, undefined, onRejected); + } + function transformPromiseWith(promise, fulfillmentHandler, rejectionHandler) { + return PerformPromiseThen(promise, fulfillmentHandler, rejectionHandler); + } + function setPromiseIsHandledToTrue(promise) { + PerformPromiseThen(promise, undefined, rethrowAssertionErrorRejection); + } + const queueMicrotask = (() => { + const globalQueueMicrotask = globals && globals.queueMicrotask; + if (typeof globalQueueMicrotask === 'function') { + return globalQueueMicrotask; + } + const resolvedPromise = promiseResolvedWith(undefined); + return (fn) => PerformPromiseThen(resolvedPromise, fn); + })(); + function reflectCall(F, V, args) { + if (typeof F !== 'function') { + throw new TypeError('Argument is not a function'); + } + return Function.prototype.apply.call(F, V, args); + } + function promiseCall(F, V, args) { + try { + return promiseResolvedWith(reflectCall(F, V, args)); + } + catch (value) { + return promiseRejectedWith(value); + } + } + + // Original from Chromium + // https://chromium.googlesource.com/chromium/src/+/0aee4434a4dba42a42abaea9bfbc0cd196a63bc1/third_party/blink/renderer/core/streams/SimpleQueue.js + const QUEUE_MAX_ARRAY_SIZE = 16384; + /** + * Simple queue structure. + * + * Avoids scalability issues with using a packed array directly by using + * multiple arrays in a linked list and keeping the array size bounded. + */ + class SimpleQueue { + constructor() { + this._cursor = 0; + this._size = 0; + // _front and _back are always defined. + this._front = { + _elements: [], + _next: undefined + }; + this._back = this._front; + // The cursor is used to avoid calling Array.shift(). + // It contains the index of the front element of the array inside the + // front-most node. It is always in the range [0, QUEUE_MAX_ARRAY_SIZE). + this._cursor = 0; + // When there is only one node, size === elements.length - cursor. + this._size = 0; + } + get length() { + return this._size; + } + // For exception safety, this method is structured in order: + // 1. Read state + // 2. Calculate required state mutations + // 3. Perform state mutations + push(element) { + const oldBack = this._back; + let newBack = oldBack; + if (oldBack._elements.length === QUEUE_MAX_ARRAY_SIZE - 1) { + newBack = { + _elements: [], + _next: undefined + }; + } + // push() is the mutation most likely to throw an exception, so it + // goes first. + oldBack._elements.push(element); + if (newBack !== oldBack) { + this._back = newBack; + oldBack._next = newBack; + } + ++this._size; + } + // Like push(), shift() follows the read -> calculate -> mutate pattern for + // exception safety. + shift() { // must not be called on an empty queue + const oldFront = this._front; + let newFront = oldFront; + const oldCursor = this._cursor; + let newCursor = oldCursor + 1; + const elements = oldFront._elements; + const element = elements[oldCursor]; + if (newCursor === QUEUE_MAX_ARRAY_SIZE) { + newFront = oldFront._next; + newCursor = 0; + } + // No mutations before this point. + --this._size; + this._cursor = newCursor; + if (oldFront !== newFront) { + this._front = newFront; + } + // Permit shifted element to be garbage collected. + elements[oldCursor] = undefined; + return element; + } + // The tricky thing about forEach() is that it can be called + // re-entrantly. The queue may be mutated inside the callback. It is easy to + // see that push() within the callback has no negative effects since the end + // of the queue is checked for on every iteration. If shift() is called + // repeatedly within the callback then the next iteration may return an + // element that has been removed. In this case the callback will be called + // with undefined values until we either "catch up" with elements that still + // exist or reach the back of the queue. + forEach(callback) { + let i = this._cursor; + let node = this._front; + let elements = node._elements; + while (i !== elements.length || node._next !== undefined) { + if (i === elements.length) { + node = node._next; + elements = node._elements; + i = 0; + if (elements.length === 0) { + break; + } + } + callback(elements[i]); + ++i; + } + } + // Return the element that would be returned if shift() was called now, + // without modifying the queue. + peek() { // must not be called on an empty queue + const front = this._front; + const cursor = this._cursor; + return front._elements[cursor]; + } + } + + function ReadableStreamReaderGenericInitialize(reader, stream) { + reader._ownerReadableStream = stream; + stream._reader = reader; + if (stream._state === 'readable') { + defaultReaderClosedPromiseInitialize(reader); + } + else if (stream._state === 'closed') { + defaultReaderClosedPromiseInitializeAsResolved(reader); + } + else { + defaultReaderClosedPromiseInitializeAsRejected(reader, stream._storedError); + } + } + // A client of ReadableStreamDefaultReader and ReadableStreamBYOBReader may use these functions directly to bypass state + // check. + function ReadableStreamReaderGenericCancel(reader, reason) { + const stream = reader._ownerReadableStream; + return ReadableStreamCancel(stream, reason); + } + function ReadableStreamReaderGenericRelease(reader) { + if (reader._ownerReadableStream._state === 'readable') { + defaultReaderClosedPromiseReject(reader, new TypeError(`Reader was released and can no longer be used to monitor the stream's closedness`)); + } + else { + defaultReaderClosedPromiseResetToRejected(reader, new TypeError(`Reader was released and can no longer be used to monitor the stream's closedness`)); + } + reader._ownerReadableStream._reader = undefined; + reader._ownerReadableStream = undefined; + } + // Helper functions for the readers. + function readerLockException(name) { + return new TypeError('Cannot ' + name + ' a stream using a released reader'); + } + // Helper functions for the ReadableStreamDefaultReader. + function defaultReaderClosedPromiseInitialize(reader) { + reader._closedPromise = newPromise((resolve, reject) => { + reader._closedPromise_resolve = resolve; + reader._closedPromise_reject = reject; + }); + } + function defaultReaderClosedPromiseInitializeAsRejected(reader, reason) { + defaultReaderClosedPromiseInitialize(reader); + defaultReaderClosedPromiseReject(reader, reason); + } + function defaultReaderClosedPromiseInitializeAsResolved(reader) { + defaultReaderClosedPromiseInitialize(reader); + defaultReaderClosedPromiseResolve(reader); + } + function defaultReaderClosedPromiseReject(reader, reason) { + if (reader._closedPromise_reject === undefined) { + return; + } + setPromiseIsHandledToTrue(reader._closedPromise); + reader._closedPromise_reject(reason); + reader._closedPromise_resolve = undefined; + reader._closedPromise_reject = undefined; + } + function defaultReaderClosedPromiseResetToRejected(reader, reason) { + defaultReaderClosedPromiseInitializeAsRejected(reader, reason); + } + function defaultReaderClosedPromiseResolve(reader) { + if (reader._closedPromise_resolve === undefined) { + return; + } + reader._closedPromise_resolve(undefined); + reader._closedPromise_resolve = undefined; + reader._closedPromise_reject = undefined; + } + + const AbortSteps = SymbolPolyfill('[[AbortSteps]]'); + const ErrorSteps = SymbolPolyfill('[[ErrorSteps]]'); + const CancelSteps = SymbolPolyfill('[[CancelSteps]]'); + const PullSteps = SymbolPolyfill('[[PullSteps]]'); + + /// + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isFinite#Polyfill + const NumberIsFinite = Number.isFinite || function (x) { + return typeof x === 'number' && isFinite(x); + }; + + /// + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math/trunc#Polyfill + const MathTrunc = Math.trunc || function (v) { + return v < 0 ? Math.ceil(v) : Math.floor(v); + }; + + // https://heycam.github.io/webidl/#idl-dictionaries + function isDictionary(x) { + return typeof x === 'object' || typeof x === 'function'; + } + function assertDictionary(obj, context) { + if (obj !== undefined && !isDictionary(obj)) { + throw new TypeError(`${context} is not an object.`); + } + } + // https://heycam.github.io/webidl/#idl-callback-functions + function assertFunction(x, context) { + if (typeof x !== 'function') { + throw new TypeError(`${context} is not a function.`); + } + } + // https://heycam.github.io/webidl/#idl-object + function isObject(x) { + return (typeof x === 'object' && x !== null) || typeof x === 'function'; + } + function assertObject(x, context) { + if (!isObject(x)) { + throw new TypeError(`${context} is not an object.`); + } + } + function assertRequiredArgument(x, position, context) { + if (x === undefined) { + throw new TypeError(`Parameter ${position} is required in '${context}'.`); + } + } + function assertRequiredField(x, field, context) { + if (x === undefined) { + throw new TypeError(`${field} is required in '${context}'.`); + } + } + // https://heycam.github.io/webidl/#idl-unrestricted-double + function convertUnrestrictedDouble(value) { + return Number(value); + } + function censorNegativeZero(x) { + return x === 0 ? 0 : x; + } + function integerPart(x) { + return censorNegativeZero(MathTrunc(x)); + } + // https://heycam.github.io/webidl/#idl-unsigned-long-long + function convertUnsignedLongLongWithEnforceRange(value, context) { + const lowerBound = 0; + const upperBound = Number.MAX_SAFE_INTEGER; + let x = Number(value); + x = censorNegativeZero(x); + if (!NumberIsFinite(x)) { + throw new TypeError(`${context} is not a finite number`); + } + x = integerPart(x); + if (x < lowerBound || x > upperBound) { + throw new TypeError(`${context} is outside the accepted range of ${lowerBound} to ${upperBound}, inclusive`); + } + if (!NumberIsFinite(x) || x === 0) { + return 0; + } + // TODO Use BigInt if supported? + // let xBigInt = BigInt(integerPart(x)); + // xBigInt = BigInt.asUintN(64, xBigInt); + // return Number(xBigInt); + return x; + } + + function assertReadableStream(x, context) { + if (!IsReadableStream(x)) { + throw new TypeError(`${context} is not a ReadableStream.`); + } + } + + // Abstract operations for the ReadableStream. + function AcquireReadableStreamDefaultReader(stream) { + return new ReadableStreamDefaultReader(stream); + } + // ReadableStream API exposed for controllers. + function ReadableStreamAddReadRequest(stream, readRequest) { + stream._reader._readRequests.push(readRequest); + } + function ReadableStreamFulfillReadRequest(stream, chunk, done) { + const reader = stream._reader; + const readRequest = reader._readRequests.shift(); + if (done) { + readRequest._closeSteps(); + } + else { + readRequest._chunkSteps(chunk); + } + } + function ReadableStreamGetNumReadRequests(stream) { + return stream._reader._readRequests.length; + } + function ReadableStreamHasDefaultReader(stream) { + const reader = stream._reader; + if (reader === undefined) { + return false; + } + if (!IsReadableStreamDefaultReader(reader)) { + return false; + } + return true; + } + /** + * A default reader vended by a {@link ReadableStream}. + * + * @public + */ + class ReadableStreamDefaultReader { + constructor(stream) { + assertRequiredArgument(stream, 1, 'ReadableStreamDefaultReader'); + assertReadableStream(stream, 'First parameter'); + if (IsReadableStreamLocked(stream)) { + throw new TypeError('This stream has already been locked for exclusive reading by another reader'); + } + ReadableStreamReaderGenericInitialize(this, stream); + this._readRequests = new SimpleQueue(); + } + /** + * Returns a promise that will be fulfilled when the stream becomes closed, + * or rejected if the stream ever errors or the reader's lock is released before the stream finishes closing. + */ + get closed() { + if (!IsReadableStreamDefaultReader(this)) { + return promiseRejectedWith(defaultReaderBrandCheckException('closed')); + } + return this._closedPromise; + } + /** + * If the reader is active, behaves the same as {@link ReadableStream.cancel | stream.cancel(reason)}. + */ + cancel(reason = undefined) { + if (!IsReadableStreamDefaultReader(this)) { + return promiseRejectedWith(defaultReaderBrandCheckException('cancel')); + } + if (this._ownerReadableStream === undefined) { + return promiseRejectedWith(readerLockException('cancel')); + } + return ReadableStreamReaderGenericCancel(this, reason); + } + /** + * Returns a promise that allows access to the next chunk from the stream's internal queue, if available. + * + * If reading a chunk causes the queue to become empty, more data will be pulled from the underlying source. + */ + read() { + if (!IsReadableStreamDefaultReader(this)) { + return promiseRejectedWith(defaultReaderBrandCheckException('read')); + } + if (this._ownerReadableStream === undefined) { + return promiseRejectedWith(readerLockException('read from')); + } + let resolvePromise; + let rejectPromise; + const promise = newPromise((resolve, reject) => { + resolvePromise = resolve; + rejectPromise = reject; + }); + const readRequest = { + _chunkSteps: chunk => resolvePromise({ value: chunk, done: false }), + _closeSteps: () => resolvePromise({ value: undefined, done: true }), + _errorSteps: e => rejectPromise(e) + }; + ReadableStreamDefaultReaderRead(this, readRequest); + return promise; + } + /** + * Releases the reader's lock on the corresponding stream. After the lock is released, the reader is no longer active. + * If the associated stream is errored when the lock is released, the reader will appear errored in the same way + * from now on; otherwise, the reader will appear closed. + * + * A reader's lock cannot be released while it still has a pending read request, i.e., if a promise returned by + * the reader's {@link ReadableStreamDefaultReader.read | read()} method has not yet been settled. Attempting to + * do so will throw a `TypeError` and leave the reader locked to the stream. + */ + releaseLock() { + if (!IsReadableStreamDefaultReader(this)) { + throw defaultReaderBrandCheckException('releaseLock'); + } + if (this._ownerReadableStream === undefined) { + return; + } + if (this._readRequests.length > 0) { + throw new TypeError('Tried to release a reader lock when that reader has pending read() calls un-settled'); + } + ReadableStreamReaderGenericRelease(this); + } + } + Object.defineProperties(ReadableStreamDefaultReader.prototype, { + cancel: { enumerable: true }, + read: { enumerable: true }, + releaseLock: { enumerable: true }, + closed: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(ReadableStreamDefaultReader.prototype, SymbolPolyfill.toStringTag, { + value: 'ReadableStreamDefaultReader', + configurable: true + }); + } + // Abstract operations for the readers. + function IsReadableStreamDefaultReader(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_readRequests')) { + return false; + } + return true; + } + function ReadableStreamDefaultReaderRead(reader, readRequest) { + const stream = reader._ownerReadableStream; + stream._disturbed = true; + if (stream._state === 'closed') { + readRequest._closeSteps(); + } + else if (stream._state === 'errored') { + readRequest._errorSteps(stream._storedError); + } + else { + stream._readableStreamController[PullSteps](readRequest); + } + } + // Helper functions for the ReadableStreamDefaultReader. + function defaultReaderBrandCheckException(name) { + return new TypeError(`ReadableStreamDefaultReader.prototype.${name} can only be used on a ReadableStreamDefaultReader`); + } + + /// + let AsyncIteratorPrototype; + if (typeof SymbolPolyfill.asyncIterator === 'symbol') { + // We're running inside a ES2018+ environment, but we're compiling to an older syntax. + // We cannot access %AsyncIteratorPrototype% without non-ES2018 syntax, but we can re-create it. + AsyncIteratorPrototype = { + // 25.1.3.1 %AsyncIteratorPrototype% [ @@asyncIterator ] ( ) + // https://tc39.github.io/ecma262/#sec-asynciteratorprototype-asynciterator + [SymbolPolyfill.asyncIterator]() { + return this; + } + }; + Object.defineProperty(AsyncIteratorPrototype, SymbolPolyfill.asyncIterator, { enumerable: false }); + } + + /// + class ReadableStreamAsyncIteratorImpl { + constructor(reader, preventCancel) { + this._ongoingPromise = undefined; + this._isFinished = false; + this._reader = reader; + this._preventCancel = preventCancel; + } + next() { + const nextSteps = () => this._nextSteps(); + this._ongoingPromise = this._ongoingPromise ? + transformPromiseWith(this._ongoingPromise, nextSteps, nextSteps) : + nextSteps(); + return this._ongoingPromise; + } + return(value) { + const returnSteps = () => this._returnSteps(value); + return this._ongoingPromise ? + transformPromiseWith(this._ongoingPromise, returnSteps, returnSteps) : + returnSteps(); + } + _nextSteps() { + if (this._isFinished) { + return Promise.resolve({ value: undefined, done: true }); + } + const reader = this._reader; + if (reader._ownerReadableStream === undefined) { + return promiseRejectedWith(readerLockException('iterate')); + } + let resolvePromise; + let rejectPromise; + const promise = newPromise((resolve, reject) => { + resolvePromise = resolve; + rejectPromise = reject; + }); + const readRequest = { + _chunkSteps: chunk => { + this._ongoingPromise = undefined; + // This needs to be delayed by one microtask, otherwise we stop pulling too early which breaks a test. + // FIXME Is this a bug in the specification, or in the test? + queueMicrotask(() => resolvePromise({ value: chunk, done: false })); + }, + _closeSteps: () => { + this._ongoingPromise = undefined; + this._isFinished = true; + ReadableStreamReaderGenericRelease(reader); + resolvePromise({ value: undefined, done: true }); + }, + _errorSteps: reason => { + this._ongoingPromise = undefined; + this._isFinished = true; + ReadableStreamReaderGenericRelease(reader); + rejectPromise(reason); + } + }; + ReadableStreamDefaultReaderRead(reader, readRequest); + return promise; + } + _returnSteps(value) { + if (this._isFinished) { + return Promise.resolve({ value, done: true }); + } + this._isFinished = true; + const reader = this._reader; + if (reader._ownerReadableStream === undefined) { + return promiseRejectedWith(readerLockException('finish iterating')); + } + if (!this._preventCancel) { + const result = ReadableStreamReaderGenericCancel(reader, value); + ReadableStreamReaderGenericRelease(reader); + return transformPromiseWith(result, () => ({ value, done: true })); + } + ReadableStreamReaderGenericRelease(reader); + return promiseResolvedWith({ value, done: true }); + } + } + const ReadableStreamAsyncIteratorPrototype = { + next() { + if (!IsReadableStreamAsyncIterator(this)) { + return promiseRejectedWith(streamAsyncIteratorBrandCheckException('next')); + } + return this._asyncIteratorImpl.next(); + }, + return(value) { + if (!IsReadableStreamAsyncIterator(this)) { + return promiseRejectedWith(streamAsyncIteratorBrandCheckException('return')); + } + return this._asyncIteratorImpl.return(value); + } + }; + if (AsyncIteratorPrototype !== undefined) { + Object.setPrototypeOf(ReadableStreamAsyncIteratorPrototype, AsyncIteratorPrototype); + } + // Abstract operations for the ReadableStream. + function AcquireReadableStreamAsyncIterator(stream, preventCancel) { + const reader = AcquireReadableStreamDefaultReader(stream); + const impl = new ReadableStreamAsyncIteratorImpl(reader, preventCancel); + const iterator = Object.create(ReadableStreamAsyncIteratorPrototype); + iterator._asyncIteratorImpl = impl; + return iterator; + } + function IsReadableStreamAsyncIterator(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_asyncIteratorImpl')) { + return false; + } + return true; + } + // Helper functions for the ReadableStream. + function streamAsyncIteratorBrandCheckException(name) { + return new TypeError(`ReadableStreamAsyncIterator.${name} can only be used on a ReadableSteamAsyncIterator`); + } + + /// + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isNaN#Polyfill + const NumberIsNaN = Number.isNaN || function (x) { + // eslint-disable-next-line no-self-compare + return x !== x; + }; + + function IsFiniteNonNegativeNumber(v) { + if (!IsNonNegativeNumber(v)) { + return false; + } + if (v === Infinity) { + return false; + } + return true; + } + function IsNonNegativeNumber(v) { + if (typeof v !== 'number') { + return false; + } + if (NumberIsNaN(v)) { + return false; + } + if (v < 0) { + return false; + } + return true; + } + + function DequeueValue(container) { + const pair = container._queue.shift(); + container._queueTotalSize -= pair.size; + if (container._queueTotalSize < 0) { + container._queueTotalSize = 0; + } + return pair.value; + } + function EnqueueValueWithSize(container, value, size) { + size = Number(size); + if (!IsFiniteNonNegativeNumber(size)) { + throw new RangeError('Size must be a finite, non-NaN, non-negative number.'); + } + container._queue.push({ value, size }); + container._queueTotalSize += size; + } + function PeekQueueValue(container) { + const pair = container._queue.peek(); + return pair.value; + } + function ResetQueue(container) { + container._queue = new SimpleQueue(); + container._queueTotalSize = 0; + } + + function CreateArrayFromList(elements) { + // We use arrays to represent lists, so this is basically a no-op. + // Do a slice though just in case we happen to depend on the unique-ness. + return elements.slice(); + } + function CopyDataBlockBytes(dest, destOffset, src, srcOffset, n) { + new Uint8Array(dest).set(new Uint8Array(src, srcOffset, n), destOffset); + } + // Not implemented correctly + function TransferArrayBuffer(O) { + return O; + } + // Not implemented correctly + function IsDetachedBuffer(O) { + return false; + } + + /** + * A pull-into request in a {@link ReadableByteStreamController}. + * + * @public + */ + class ReadableStreamBYOBRequest { + constructor() { + throw new TypeError('Illegal constructor'); + } + /** + * Returns the view for writing in to, or `null` if the BYOB request has already been responded to. + */ + get view() { + if (!IsReadableStreamBYOBRequest(this)) { + throw byobRequestBrandCheckException('view'); + } + return this._view; + } + respond(bytesWritten) { + if (!IsReadableStreamBYOBRequest(this)) { + throw byobRequestBrandCheckException('respond'); + } + assertRequiredArgument(bytesWritten, 1, 'respond'); + bytesWritten = convertUnsignedLongLongWithEnforceRange(bytesWritten, 'First parameter'); + if (this._associatedReadableByteStreamController === undefined) { + throw new TypeError('This BYOB request has been invalidated'); + } + if (IsDetachedBuffer(this._view.buffer)) ; + ReadableByteStreamControllerRespond(this._associatedReadableByteStreamController, bytesWritten); + } + respondWithNewView(view) { + if (!IsReadableStreamBYOBRequest(this)) { + throw byobRequestBrandCheckException('respondWithNewView'); + } + assertRequiredArgument(view, 1, 'respondWithNewView'); + if (!ArrayBuffer.isView(view)) { + throw new TypeError('You can only respond with array buffer views'); + } + if (view.byteLength === 0) { + throw new TypeError('chunk must have non-zero byteLength'); + } + if (view.buffer.byteLength === 0) { + throw new TypeError(`chunk's buffer must have non-zero byteLength`); + } + if (this._associatedReadableByteStreamController === undefined) { + throw new TypeError('This BYOB request has been invalidated'); + } + ReadableByteStreamControllerRespondWithNewView(this._associatedReadableByteStreamController, view); + } + } + Object.defineProperties(ReadableStreamBYOBRequest.prototype, { + respond: { enumerable: true }, + respondWithNewView: { enumerable: true }, + view: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(ReadableStreamBYOBRequest.prototype, SymbolPolyfill.toStringTag, { + value: 'ReadableStreamBYOBRequest', + configurable: true + }); + } + /** + * Allows control of a {@link ReadableStream | readable byte stream}'s state and internal queue. + * + * @public + */ + class ReadableByteStreamController { + constructor() { + throw new TypeError('Illegal constructor'); + } + /** + * Returns the current BYOB pull request, or `null` if there isn't one. + */ + get byobRequest() { + if (!IsReadableByteStreamController(this)) { + throw byteStreamControllerBrandCheckException('byobRequest'); + } + if (this._byobRequest === null && this._pendingPullIntos.length > 0) { + const firstDescriptor = this._pendingPullIntos.peek(); + const view = new Uint8Array(firstDescriptor.buffer, firstDescriptor.byteOffset + firstDescriptor.bytesFilled, firstDescriptor.byteLength - firstDescriptor.bytesFilled); + const byobRequest = Object.create(ReadableStreamBYOBRequest.prototype); + SetUpReadableStreamBYOBRequest(byobRequest, this, view); + this._byobRequest = byobRequest; + } + return this._byobRequest; + } + /** + * Returns the desired size to fill the controlled stream's internal queue. It can be negative, if the queue is + * over-full. An underlying byte source ought to use this information to determine when and how to apply backpressure. + */ + get desiredSize() { + if (!IsReadableByteStreamController(this)) { + throw byteStreamControllerBrandCheckException('desiredSize'); + } + return ReadableByteStreamControllerGetDesiredSize(this); + } + /** + * Closes the controlled readable stream. Consumers will still be able to read any previously-enqueued chunks from + * the stream, but once those are read, the stream will become closed. + */ + close() { + if (!IsReadableByteStreamController(this)) { + throw byteStreamControllerBrandCheckException('close'); + } + if (this._closeRequested) { + throw new TypeError('The stream has already been closed; do not close it again!'); + } + const state = this._controlledReadableByteStream._state; + if (state !== 'readable') { + throw new TypeError(`The stream (in ${state} state) is not in the readable state and cannot be closed`); + } + ReadableByteStreamControllerClose(this); + } + enqueue(chunk) { + if (!IsReadableByteStreamController(this)) { + throw byteStreamControllerBrandCheckException('enqueue'); + } + assertRequiredArgument(chunk, 1, 'enqueue'); + if (!ArrayBuffer.isView(chunk)) { + throw new TypeError('chunk must be an array buffer view'); + } + if (chunk.byteLength === 0) { + throw new TypeError('chunk must have non-zero byteLength'); + } + if (chunk.buffer.byteLength === 0) { + throw new TypeError(`chunk's buffer must have non-zero byteLength`); + } + if (this._closeRequested) { + throw new TypeError('stream is closed or draining'); + } + const state = this._controlledReadableByteStream._state; + if (state !== 'readable') { + throw new TypeError(`The stream (in ${state} state) is not in the readable state and cannot be enqueued to`); + } + ReadableByteStreamControllerEnqueue(this, chunk); + } + /** + * Errors the controlled readable stream, making all future interactions with it fail with the given error `e`. + */ + error(e = undefined) { + if (!IsReadableByteStreamController(this)) { + throw byteStreamControllerBrandCheckException('error'); + } + ReadableByteStreamControllerError(this, e); + } + /** @internal */ + [CancelSteps](reason) { + if (this._pendingPullIntos.length > 0) { + const firstDescriptor = this._pendingPullIntos.peek(); + firstDescriptor.bytesFilled = 0; + } + ResetQueue(this); + const result = this._cancelAlgorithm(reason); + ReadableByteStreamControllerClearAlgorithms(this); + return result; + } + /** @internal */ + [PullSteps](readRequest) { + const stream = this._controlledReadableByteStream; + if (this._queueTotalSize > 0) { + const entry = this._queue.shift(); + this._queueTotalSize -= entry.byteLength; + ReadableByteStreamControllerHandleQueueDrain(this); + const view = new Uint8Array(entry.buffer, entry.byteOffset, entry.byteLength); + readRequest._chunkSteps(view); + return; + } + const autoAllocateChunkSize = this._autoAllocateChunkSize; + if (autoAllocateChunkSize !== undefined) { + let buffer; + try { + buffer = new ArrayBuffer(autoAllocateChunkSize); + } + catch (bufferE) { + readRequest._errorSteps(bufferE); + return; + } + const pullIntoDescriptor = { + buffer, + byteOffset: 0, + byteLength: autoAllocateChunkSize, + bytesFilled: 0, + elementSize: 1, + viewConstructor: Uint8Array, + readerType: 'default' + }; + this._pendingPullIntos.push(pullIntoDescriptor); + } + ReadableStreamAddReadRequest(stream, readRequest); + ReadableByteStreamControllerCallPullIfNeeded(this); + } + } + Object.defineProperties(ReadableByteStreamController.prototype, { + close: { enumerable: true }, + enqueue: { enumerable: true }, + error: { enumerable: true }, + byobRequest: { enumerable: true }, + desiredSize: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(ReadableByteStreamController.prototype, SymbolPolyfill.toStringTag, { + value: 'ReadableByteStreamController', + configurable: true + }); + } + // Abstract operations for the ReadableByteStreamController. + function IsReadableByteStreamController(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_controlledReadableByteStream')) { + return false; + } + return true; + } + function IsReadableStreamBYOBRequest(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_associatedReadableByteStreamController')) { + return false; + } + return true; + } + function ReadableByteStreamControllerCallPullIfNeeded(controller) { + const shouldPull = ReadableByteStreamControllerShouldCallPull(controller); + if (!shouldPull) { + return; + } + if (controller._pulling) { + controller._pullAgain = true; + return; + } + controller._pulling = true; + // TODO: Test controller argument + const pullPromise = controller._pullAlgorithm(); + uponPromise(pullPromise, () => { + controller._pulling = false; + if (controller._pullAgain) { + controller._pullAgain = false; + ReadableByteStreamControllerCallPullIfNeeded(controller); + } + }, e => { + ReadableByteStreamControllerError(controller, e); + }); + } + function ReadableByteStreamControllerClearPendingPullIntos(controller) { + ReadableByteStreamControllerInvalidateBYOBRequest(controller); + controller._pendingPullIntos = new SimpleQueue(); + } + function ReadableByteStreamControllerCommitPullIntoDescriptor(stream, pullIntoDescriptor) { + let done = false; + if (stream._state === 'closed') { + done = true; + } + const filledView = ReadableByteStreamControllerConvertPullIntoDescriptor(pullIntoDescriptor); + if (pullIntoDescriptor.readerType === 'default') { + ReadableStreamFulfillReadRequest(stream, filledView, done); + } + else { + ReadableStreamFulfillReadIntoRequest(stream, filledView, done); + } + } + function ReadableByteStreamControllerConvertPullIntoDescriptor(pullIntoDescriptor) { + const bytesFilled = pullIntoDescriptor.bytesFilled; + const elementSize = pullIntoDescriptor.elementSize; + return new pullIntoDescriptor.viewConstructor(pullIntoDescriptor.buffer, pullIntoDescriptor.byteOffset, bytesFilled / elementSize); + } + function ReadableByteStreamControllerEnqueueChunkToQueue(controller, buffer, byteOffset, byteLength) { + controller._queue.push({ buffer, byteOffset, byteLength }); + controller._queueTotalSize += byteLength; + } + function ReadableByteStreamControllerFillPullIntoDescriptorFromQueue(controller, pullIntoDescriptor) { + const elementSize = pullIntoDescriptor.elementSize; + const currentAlignedBytes = pullIntoDescriptor.bytesFilled - pullIntoDescriptor.bytesFilled % elementSize; + const maxBytesToCopy = Math.min(controller._queueTotalSize, pullIntoDescriptor.byteLength - pullIntoDescriptor.bytesFilled); + const maxBytesFilled = pullIntoDescriptor.bytesFilled + maxBytesToCopy; + const maxAlignedBytes = maxBytesFilled - maxBytesFilled % elementSize; + let totalBytesToCopyRemaining = maxBytesToCopy; + let ready = false; + if (maxAlignedBytes > currentAlignedBytes) { + totalBytesToCopyRemaining = maxAlignedBytes - pullIntoDescriptor.bytesFilled; + ready = true; + } + const queue = controller._queue; + while (totalBytesToCopyRemaining > 0) { + const headOfQueue = queue.peek(); + const bytesToCopy = Math.min(totalBytesToCopyRemaining, headOfQueue.byteLength); + const destStart = pullIntoDescriptor.byteOffset + pullIntoDescriptor.bytesFilled; + CopyDataBlockBytes(pullIntoDescriptor.buffer, destStart, headOfQueue.buffer, headOfQueue.byteOffset, bytesToCopy); + if (headOfQueue.byteLength === bytesToCopy) { + queue.shift(); + } + else { + headOfQueue.byteOffset += bytesToCopy; + headOfQueue.byteLength -= bytesToCopy; + } + controller._queueTotalSize -= bytesToCopy; + ReadableByteStreamControllerFillHeadPullIntoDescriptor(controller, bytesToCopy, pullIntoDescriptor); + totalBytesToCopyRemaining -= bytesToCopy; + } + return ready; + } + function ReadableByteStreamControllerFillHeadPullIntoDescriptor(controller, size, pullIntoDescriptor) { + ReadableByteStreamControllerInvalidateBYOBRequest(controller); + pullIntoDescriptor.bytesFilled += size; + } + function ReadableByteStreamControllerHandleQueueDrain(controller) { + if (controller._queueTotalSize === 0 && controller._closeRequested) { + ReadableByteStreamControllerClearAlgorithms(controller); + ReadableStreamClose(controller._controlledReadableByteStream); + } + else { + ReadableByteStreamControllerCallPullIfNeeded(controller); + } + } + function ReadableByteStreamControllerInvalidateBYOBRequest(controller) { + if (controller._byobRequest === null) { + return; + } + controller._byobRequest._associatedReadableByteStreamController = undefined; + controller._byobRequest._view = null; + controller._byobRequest = null; + } + function ReadableByteStreamControllerProcessPullIntoDescriptorsUsingQueue(controller) { + while (controller._pendingPullIntos.length > 0) { + if (controller._queueTotalSize === 0) { + return; + } + const pullIntoDescriptor = controller._pendingPullIntos.peek(); + if (ReadableByteStreamControllerFillPullIntoDescriptorFromQueue(controller, pullIntoDescriptor)) { + ReadableByteStreamControllerShiftPendingPullInto(controller); + ReadableByteStreamControllerCommitPullIntoDescriptor(controller._controlledReadableByteStream, pullIntoDescriptor); + } + } + } + function ReadableByteStreamControllerPullInto(controller, view, readIntoRequest) { + const stream = controller._controlledReadableByteStream; + let elementSize = 1; + if (view.constructor !== DataView) { + elementSize = view.constructor.BYTES_PER_ELEMENT; + } + const ctor = view.constructor; + const buffer = TransferArrayBuffer(view.buffer); + const pullIntoDescriptor = { + buffer, + byteOffset: view.byteOffset, + byteLength: view.byteLength, + bytesFilled: 0, + elementSize, + viewConstructor: ctor, + readerType: 'byob' + }; + if (controller._pendingPullIntos.length > 0) { + controller._pendingPullIntos.push(pullIntoDescriptor); + // No ReadableByteStreamControllerCallPullIfNeeded() call since: + // - No change happens on desiredSize + // - The source has already been notified of that there's at least 1 pending read(view) + ReadableStreamAddReadIntoRequest(stream, readIntoRequest); + return; + } + if (stream._state === 'closed') { + const emptyView = new ctor(pullIntoDescriptor.buffer, pullIntoDescriptor.byteOffset, 0); + readIntoRequest._closeSteps(emptyView); + return; + } + if (controller._queueTotalSize > 0) { + if (ReadableByteStreamControllerFillPullIntoDescriptorFromQueue(controller, pullIntoDescriptor)) { + const filledView = ReadableByteStreamControllerConvertPullIntoDescriptor(pullIntoDescriptor); + ReadableByteStreamControllerHandleQueueDrain(controller); + readIntoRequest._chunkSteps(filledView); + return; + } + if (controller._closeRequested) { + const e = new TypeError('Insufficient bytes to fill elements in the given buffer'); + ReadableByteStreamControllerError(controller, e); + readIntoRequest._errorSteps(e); + return; + } + } + controller._pendingPullIntos.push(pullIntoDescriptor); + ReadableStreamAddReadIntoRequest(stream, readIntoRequest); + ReadableByteStreamControllerCallPullIfNeeded(controller); + } + function ReadableByteStreamControllerRespondInClosedState(controller, firstDescriptor) { + firstDescriptor.buffer = TransferArrayBuffer(firstDescriptor.buffer); + const stream = controller._controlledReadableByteStream; + if (ReadableStreamHasBYOBReader(stream)) { + while (ReadableStreamGetNumReadIntoRequests(stream) > 0) { + const pullIntoDescriptor = ReadableByteStreamControllerShiftPendingPullInto(controller); + ReadableByteStreamControllerCommitPullIntoDescriptor(stream, pullIntoDescriptor); + } + } + } + function ReadableByteStreamControllerRespondInReadableState(controller, bytesWritten, pullIntoDescriptor) { + if (pullIntoDescriptor.bytesFilled + bytesWritten > pullIntoDescriptor.byteLength) { + throw new RangeError('bytesWritten out of range'); + } + ReadableByteStreamControllerFillHeadPullIntoDescriptor(controller, bytesWritten, pullIntoDescriptor); + if (pullIntoDescriptor.bytesFilled < pullIntoDescriptor.elementSize) { + // TODO: Figure out whether we should detach the buffer or not here. + return; + } + ReadableByteStreamControllerShiftPendingPullInto(controller); + const remainderSize = pullIntoDescriptor.bytesFilled % pullIntoDescriptor.elementSize; + if (remainderSize > 0) { + const end = pullIntoDescriptor.byteOffset + pullIntoDescriptor.bytesFilled; + const remainder = pullIntoDescriptor.buffer.slice(end - remainderSize, end); + ReadableByteStreamControllerEnqueueChunkToQueue(controller, remainder, 0, remainder.byteLength); + } + pullIntoDescriptor.buffer = TransferArrayBuffer(pullIntoDescriptor.buffer); + pullIntoDescriptor.bytesFilled -= remainderSize; + ReadableByteStreamControllerCommitPullIntoDescriptor(controller._controlledReadableByteStream, pullIntoDescriptor); + ReadableByteStreamControllerProcessPullIntoDescriptorsUsingQueue(controller); + } + function ReadableByteStreamControllerRespondInternal(controller, bytesWritten) { + const firstDescriptor = controller._pendingPullIntos.peek(); + const state = controller._controlledReadableByteStream._state; + if (state === 'closed') { + if (bytesWritten !== 0) { + throw new TypeError('bytesWritten must be 0 when calling respond() on a closed stream'); + } + ReadableByteStreamControllerRespondInClosedState(controller, firstDescriptor); + } + else { + ReadableByteStreamControllerRespondInReadableState(controller, bytesWritten, firstDescriptor); + } + ReadableByteStreamControllerCallPullIfNeeded(controller); + } + function ReadableByteStreamControllerShiftPendingPullInto(controller) { + const descriptor = controller._pendingPullIntos.shift(); + ReadableByteStreamControllerInvalidateBYOBRequest(controller); + return descriptor; + } + function ReadableByteStreamControllerShouldCallPull(controller) { + const stream = controller._controlledReadableByteStream; + if (stream._state !== 'readable') { + return false; + } + if (controller._closeRequested) { + return false; + } + if (!controller._started) { + return false; + } + if (ReadableStreamHasDefaultReader(stream) && ReadableStreamGetNumReadRequests(stream) > 0) { + return true; + } + if (ReadableStreamHasBYOBReader(stream) && ReadableStreamGetNumReadIntoRequests(stream) > 0) { + return true; + } + const desiredSize = ReadableByteStreamControllerGetDesiredSize(controller); + if (desiredSize > 0) { + return true; + } + return false; + } + function ReadableByteStreamControllerClearAlgorithms(controller) { + controller._pullAlgorithm = undefined; + controller._cancelAlgorithm = undefined; + } + // A client of ReadableByteStreamController may use these functions directly to bypass state check. + function ReadableByteStreamControllerClose(controller) { + const stream = controller._controlledReadableByteStream; + if (controller._closeRequested || stream._state !== 'readable') { + return; + } + if (controller._queueTotalSize > 0) { + controller._closeRequested = true; + return; + } + if (controller._pendingPullIntos.length > 0) { + const firstPendingPullInto = controller._pendingPullIntos.peek(); + if (firstPendingPullInto.bytesFilled > 0) { + const e = new TypeError('Insufficient bytes to fill elements in the given buffer'); + ReadableByteStreamControllerError(controller, e); + throw e; + } + } + ReadableByteStreamControllerClearAlgorithms(controller); + ReadableStreamClose(stream); + } + function ReadableByteStreamControllerEnqueue(controller, chunk) { + const stream = controller._controlledReadableByteStream; + if (controller._closeRequested || stream._state !== 'readable') { + return; + } + const buffer = chunk.buffer; + const byteOffset = chunk.byteOffset; + const byteLength = chunk.byteLength; + const transferredBuffer = TransferArrayBuffer(buffer); + if (ReadableStreamHasDefaultReader(stream)) { + if (ReadableStreamGetNumReadRequests(stream) === 0) { + ReadableByteStreamControllerEnqueueChunkToQueue(controller, transferredBuffer, byteOffset, byteLength); + } + else { + const transferredView = new Uint8Array(transferredBuffer, byteOffset, byteLength); + ReadableStreamFulfillReadRequest(stream, transferredView, false); + } + } + else if (ReadableStreamHasBYOBReader(stream)) { + // TODO: Ideally in this branch detaching should happen only if the buffer is not consumed fully. + ReadableByteStreamControllerEnqueueChunkToQueue(controller, transferredBuffer, byteOffset, byteLength); + ReadableByteStreamControllerProcessPullIntoDescriptorsUsingQueue(controller); + } + else { + ReadableByteStreamControllerEnqueueChunkToQueue(controller, transferredBuffer, byteOffset, byteLength); + } + ReadableByteStreamControllerCallPullIfNeeded(controller); + } + function ReadableByteStreamControllerError(controller, e) { + const stream = controller._controlledReadableByteStream; + if (stream._state !== 'readable') { + return; + } + ReadableByteStreamControllerClearPendingPullIntos(controller); + ResetQueue(controller); + ReadableByteStreamControllerClearAlgorithms(controller); + ReadableStreamError(stream, e); + } + function ReadableByteStreamControllerGetDesiredSize(controller) { + const state = controller._controlledReadableByteStream._state; + if (state === 'errored') { + return null; + } + if (state === 'closed') { + return 0; + } + return controller._strategyHWM - controller._queueTotalSize; + } + function ReadableByteStreamControllerRespond(controller, bytesWritten) { + bytesWritten = Number(bytesWritten); + if (!IsFiniteNonNegativeNumber(bytesWritten)) { + throw new RangeError('bytesWritten must be a finite'); + } + ReadableByteStreamControllerRespondInternal(controller, bytesWritten); + } + function ReadableByteStreamControllerRespondWithNewView(controller, view) { + const firstDescriptor = controller._pendingPullIntos.peek(); + if (firstDescriptor.byteOffset + firstDescriptor.bytesFilled !== view.byteOffset) { + throw new RangeError('The region specified by view does not match byobRequest'); + } + if (firstDescriptor.byteLength !== view.byteLength) { + throw new RangeError('The buffer of view has different capacity than byobRequest'); + } + firstDescriptor.buffer = view.buffer; + ReadableByteStreamControllerRespondInternal(controller, view.byteLength); + } + function SetUpReadableByteStreamController(stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, autoAllocateChunkSize) { + controller._controlledReadableByteStream = stream; + controller._pullAgain = false; + controller._pulling = false; + controller._byobRequest = null; + // Need to set the slots so that the assert doesn't fire. In the spec the slots already exist implicitly. + controller._queue = controller._queueTotalSize = undefined; + ResetQueue(controller); + controller._closeRequested = false; + controller._started = false; + controller._strategyHWM = highWaterMark; + controller._pullAlgorithm = pullAlgorithm; + controller._cancelAlgorithm = cancelAlgorithm; + controller._autoAllocateChunkSize = autoAllocateChunkSize; + controller._pendingPullIntos = new SimpleQueue(); + stream._readableStreamController = controller; + const startResult = startAlgorithm(); + uponPromise(promiseResolvedWith(startResult), () => { + controller._started = true; + ReadableByteStreamControllerCallPullIfNeeded(controller); + }, r => { + ReadableByteStreamControllerError(controller, r); + }); + } + function SetUpReadableByteStreamControllerFromUnderlyingSource(stream, underlyingByteSource, highWaterMark) { + const controller = Object.create(ReadableByteStreamController.prototype); + let startAlgorithm = () => undefined; + let pullAlgorithm = () => promiseResolvedWith(undefined); + let cancelAlgorithm = () => promiseResolvedWith(undefined); + if (underlyingByteSource.start !== undefined) { + startAlgorithm = () => underlyingByteSource.start(controller); + } + if (underlyingByteSource.pull !== undefined) { + pullAlgorithm = () => underlyingByteSource.pull(controller); + } + if (underlyingByteSource.cancel !== undefined) { + cancelAlgorithm = reason => underlyingByteSource.cancel(reason); + } + const autoAllocateChunkSize = underlyingByteSource.autoAllocateChunkSize; + if (autoAllocateChunkSize === 0) { + throw new TypeError('autoAllocateChunkSize must be greater than 0'); + } + SetUpReadableByteStreamController(stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, autoAllocateChunkSize); + } + function SetUpReadableStreamBYOBRequest(request, controller, view) { + request._associatedReadableByteStreamController = controller; + request._view = view; + } + // Helper functions for the ReadableStreamBYOBRequest. + function byobRequestBrandCheckException(name) { + return new TypeError(`ReadableStreamBYOBRequest.prototype.${name} can only be used on a ReadableStreamBYOBRequest`); + } + // Helper functions for the ReadableByteStreamController. + function byteStreamControllerBrandCheckException(name) { + return new TypeError(`ReadableByteStreamController.prototype.${name} can only be used on a ReadableByteStreamController`); + } + + // Abstract operations for the ReadableStream. + function AcquireReadableStreamBYOBReader(stream) { + return new ReadableStreamBYOBReader(stream); + } + // ReadableStream API exposed for controllers. + function ReadableStreamAddReadIntoRequest(stream, readIntoRequest) { + stream._reader._readIntoRequests.push(readIntoRequest); + } + function ReadableStreamFulfillReadIntoRequest(stream, chunk, done) { + const reader = stream._reader; + const readIntoRequest = reader._readIntoRequests.shift(); + if (done) { + readIntoRequest._closeSteps(chunk); + } + else { + readIntoRequest._chunkSteps(chunk); + } + } + function ReadableStreamGetNumReadIntoRequests(stream) { + return stream._reader._readIntoRequests.length; + } + function ReadableStreamHasBYOBReader(stream) { + const reader = stream._reader; + if (reader === undefined) { + return false; + } + if (!IsReadableStreamBYOBReader(reader)) { + return false; + } + return true; + } + /** + * A BYOB reader vended by a {@link ReadableStream}. + * + * @public + */ + class ReadableStreamBYOBReader { + constructor(stream) { + assertRequiredArgument(stream, 1, 'ReadableStreamBYOBReader'); + assertReadableStream(stream, 'First parameter'); + if (IsReadableStreamLocked(stream)) { + throw new TypeError('This stream has already been locked for exclusive reading by another reader'); + } + if (!IsReadableByteStreamController(stream._readableStreamController)) { + throw new TypeError('Cannot construct a ReadableStreamBYOBReader for a stream not constructed with a byte ' + + 'source'); + } + ReadableStreamReaderGenericInitialize(this, stream); + this._readIntoRequests = new SimpleQueue(); + } + /** + * Returns a promise that will be fulfilled when the stream becomes closed, or rejected if the stream ever errors or + * the reader's lock is released before the stream finishes closing. + */ + get closed() { + if (!IsReadableStreamBYOBReader(this)) { + return promiseRejectedWith(byobReaderBrandCheckException('closed')); + } + return this._closedPromise; + } + /** + * If the reader is active, behaves the same as {@link ReadableStream.cancel | stream.cancel(reason)}. + */ + cancel(reason = undefined) { + if (!IsReadableStreamBYOBReader(this)) { + return promiseRejectedWith(byobReaderBrandCheckException('cancel')); + } + if (this._ownerReadableStream === undefined) { + return promiseRejectedWith(readerLockException('cancel')); + } + return ReadableStreamReaderGenericCancel(this, reason); + } + /** + * Attempts to reads bytes into view, and returns a promise resolved with the result. + * + * If reading a chunk causes the queue to become empty, more data will be pulled from the underlying source. + */ + read(view) { + if (!IsReadableStreamBYOBReader(this)) { + return promiseRejectedWith(byobReaderBrandCheckException('read')); + } + if (!ArrayBuffer.isView(view)) { + return promiseRejectedWith(new TypeError('view must be an array buffer view')); + } + if (view.byteLength === 0) { + return promiseRejectedWith(new TypeError('view must have non-zero byteLength')); + } + if (view.buffer.byteLength === 0) { + return promiseRejectedWith(new TypeError(`view's buffer must have non-zero byteLength`)); + } + if (this._ownerReadableStream === undefined) { + return promiseRejectedWith(readerLockException('read from')); + } + let resolvePromise; + let rejectPromise; + const promise = newPromise((resolve, reject) => { + resolvePromise = resolve; + rejectPromise = reject; + }); + const readIntoRequest = { + _chunkSteps: chunk => resolvePromise({ value: chunk, done: false }), + _closeSteps: chunk => resolvePromise({ value: chunk, done: true }), + _errorSteps: e => rejectPromise(e) + }; + ReadableStreamBYOBReaderRead(this, view, readIntoRequest); + return promise; + } + /** + * Releases the reader's lock on the corresponding stream. After the lock is released, the reader is no longer active. + * If the associated stream is errored when the lock is released, the reader will appear errored in the same way + * from now on; otherwise, the reader will appear closed. + * + * A reader's lock cannot be released while it still has a pending read request, i.e., if a promise returned by + * the reader's {@link ReadableStreamBYOBReader.read | read()} method has not yet been settled. Attempting to + * do so will throw a `TypeError` and leave the reader locked to the stream. + */ + releaseLock() { + if (!IsReadableStreamBYOBReader(this)) { + throw byobReaderBrandCheckException('releaseLock'); + } + if (this._ownerReadableStream === undefined) { + return; + } + if (this._readIntoRequests.length > 0) { + throw new TypeError('Tried to release a reader lock when that reader has pending read() calls un-settled'); + } + ReadableStreamReaderGenericRelease(this); + } + } + Object.defineProperties(ReadableStreamBYOBReader.prototype, { + cancel: { enumerable: true }, + read: { enumerable: true }, + releaseLock: { enumerable: true }, + closed: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(ReadableStreamBYOBReader.prototype, SymbolPolyfill.toStringTag, { + value: 'ReadableStreamBYOBReader', + configurable: true + }); + } + // Abstract operations for the readers. + function IsReadableStreamBYOBReader(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_readIntoRequests')) { + return false; + } + return true; + } + function ReadableStreamBYOBReaderRead(reader, view, readIntoRequest) { + const stream = reader._ownerReadableStream; + stream._disturbed = true; + if (stream._state === 'errored') { + readIntoRequest._errorSteps(stream._storedError); + } + else { + ReadableByteStreamControllerPullInto(stream._readableStreamController, view, readIntoRequest); + } + } + // Helper functions for the ReadableStreamBYOBReader. + function byobReaderBrandCheckException(name) { + return new TypeError(`ReadableStreamBYOBReader.prototype.${name} can only be used on a ReadableStreamBYOBReader`); + } + + function ExtractHighWaterMark(strategy, defaultHWM) { + const { highWaterMark } = strategy; + if (highWaterMark === undefined) { + return defaultHWM; + } + if (NumberIsNaN(highWaterMark) || highWaterMark < 0) { + throw new RangeError('Invalid highWaterMark'); + } + return highWaterMark; + } + function ExtractSizeAlgorithm(strategy) { + const { size } = strategy; + if (!size) { + return () => 1; + } + return size; + } + + function convertQueuingStrategy(init, context) { + assertDictionary(init, context); + const highWaterMark = init === null || init === void 0 ? void 0 : init.highWaterMark; + const size = init === null || init === void 0 ? void 0 : init.size; + return { + highWaterMark: highWaterMark === undefined ? undefined : convertUnrestrictedDouble(highWaterMark), + size: size === undefined ? undefined : convertQueuingStrategySize(size, `${context} has member 'size' that`) + }; + } + function convertQueuingStrategySize(fn, context) { + assertFunction(fn, context); + return chunk => convertUnrestrictedDouble(fn(chunk)); + } + + function convertUnderlyingSink(original, context) { + assertDictionary(original, context); + const abort = original === null || original === void 0 ? void 0 : original.abort; + const close = original === null || original === void 0 ? void 0 : original.close; + const start = original === null || original === void 0 ? void 0 : original.start; + const type = original === null || original === void 0 ? void 0 : original.type; + const write = original === null || original === void 0 ? void 0 : original.write; + return { + abort: abort === undefined ? + undefined : + convertUnderlyingSinkAbortCallback(abort, original, `${context} has member 'abort' that`), + close: close === undefined ? + undefined : + convertUnderlyingSinkCloseCallback(close, original, `${context} has member 'close' that`), + start: start === undefined ? + undefined : + convertUnderlyingSinkStartCallback(start, original, `${context} has member 'start' that`), + write: write === undefined ? + undefined : + convertUnderlyingSinkWriteCallback(write, original, `${context} has member 'write' that`), + type + }; + } + function convertUnderlyingSinkAbortCallback(fn, original, context) { + assertFunction(fn, context); + return (reason) => promiseCall(fn, original, [reason]); + } + function convertUnderlyingSinkCloseCallback(fn, original, context) { + assertFunction(fn, context); + return () => promiseCall(fn, original, []); + } + function convertUnderlyingSinkStartCallback(fn, original, context) { + assertFunction(fn, context); + return (controller) => reflectCall(fn, original, [controller]); + } + function convertUnderlyingSinkWriteCallback(fn, original, context) { + assertFunction(fn, context); + return (chunk, controller) => promiseCall(fn, original, [chunk, controller]); + } + + function assertWritableStream(x, context) { + if (!IsWritableStream(x)) { + throw new TypeError(`${context} is not a WritableStream.`); + } + } + + /** + * A writable stream represents a destination for data, into which you can write. + * + * @public + */ + class WritableStream$1 { + constructor(rawUnderlyingSink = {}, rawStrategy = {}) { + if (rawUnderlyingSink === undefined) { + rawUnderlyingSink = null; + } + else { + assertObject(rawUnderlyingSink, 'First parameter'); + } + const strategy = convertQueuingStrategy(rawStrategy, 'Second parameter'); + const underlyingSink = convertUnderlyingSink(rawUnderlyingSink, 'First parameter'); + InitializeWritableStream(this); + const type = underlyingSink.type; + if (type !== undefined) { + throw new RangeError('Invalid type is specified'); + } + const sizeAlgorithm = ExtractSizeAlgorithm(strategy); + const highWaterMark = ExtractHighWaterMark(strategy, 1); + SetUpWritableStreamDefaultControllerFromUnderlyingSink(this, underlyingSink, highWaterMark, sizeAlgorithm); + } + /** + * Returns whether or not the writable stream is locked to a writer. + */ + get locked() { + if (!IsWritableStream(this)) { + throw streamBrandCheckException$2('locked'); + } + return IsWritableStreamLocked(this); + } + /** + * Aborts the stream, signaling that the producer can no longer successfully write to the stream and it is to be + * immediately moved to an errored state, with any queued-up writes discarded. This will also execute any abort + * mechanism of the underlying sink. + * + * The returned promise will fulfill if the stream shuts down successfully, or reject if the underlying sink signaled + * that there was an error doing so. Additionally, it will reject with a `TypeError` (without attempting to cancel + * the stream) if the stream is currently locked. + */ + abort(reason = undefined) { + if (!IsWritableStream(this)) { + return promiseRejectedWith(streamBrandCheckException$2('abort')); + } + if (IsWritableStreamLocked(this)) { + return promiseRejectedWith(new TypeError('Cannot abort a stream that already has a writer')); + } + return WritableStreamAbort(this, reason); + } + /** + * Closes the stream. The underlying sink will finish processing any previously-written chunks, before invoking its + * close behavior. During this time any further attempts to write will fail (without erroring the stream). + * + * The method returns a promise that will fulfill if all remaining chunks are successfully written and the stream + * successfully closes, or rejects if an error is encountered during this process. Additionally, it will reject with + * a `TypeError` (without attempting to cancel the stream) if the stream is currently locked. + */ + close() { + if (!IsWritableStream(this)) { + return promiseRejectedWith(streamBrandCheckException$2('close')); + } + if (IsWritableStreamLocked(this)) { + return promiseRejectedWith(new TypeError('Cannot close a stream that already has a writer')); + } + if (WritableStreamCloseQueuedOrInFlight(this)) { + return promiseRejectedWith(new TypeError('Cannot close an already-closing stream')); + } + return WritableStreamClose(this); + } + /** + * Creates a {@link WritableStreamDefaultWriter | writer} and locks the stream to the new writer. While the stream + * is locked, no other writer can be acquired until this one is released. + * + * This functionality is especially useful for creating abstractions that desire the ability to write to a stream + * without interruption or interleaving. By getting a writer for the stream, you can ensure nobody else can write at + * the same time, which would cause the resulting written data to be unpredictable and probably useless. + */ + getWriter() { + if (!IsWritableStream(this)) { + throw streamBrandCheckException$2('getWriter'); + } + return AcquireWritableStreamDefaultWriter(this); + } + } + Object.defineProperties(WritableStream$1.prototype, { + abort: { enumerable: true }, + close: { enumerable: true }, + getWriter: { enumerable: true }, + locked: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(WritableStream$1.prototype, SymbolPolyfill.toStringTag, { + value: 'WritableStream', + configurable: true + }); + } + // Abstract operations for the WritableStream. + function AcquireWritableStreamDefaultWriter(stream) { + return new WritableStreamDefaultWriter(stream); + } + // Throws if and only if startAlgorithm throws. + function CreateWritableStream(startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, highWaterMark = 1, sizeAlgorithm = () => 1) { + const stream = Object.create(WritableStream$1.prototype); + InitializeWritableStream(stream); + const controller = Object.create(WritableStreamDefaultController.prototype); + SetUpWritableStreamDefaultController(stream, controller, startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, highWaterMark, sizeAlgorithm); + return stream; + } + function InitializeWritableStream(stream) { + stream._state = 'writable'; + // The error that will be reported by new method calls once the state becomes errored. Only set when [[state]] is + // 'erroring' or 'errored'. May be set to an undefined value. + stream._storedError = undefined; + stream._writer = undefined; + // Initialize to undefined first because the constructor of the controller checks this + // variable to validate the caller. + stream._writableStreamController = undefined; + // This queue is placed here instead of the writer class in order to allow for passing a writer to the next data + // producer without waiting for the queued writes to finish. + stream._writeRequests = new SimpleQueue(); + // Write requests are removed from _writeRequests when write() is called on the underlying sink. This prevents + // them from being erroneously rejected on error. If a write() call is in-flight, the request is stored here. + stream._inFlightWriteRequest = undefined; + // The promise that was returned from writer.close(). Stored here because it may be fulfilled after the writer + // has been detached. + stream._closeRequest = undefined; + // Close request is removed from _closeRequest when close() is called on the underlying sink. This prevents it + // from being erroneously rejected on error. If a close() call is in-flight, the request is stored here. + stream._inFlightCloseRequest = undefined; + // The promise that was returned from writer.abort(). This may also be fulfilled after the writer has detached. + stream._pendingAbortRequest = undefined; + // The backpressure signal set by the controller. + stream._backpressure = false; + } + function IsWritableStream(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_writableStreamController')) { + return false; + } + return true; + } + function IsWritableStreamLocked(stream) { + if (stream._writer === undefined) { + return false; + } + return true; + } + function WritableStreamAbort(stream, reason) { + const state = stream._state; + if (state === 'closed' || state === 'errored') { + return promiseResolvedWith(undefined); + } + if (stream._pendingAbortRequest !== undefined) { + return stream._pendingAbortRequest._promise; + } + let wasAlreadyErroring = false; + if (state === 'erroring') { + wasAlreadyErroring = true; + // reason will not be used, so don't keep a reference to it. + reason = undefined; + } + const promise = newPromise((resolve, reject) => { + stream._pendingAbortRequest = { + _promise: undefined, + _resolve: resolve, + _reject: reject, + _reason: reason, + _wasAlreadyErroring: wasAlreadyErroring + }; + }); + stream._pendingAbortRequest._promise = promise; + if (!wasAlreadyErroring) { + WritableStreamStartErroring(stream, reason); + } + return promise; + } + function WritableStreamClose(stream) { + const state = stream._state; + if (state === 'closed' || state === 'errored') { + return promiseRejectedWith(new TypeError(`The stream (in ${state} state) is not in the writable state and cannot be closed`)); + } + const promise = newPromise((resolve, reject) => { + const closeRequest = { + _resolve: resolve, + _reject: reject + }; + stream._closeRequest = closeRequest; + }); + const writer = stream._writer; + if (writer !== undefined && stream._backpressure && state === 'writable') { + defaultWriterReadyPromiseResolve(writer); + } + WritableStreamDefaultControllerClose(stream._writableStreamController); + return promise; + } + // WritableStream API exposed for controllers. + function WritableStreamAddWriteRequest(stream) { + const promise = newPromise((resolve, reject) => { + const writeRequest = { + _resolve: resolve, + _reject: reject + }; + stream._writeRequests.push(writeRequest); + }); + return promise; + } + function WritableStreamDealWithRejection(stream, error) { + const state = stream._state; + if (state === 'writable') { + WritableStreamStartErroring(stream, error); + return; + } + WritableStreamFinishErroring(stream); + } + function WritableStreamStartErroring(stream, reason) { + const controller = stream._writableStreamController; + stream._state = 'erroring'; + stream._storedError = reason; + const writer = stream._writer; + if (writer !== undefined) { + WritableStreamDefaultWriterEnsureReadyPromiseRejected(writer, reason); + } + if (!WritableStreamHasOperationMarkedInFlight(stream) && controller._started) { + WritableStreamFinishErroring(stream); + } + } + function WritableStreamFinishErroring(stream) { + stream._state = 'errored'; + stream._writableStreamController[ErrorSteps](); + const storedError = stream._storedError; + stream._writeRequests.forEach(writeRequest => { + writeRequest._reject(storedError); + }); + stream._writeRequests = new SimpleQueue(); + if (stream._pendingAbortRequest === undefined) { + WritableStreamRejectCloseAndClosedPromiseIfNeeded(stream); + return; + } + const abortRequest = stream._pendingAbortRequest; + stream._pendingAbortRequest = undefined; + if (abortRequest._wasAlreadyErroring) { + abortRequest._reject(storedError); + WritableStreamRejectCloseAndClosedPromiseIfNeeded(stream); + return; + } + const promise = stream._writableStreamController[AbortSteps](abortRequest._reason); + uponPromise(promise, () => { + abortRequest._resolve(); + WritableStreamRejectCloseAndClosedPromiseIfNeeded(stream); + }, (reason) => { + abortRequest._reject(reason); + WritableStreamRejectCloseAndClosedPromiseIfNeeded(stream); + }); + } + function WritableStreamFinishInFlightWrite(stream) { + stream._inFlightWriteRequest._resolve(undefined); + stream._inFlightWriteRequest = undefined; + } + function WritableStreamFinishInFlightWriteWithError(stream, error) { + stream._inFlightWriteRequest._reject(error); + stream._inFlightWriteRequest = undefined; + WritableStreamDealWithRejection(stream, error); + } + function WritableStreamFinishInFlightClose(stream) { + stream._inFlightCloseRequest._resolve(undefined); + stream._inFlightCloseRequest = undefined; + const state = stream._state; + if (state === 'erroring') { + // The error was too late to do anything, so it is ignored. + stream._storedError = undefined; + if (stream._pendingAbortRequest !== undefined) { + stream._pendingAbortRequest._resolve(); + stream._pendingAbortRequest = undefined; + } + } + stream._state = 'closed'; + const writer = stream._writer; + if (writer !== undefined) { + defaultWriterClosedPromiseResolve(writer); + } + } + function WritableStreamFinishInFlightCloseWithError(stream, error) { + stream._inFlightCloseRequest._reject(error); + stream._inFlightCloseRequest = undefined; + // Never execute sink abort() after sink close(). + if (stream._pendingAbortRequest !== undefined) { + stream._pendingAbortRequest._reject(error); + stream._pendingAbortRequest = undefined; + } + WritableStreamDealWithRejection(stream, error); + } + // TODO(ricea): Fix alphabetical order. + function WritableStreamCloseQueuedOrInFlight(stream) { + if (stream._closeRequest === undefined && stream._inFlightCloseRequest === undefined) { + return false; + } + return true; + } + function WritableStreamHasOperationMarkedInFlight(stream) { + if (stream._inFlightWriteRequest === undefined && stream._inFlightCloseRequest === undefined) { + return false; + } + return true; + } + function WritableStreamMarkCloseRequestInFlight(stream) { + stream._inFlightCloseRequest = stream._closeRequest; + stream._closeRequest = undefined; + } + function WritableStreamMarkFirstWriteRequestInFlight(stream) { + stream._inFlightWriteRequest = stream._writeRequests.shift(); + } + function WritableStreamRejectCloseAndClosedPromiseIfNeeded(stream) { + if (stream._closeRequest !== undefined) { + stream._closeRequest._reject(stream._storedError); + stream._closeRequest = undefined; + } + const writer = stream._writer; + if (writer !== undefined) { + defaultWriterClosedPromiseReject(writer, stream._storedError); + } + } + function WritableStreamUpdateBackpressure(stream, backpressure) { + const writer = stream._writer; + if (writer !== undefined && backpressure !== stream._backpressure) { + if (backpressure) { + defaultWriterReadyPromiseReset(writer); + } + else { + defaultWriterReadyPromiseResolve(writer); + } + } + stream._backpressure = backpressure; + } + /** + * A default writer vended by a {@link WritableStream}. + * + * @public + */ + class WritableStreamDefaultWriter { + constructor(stream) { + assertRequiredArgument(stream, 1, 'WritableStreamDefaultWriter'); + assertWritableStream(stream, 'First parameter'); + if (IsWritableStreamLocked(stream)) { + throw new TypeError('This stream has already been locked for exclusive writing by another writer'); + } + this._ownerWritableStream = stream; + stream._writer = this; + const state = stream._state; + if (state === 'writable') { + if (!WritableStreamCloseQueuedOrInFlight(stream) && stream._backpressure) { + defaultWriterReadyPromiseInitialize(this); + } + else { + defaultWriterReadyPromiseInitializeAsResolved(this); + } + defaultWriterClosedPromiseInitialize(this); + } + else if (state === 'erroring') { + defaultWriterReadyPromiseInitializeAsRejected(this, stream._storedError); + defaultWriterClosedPromiseInitialize(this); + } + else if (state === 'closed') { + defaultWriterReadyPromiseInitializeAsResolved(this); + defaultWriterClosedPromiseInitializeAsResolved(this); + } + else { + const storedError = stream._storedError; + defaultWriterReadyPromiseInitializeAsRejected(this, storedError); + defaultWriterClosedPromiseInitializeAsRejected(this, storedError); + } + } + /** + * Returns a promise that will be fulfilled when the stream becomes closed, or rejected if the stream ever errors or + * the writer’s lock is released before the stream finishes closing. + */ + get closed() { + if (!IsWritableStreamDefaultWriter(this)) { + return promiseRejectedWith(defaultWriterBrandCheckException('closed')); + } + return this._closedPromise; + } + /** + * Returns the desired size to fill the stream’s internal queue. It can be negative, if the queue is over-full. + * A producer can use this information to determine the right amount of data to write. + * + * It will be `null` if the stream cannot be successfully written to (due to either being errored, or having an abort + * queued up). It will return zero if the stream is closed. And the getter will throw an exception if invoked when + * the writer’s lock is released. + */ + get desiredSize() { + if (!IsWritableStreamDefaultWriter(this)) { + throw defaultWriterBrandCheckException('desiredSize'); + } + if (this._ownerWritableStream === undefined) { + throw defaultWriterLockException('desiredSize'); + } + return WritableStreamDefaultWriterGetDesiredSize(this); + } + /** + * Returns a promise that will be fulfilled when the desired size to fill the stream’s internal queue transitions + * from non-positive to positive, signaling that it is no longer applying backpressure. Once the desired size dips + * back to zero or below, the getter will return a new promise that stays pending until the next transition. + * + * If the stream becomes errored or aborted, or the writer’s lock is released, the returned promise will become + * rejected. + */ + get ready() { + if (!IsWritableStreamDefaultWriter(this)) { + return promiseRejectedWith(defaultWriterBrandCheckException('ready')); + } + return this._readyPromise; + } + /** + * If the reader is active, behaves the same as {@link WritableStream.abort | stream.abort(reason)}. + */ + abort(reason = undefined) { + if (!IsWritableStreamDefaultWriter(this)) { + return promiseRejectedWith(defaultWriterBrandCheckException('abort')); + } + if (this._ownerWritableStream === undefined) { + return promiseRejectedWith(defaultWriterLockException('abort')); + } + return WritableStreamDefaultWriterAbort(this, reason); + } + /** + * If the reader is active, behaves the same as {@link WritableStream.close | stream.close()}. + */ + close() { + if (!IsWritableStreamDefaultWriter(this)) { + return promiseRejectedWith(defaultWriterBrandCheckException('close')); + } + const stream = this._ownerWritableStream; + if (stream === undefined) { + return promiseRejectedWith(defaultWriterLockException('close')); + } + if (WritableStreamCloseQueuedOrInFlight(stream)) { + return promiseRejectedWith(new TypeError('Cannot close an already-closing stream')); + } + return WritableStreamDefaultWriterClose(this); + } + /** + * Releases the writer’s lock on the corresponding stream. After the lock is released, the writer is no longer active. + * If the associated stream is errored when the lock is released, the writer will appear errored in the same way from + * now on; otherwise, the writer will appear closed. + * + * Note that the lock can still be released even if some ongoing writes have not yet finished (i.e. even if the + * promises returned from previous calls to {@link WritableStreamDefaultWriter.write | write()} have not yet settled). + * It’s not necessary to hold the lock on the writer for the duration of the write; the lock instead simply prevents + * other producers from writing in an interleaved manner. + */ + releaseLock() { + if (!IsWritableStreamDefaultWriter(this)) { + throw defaultWriterBrandCheckException('releaseLock'); + } + const stream = this._ownerWritableStream; + if (stream === undefined) { + return; + } + WritableStreamDefaultWriterRelease(this); + } + write(chunk = undefined) { + if (!IsWritableStreamDefaultWriter(this)) { + return promiseRejectedWith(defaultWriterBrandCheckException('write')); + } + if (this._ownerWritableStream === undefined) { + return promiseRejectedWith(defaultWriterLockException('write to')); + } + return WritableStreamDefaultWriterWrite(this, chunk); + } + } + Object.defineProperties(WritableStreamDefaultWriter.prototype, { + abort: { enumerable: true }, + close: { enumerable: true }, + releaseLock: { enumerable: true }, + write: { enumerable: true }, + closed: { enumerable: true }, + desiredSize: { enumerable: true }, + ready: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(WritableStreamDefaultWriter.prototype, SymbolPolyfill.toStringTag, { + value: 'WritableStreamDefaultWriter', + configurable: true + }); + } + // Abstract operations for the WritableStreamDefaultWriter. + function IsWritableStreamDefaultWriter(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_ownerWritableStream')) { + return false; + } + return true; + } + // A client of WritableStreamDefaultWriter may use these functions directly to bypass state check. + function WritableStreamDefaultWriterAbort(writer, reason) { + const stream = writer._ownerWritableStream; + return WritableStreamAbort(stream, reason); + } + function WritableStreamDefaultWriterClose(writer) { + const stream = writer._ownerWritableStream; + return WritableStreamClose(stream); + } + function WritableStreamDefaultWriterCloseWithErrorPropagation(writer) { + const stream = writer._ownerWritableStream; + const state = stream._state; + if (WritableStreamCloseQueuedOrInFlight(stream) || state === 'closed') { + return promiseResolvedWith(undefined); + } + if (state === 'errored') { + return promiseRejectedWith(stream._storedError); + } + return WritableStreamDefaultWriterClose(writer); + } + function WritableStreamDefaultWriterEnsureClosedPromiseRejected(writer, error) { + if (writer._closedPromiseState === 'pending') { + defaultWriterClosedPromiseReject(writer, error); + } + else { + defaultWriterClosedPromiseResetToRejected(writer, error); + } + } + function WritableStreamDefaultWriterEnsureReadyPromiseRejected(writer, error) { + if (writer._readyPromiseState === 'pending') { + defaultWriterReadyPromiseReject(writer, error); + } + else { + defaultWriterReadyPromiseResetToRejected(writer, error); + } + } + function WritableStreamDefaultWriterGetDesiredSize(writer) { + const stream = writer._ownerWritableStream; + const state = stream._state; + if (state === 'errored' || state === 'erroring') { + return null; + } + if (state === 'closed') { + return 0; + } + return WritableStreamDefaultControllerGetDesiredSize(stream._writableStreamController); + } + function WritableStreamDefaultWriterRelease(writer) { + const stream = writer._ownerWritableStream; + const releasedError = new TypeError(`Writer was released and can no longer be used to monitor the stream's closedness`); + WritableStreamDefaultWriterEnsureReadyPromiseRejected(writer, releasedError); + // The state transitions to "errored" before the sink abort() method runs, but the writer.closed promise is not + // rejected until afterwards. This means that simply testing state will not work. + WritableStreamDefaultWriterEnsureClosedPromiseRejected(writer, releasedError); + stream._writer = undefined; + writer._ownerWritableStream = undefined; + } + function WritableStreamDefaultWriterWrite(writer, chunk) { + const stream = writer._ownerWritableStream; + const controller = stream._writableStreamController; + const chunkSize = WritableStreamDefaultControllerGetChunkSize(controller, chunk); + if (stream !== writer._ownerWritableStream) { + return promiseRejectedWith(defaultWriterLockException('write to')); + } + const state = stream._state; + if (state === 'errored') { + return promiseRejectedWith(stream._storedError); + } + if (WritableStreamCloseQueuedOrInFlight(stream) || state === 'closed') { + return promiseRejectedWith(new TypeError('The stream is closing or closed and cannot be written to')); + } + if (state === 'erroring') { + return promiseRejectedWith(stream._storedError); + } + const promise = WritableStreamAddWriteRequest(stream); + WritableStreamDefaultControllerWrite(controller, chunk, chunkSize); + return promise; + } + const closeSentinel = {}; + /** + * Allows control of a {@link WritableStream | writable stream}'s state and internal queue. + * + * @public + */ + class WritableStreamDefaultController { + constructor() { + throw new TypeError('Illegal constructor'); + } + /** + * Closes the controlled writable stream, making all future interactions with it fail with the given error `e`. + * + * This method is rarely used, since usually it suffices to return a rejected promise from one of the underlying + * sink's methods. However, it can be useful for suddenly shutting down a stream in response to an event outside the + * normal lifecycle of interactions with the underlying sink. + */ + error(e = undefined) { + if (!IsWritableStreamDefaultController(this)) { + throw new TypeError('WritableStreamDefaultController.prototype.error can only be used on a WritableStreamDefaultController'); + } + const state = this._controlledWritableStream._state; + if (state !== 'writable') { + // The stream is closed, errored or will be soon. The sink can't do anything useful if it gets an error here, so + // just treat it as a no-op. + return; + } + WritableStreamDefaultControllerError(this, e); + } + /** @internal */ + [AbortSteps](reason) { + const result = this._abortAlgorithm(reason); + WritableStreamDefaultControllerClearAlgorithms(this); + return result; + } + /** @internal */ + [ErrorSteps]() { + ResetQueue(this); + } + } + Object.defineProperties(WritableStreamDefaultController.prototype, { + error: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(WritableStreamDefaultController.prototype, SymbolPolyfill.toStringTag, { + value: 'WritableStreamDefaultController', + configurable: true + }); + } + // Abstract operations implementing interface required by the WritableStream. + function IsWritableStreamDefaultController(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_controlledWritableStream')) { + return false; + } + return true; + } + function SetUpWritableStreamDefaultController(stream, controller, startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, highWaterMark, sizeAlgorithm) { + controller._controlledWritableStream = stream; + stream._writableStreamController = controller; + // Need to set the slots so that the assert doesn't fire. In the spec the slots already exist implicitly. + controller._queue = undefined; + controller._queueTotalSize = undefined; + ResetQueue(controller); + controller._started = false; + controller._strategySizeAlgorithm = sizeAlgorithm; + controller._strategyHWM = highWaterMark; + controller._writeAlgorithm = writeAlgorithm; + controller._closeAlgorithm = closeAlgorithm; + controller._abortAlgorithm = abortAlgorithm; + const backpressure = WritableStreamDefaultControllerGetBackpressure(controller); + WritableStreamUpdateBackpressure(stream, backpressure); + const startResult = startAlgorithm(); + const startPromise = promiseResolvedWith(startResult); + uponPromise(startPromise, () => { + controller._started = true; + WritableStreamDefaultControllerAdvanceQueueIfNeeded(controller); + }, r => { + controller._started = true; + WritableStreamDealWithRejection(stream, r); + }); + } + function SetUpWritableStreamDefaultControllerFromUnderlyingSink(stream, underlyingSink, highWaterMark, sizeAlgorithm) { + const controller = Object.create(WritableStreamDefaultController.prototype); + let startAlgorithm = () => undefined; + let writeAlgorithm = () => promiseResolvedWith(undefined); + let closeAlgorithm = () => promiseResolvedWith(undefined); + let abortAlgorithm = () => promiseResolvedWith(undefined); + if (underlyingSink.start !== undefined) { + startAlgorithm = () => underlyingSink.start(controller); + } + if (underlyingSink.write !== undefined) { + writeAlgorithm = chunk => underlyingSink.write(chunk, controller); + } + if (underlyingSink.close !== undefined) { + closeAlgorithm = () => underlyingSink.close(); + } + if (underlyingSink.abort !== undefined) { + abortAlgorithm = reason => underlyingSink.abort(reason); + } + SetUpWritableStreamDefaultController(stream, controller, startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, highWaterMark, sizeAlgorithm); + } + // ClearAlgorithms may be called twice. Erroring the same stream in multiple ways will often result in redundant calls. + function WritableStreamDefaultControllerClearAlgorithms(controller) { + controller._writeAlgorithm = undefined; + controller._closeAlgorithm = undefined; + controller._abortAlgorithm = undefined; + controller._strategySizeAlgorithm = undefined; + } + function WritableStreamDefaultControllerClose(controller) { + EnqueueValueWithSize(controller, closeSentinel, 0); + WritableStreamDefaultControllerAdvanceQueueIfNeeded(controller); + } + function WritableStreamDefaultControllerGetChunkSize(controller, chunk) { + try { + return controller._strategySizeAlgorithm(chunk); + } + catch (chunkSizeE) { + WritableStreamDefaultControllerErrorIfNeeded(controller, chunkSizeE); + return 1; + } + } + function WritableStreamDefaultControllerGetDesiredSize(controller) { + return controller._strategyHWM - controller._queueTotalSize; + } + function WritableStreamDefaultControllerWrite(controller, chunk, chunkSize) { + try { + EnqueueValueWithSize(controller, chunk, chunkSize); + } + catch (enqueueE) { + WritableStreamDefaultControllerErrorIfNeeded(controller, enqueueE); + return; + } + const stream = controller._controlledWritableStream; + if (!WritableStreamCloseQueuedOrInFlight(stream) && stream._state === 'writable') { + const backpressure = WritableStreamDefaultControllerGetBackpressure(controller); + WritableStreamUpdateBackpressure(stream, backpressure); + } + WritableStreamDefaultControllerAdvanceQueueIfNeeded(controller); + } + // Abstract operations for the WritableStreamDefaultController. + function WritableStreamDefaultControllerAdvanceQueueIfNeeded(controller) { + const stream = controller._controlledWritableStream; + if (!controller._started) { + return; + } + if (stream._inFlightWriteRequest !== undefined) { + return; + } + const state = stream._state; + if (state === 'erroring') { + WritableStreamFinishErroring(stream); + return; + } + if (controller._queue.length === 0) { + return; + } + const value = PeekQueueValue(controller); + if (value === closeSentinel) { + WritableStreamDefaultControllerProcessClose(controller); + } + else { + WritableStreamDefaultControllerProcessWrite(controller, value); + } + } + function WritableStreamDefaultControllerErrorIfNeeded(controller, error) { + if (controller._controlledWritableStream._state === 'writable') { + WritableStreamDefaultControllerError(controller, error); + } + } + function WritableStreamDefaultControllerProcessClose(controller) { + const stream = controller._controlledWritableStream; + WritableStreamMarkCloseRequestInFlight(stream); + DequeueValue(controller); + const sinkClosePromise = controller._closeAlgorithm(); + WritableStreamDefaultControllerClearAlgorithms(controller); + uponPromise(sinkClosePromise, () => { + WritableStreamFinishInFlightClose(stream); + }, reason => { + WritableStreamFinishInFlightCloseWithError(stream, reason); + }); + } + function WritableStreamDefaultControllerProcessWrite(controller, chunk) { + const stream = controller._controlledWritableStream; + WritableStreamMarkFirstWriteRequestInFlight(stream); + const sinkWritePromise = controller._writeAlgorithm(chunk); + uponPromise(sinkWritePromise, () => { + WritableStreamFinishInFlightWrite(stream); + const state = stream._state; + DequeueValue(controller); + if (!WritableStreamCloseQueuedOrInFlight(stream) && state === 'writable') { + const backpressure = WritableStreamDefaultControllerGetBackpressure(controller); + WritableStreamUpdateBackpressure(stream, backpressure); + } + WritableStreamDefaultControllerAdvanceQueueIfNeeded(controller); + }, reason => { + if (stream._state === 'writable') { + WritableStreamDefaultControllerClearAlgorithms(controller); + } + WritableStreamFinishInFlightWriteWithError(stream, reason); + }); + } + function WritableStreamDefaultControllerGetBackpressure(controller) { + const desiredSize = WritableStreamDefaultControllerGetDesiredSize(controller); + return desiredSize <= 0; + } + // A client of WritableStreamDefaultController may use these functions directly to bypass state check. + function WritableStreamDefaultControllerError(controller, error) { + const stream = controller._controlledWritableStream; + WritableStreamDefaultControllerClearAlgorithms(controller); + WritableStreamStartErroring(stream, error); + } + // Helper functions for the WritableStream. + function streamBrandCheckException$2(name) { + return new TypeError(`WritableStream.prototype.${name} can only be used on a WritableStream`); + } + // Helper functions for the WritableStreamDefaultWriter. + function defaultWriterBrandCheckException(name) { + return new TypeError(`WritableStreamDefaultWriter.prototype.${name} can only be used on a WritableStreamDefaultWriter`); + } + function defaultWriterLockException(name) { + return new TypeError('Cannot ' + name + ' a stream using a released writer'); + } + function defaultWriterClosedPromiseInitialize(writer) { + writer._closedPromise = newPromise((resolve, reject) => { + writer._closedPromise_resolve = resolve; + writer._closedPromise_reject = reject; + writer._closedPromiseState = 'pending'; + }); + } + function defaultWriterClosedPromiseInitializeAsRejected(writer, reason) { + defaultWriterClosedPromiseInitialize(writer); + defaultWriterClosedPromiseReject(writer, reason); + } + function defaultWriterClosedPromiseInitializeAsResolved(writer) { + defaultWriterClosedPromiseInitialize(writer); + defaultWriterClosedPromiseResolve(writer); + } + function defaultWriterClosedPromiseReject(writer, reason) { + if (writer._closedPromise_reject === undefined) { + return; + } + setPromiseIsHandledToTrue(writer._closedPromise); + writer._closedPromise_reject(reason); + writer._closedPromise_resolve = undefined; + writer._closedPromise_reject = undefined; + writer._closedPromiseState = 'rejected'; + } + function defaultWriterClosedPromiseResetToRejected(writer, reason) { + defaultWriterClosedPromiseInitializeAsRejected(writer, reason); + } + function defaultWriterClosedPromiseResolve(writer) { + if (writer._closedPromise_resolve === undefined) { + return; + } + writer._closedPromise_resolve(undefined); + writer._closedPromise_resolve = undefined; + writer._closedPromise_reject = undefined; + writer._closedPromiseState = 'resolved'; + } + function defaultWriterReadyPromiseInitialize(writer) { + writer._readyPromise = newPromise((resolve, reject) => { + writer._readyPromise_resolve = resolve; + writer._readyPromise_reject = reject; + }); + writer._readyPromiseState = 'pending'; + } + function defaultWriterReadyPromiseInitializeAsRejected(writer, reason) { + defaultWriterReadyPromiseInitialize(writer); + defaultWriterReadyPromiseReject(writer, reason); + } + function defaultWriterReadyPromiseInitializeAsResolved(writer) { + defaultWriterReadyPromiseInitialize(writer); + defaultWriterReadyPromiseResolve(writer); + } + function defaultWriterReadyPromiseReject(writer, reason) { + if (writer._readyPromise_reject === undefined) { + return; + } + setPromiseIsHandledToTrue(writer._readyPromise); + writer._readyPromise_reject(reason); + writer._readyPromise_resolve = undefined; + writer._readyPromise_reject = undefined; + writer._readyPromiseState = 'rejected'; + } + function defaultWriterReadyPromiseReset(writer) { + defaultWriterReadyPromiseInitialize(writer); + } + function defaultWriterReadyPromiseResetToRejected(writer, reason) { + defaultWriterReadyPromiseInitializeAsRejected(writer, reason); + } + function defaultWriterReadyPromiseResolve(writer) { + if (writer._readyPromise_resolve === undefined) { + return; + } + writer._readyPromise_resolve(undefined); + writer._readyPromise_resolve = undefined; + writer._readyPromise_reject = undefined; + writer._readyPromiseState = 'fulfilled'; + } + + function isAbortSignal(value) { + if (typeof value !== 'object' || value === null) { + return false; + } + try { + return typeof value.aborted === 'boolean'; + } + catch (_a) { + // AbortSignal.prototype.aborted throws if its brand check fails + return false; + } + } + + /// + const NativeDOMException = typeof DOMException !== 'undefined' ? DOMException : undefined; + + /// + function isDOMExceptionConstructor(ctor) { + if (!(typeof ctor === 'function' || typeof ctor === 'object')) { + return false; + } + try { + new ctor(); + return true; + } + catch (_a) { + return false; + } + } + function createDOMExceptionPolyfill() { + // eslint-disable-next-line no-shadow + const ctor = function DOMException(message, name) { + this.message = message || ''; + this.name = name || 'Error'; + if (Error.captureStackTrace) { + Error.captureStackTrace(this, this.constructor); + } + }; + ctor.prototype = Object.create(Error.prototype); + Object.defineProperty(ctor.prototype, 'constructor', { value: ctor, writable: true, configurable: true }); + return ctor; + } + // eslint-disable-next-line no-redeclare + const DOMException$1 = isDOMExceptionConstructor(NativeDOMException) ? NativeDOMException : createDOMExceptionPolyfill(); + + function ReadableStreamPipeTo(source, dest, preventClose, preventAbort, preventCancel, signal) { + const reader = AcquireReadableStreamDefaultReader(source); + const writer = AcquireWritableStreamDefaultWriter(dest); + source._disturbed = true; + let shuttingDown = false; + // This is used to keep track of the spec's requirement that we wait for ongoing writes during shutdown. + let currentWrite = promiseResolvedWith(undefined); + return newPromise((resolve, reject) => { + let abortAlgorithm; + if (signal !== undefined) { + abortAlgorithm = () => { + const error = new DOMException$1('Aborted', 'AbortError'); + const actions = []; + if (!preventAbort) { + actions.push(() => { + if (dest._state === 'writable') { + return WritableStreamAbort(dest, error); + } + return promiseResolvedWith(undefined); + }); + } + if (!preventCancel) { + actions.push(() => { + if (source._state === 'readable') { + return ReadableStreamCancel(source, error); + } + return promiseResolvedWith(undefined); + }); + } + shutdownWithAction(() => Promise.all(actions.map(action => action())), true, error); + }; + if (signal.aborted) { + abortAlgorithm(); + return; + } + signal.addEventListener('abort', abortAlgorithm); + } + // Using reader and writer, read all chunks from this and write them to dest + // - Backpressure must be enforced + // - Shutdown must stop all activity + function pipeLoop() { + return newPromise((resolveLoop, rejectLoop) => { + function next(done) { + if (done) { + resolveLoop(); + } + else { + // Use `PerformPromiseThen` instead of `uponPromise` to avoid + // adding unnecessary `.catch(rethrowAssertionErrorRejection)` handlers + PerformPromiseThen(pipeStep(), next, rejectLoop); + } + } + next(false); + }); + } + function pipeStep() { + if (shuttingDown) { + return promiseResolvedWith(true); + } + return PerformPromiseThen(writer._readyPromise, () => { + return newPromise((resolveRead, rejectRead) => { + ReadableStreamDefaultReaderRead(reader, { + _chunkSteps: chunk => { + currentWrite = PerformPromiseThen(WritableStreamDefaultWriterWrite(writer, chunk), undefined, noop); + resolveRead(false); + }, + _closeSteps: () => resolveRead(true), + _errorSteps: rejectRead + }); + }); + }); + } + // Errors must be propagated forward + isOrBecomesErrored(source, reader._closedPromise, storedError => { + if (!preventAbort) { + shutdownWithAction(() => WritableStreamAbort(dest, storedError), true, storedError); + } + else { + shutdown(true, storedError); + } + }); + // Errors must be propagated backward + isOrBecomesErrored(dest, writer._closedPromise, storedError => { + if (!preventCancel) { + shutdownWithAction(() => ReadableStreamCancel(source, storedError), true, storedError); + } + else { + shutdown(true, storedError); + } + }); + // Closing must be propagated forward + isOrBecomesClosed(source, reader._closedPromise, () => { + if (!preventClose) { + shutdownWithAction(() => WritableStreamDefaultWriterCloseWithErrorPropagation(writer)); + } + else { + shutdown(); + } + }); + // Closing must be propagated backward + if (WritableStreamCloseQueuedOrInFlight(dest) || dest._state === 'closed') { + const destClosed = new TypeError('the destination writable stream closed before all data could be piped to it'); + if (!preventCancel) { + shutdownWithAction(() => ReadableStreamCancel(source, destClosed), true, destClosed); + } + else { + shutdown(true, destClosed); + } + } + setPromiseIsHandledToTrue(pipeLoop()); + function waitForWritesToFinish() { + // Another write may have started while we were waiting on this currentWrite, so we have to be sure to wait + // for that too. + const oldCurrentWrite = currentWrite; + return PerformPromiseThen(currentWrite, () => oldCurrentWrite !== currentWrite ? waitForWritesToFinish() : undefined); + } + function isOrBecomesErrored(stream, promise, action) { + if (stream._state === 'errored') { + action(stream._storedError); + } + else { + uponRejection(promise, action); + } + } + function isOrBecomesClosed(stream, promise, action) { + if (stream._state === 'closed') { + action(); + } + else { + uponFulfillment(promise, action); + } + } + function shutdownWithAction(action, originalIsError, originalError) { + if (shuttingDown) { + return; + } + shuttingDown = true; + if (dest._state === 'writable' && !WritableStreamCloseQueuedOrInFlight(dest)) { + uponFulfillment(waitForWritesToFinish(), doTheRest); + } + else { + doTheRest(); + } + function doTheRest() { + uponPromise(action(), () => finalize(originalIsError, originalError), newError => finalize(true, newError)); + } + } + function shutdown(isError, error) { + if (shuttingDown) { + return; + } + shuttingDown = true; + if (dest._state === 'writable' && !WritableStreamCloseQueuedOrInFlight(dest)) { + uponFulfillment(waitForWritesToFinish(), () => finalize(isError, error)); + } + else { + finalize(isError, error); + } + } + function finalize(isError, error) { + WritableStreamDefaultWriterRelease(writer); + ReadableStreamReaderGenericRelease(reader); + if (signal !== undefined) { + signal.removeEventListener('abort', abortAlgorithm); + } + if (isError) { + reject(error); + } + else { + resolve(undefined); + } + } + }); + } + + /** + * Allows control of a {@link ReadableStream | readable stream}'s state and internal queue. + * + * @public + */ + class ReadableStreamDefaultController { + constructor() { + throw new TypeError('Illegal constructor'); + } + /** + * Returns the desired size to fill the controlled stream's internal queue. It can be negative, if the queue is + * over-full. An underlying source ought to use this information to determine when and how to apply backpressure. + */ + get desiredSize() { + if (!IsReadableStreamDefaultController(this)) { + throw defaultControllerBrandCheckException$1('desiredSize'); + } + return ReadableStreamDefaultControllerGetDesiredSize(this); + } + /** + * Closes the controlled readable stream. Consumers will still be able to read any previously-enqueued chunks from + * the stream, but once those are read, the stream will become closed. + */ + close() { + if (!IsReadableStreamDefaultController(this)) { + throw defaultControllerBrandCheckException$1('close'); + } + if (!ReadableStreamDefaultControllerCanCloseOrEnqueue(this)) { + throw new TypeError('The stream is not in a state that permits close'); + } + ReadableStreamDefaultControllerClose(this); + } + enqueue(chunk = undefined) { + if (!IsReadableStreamDefaultController(this)) { + throw defaultControllerBrandCheckException$1('enqueue'); + } + if (!ReadableStreamDefaultControllerCanCloseOrEnqueue(this)) { + throw new TypeError('The stream is not in a state that permits enqueue'); + } + return ReadableStreamDefaultControllerEnqueue(this, chunk); + } + /** + * Errors the controlled readable stream, making all future interactions with it fail with the given error `e`. + */ + error(e = undefined) { + if (!IsReadableStreamDefaultController(this)) { + throw defaultControllerBrandCheckException$1('error'); + } + ReadableStreamDefaultControllerError(this, e); + } + /** @internal */ + [CancelSteps](reason) { + ResetQueue(this); + const result = this._cancelAlgorithm(reason); + ReadableStreamDefaultControllerClearAlgorithms(this); + return result; + } + /** @internal */ + [PullSteps](readRequest) { + const stream = this._controlledReadableStream; + if (this._queue.length > 0) { + const chunk = DequeueValue(this); + if (this._closeRequested && this._queue.length === 0) { + ReadableStreamDefaultControllerClearAlgorithms(this); + ReadableStreamClose(stream); + } + else { + ReadableStreamDefaultControllerCallPullIfNeeded(this); + } + readRequest._chunkSteps(chunk); + } + else { + ReadableStreamAddReadRequest(stream, readRequest); + ReadableStreamDefaultControllerCallPullIfNeeded(this); + } + } + } + Object.defineProperties(ReadableStreamDefaultController.prototype, { + close: { enumerable: true }, + enqueue: { enumerable: true }, + error: { enumerable: true }, + desiredSize: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(ReadableStreamDefaultController.prototype, SymbolPolyfill.toStringTag, { + value: 'ReadableStreamDefaultController', + configurable: true + }); + } + // Abstract operations for the ReadableStreamDefaultController. + function IsReadableStreamDefaultController(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_controlledReadableStream')) { + return false; + } + return true; + } + function ReadableStreamDefaultControllerCallPullIfNeeded(controller) { + const shouldPull = ReadableStreamDefaultControllerShouldCallPull(controller); + if (!shouldPull) { + return; + } + if (controller._pulling) { + controller._pullAgain = true; + return; + } + controller._pulling = true; + const pullPromise = controller._pullAlgorithm(); + uponPromise(pullPromise, () => { + controller._pulling = false; + if (controller._pullAgain) { + controller._pullAgain = false; + ReadableStreamDefaultControllerCallPullIfNeeded(controller); + } + }, e => { + ReadableStreamDefaultControllerError(controller, e); + }); + } + function ReadableStreamDefaultControllerShouldCallPull(controller) { + const stream = controller._controlledReadableStream; + if (!ReadableStreamDefaultControllerCanCloseOrEnqueue(controller)) { + return false; + } + if (!controller._started) { + return false; + } + if (IsReadableStreamLocked(stream) && ReadableStreamGetNumReadRequests(stream) > 0) { + return true; + } + const desiredSize = ReadableStreamDefaultControllerGetDesiredSize(controller); + if (desiredSize > 0) { + return true; + } + return false; + } + function ReadableStreamDefaultControllerClearAlgorithms(controller) { + controller._pullAlgorithm = undefined; + controller._cancelAlgorithm = undefined; + controller._strategySizeAlgorithm = undefined; + } + // A client of ReadableStreamDefaultController may use these functions directly to bypass state check. + function ReadableStreamDefaultControllerClose(controller) { + if (!ReadableStreamDefaultControllerCanCloseOrEnqueue(controller)) { + return; + } + const stream = controller._controlledReadableStream; + controller._closeRequested = true; + if (controller._queue.length === 0) { + ReadableStreamDefaultControllerClearAlgorithms(controller); + ReadableStreamClose(stream); + } + } + function ReadableStreamDefaultControllerEnqueue(controller, chunk) { + if (!ReadableStreamDefaultControllerCanCloseOrEnqueue(controller)) { + return; + } + const stream = controller._controlledReadableStream; + if (IsReadableStreamLocked(stream) && ReadableStreamGetNumReadRequests(stream) > 0) { + ReadableStreamFulfillReadRequest(stream, chunk, false); + } + else { + let chunkSize; + try { + chunkSize = controller._strategySizeAlgorithm(chunk); + } + catch (chunkSizeE) { + ReadableStreamDefaultControllerError(controller, chunkSizeE); + throw chunkSizeE; + } + try { + EnqueueValueWithSize(controller, chunk, chunkSize); + } + catch (enqueueE) { + ReadableStreamDefaultControllerError(controller, enqueueE); + throw enqueueE; + } + } + ReadableStreamDefaultControllerCallPullIfNeeded(controller); + } + function ReadableStreamDefaultControllerError(controller, e) { + const stream = controller._controlledReadableStream; + if (stream._state !== 'readable') { + return; + } + ResetQueue(controller); + ReadableStreamDefaultControllerClearAlgorithms(controller); + ReadableStreamError(stream, e); + } + function ReadableStreamDefaultControllerGetDesiredSize(controller) { + const state = controller._controlledReadableStream._state; + if (state === 'errored') { + return null; + } + if (state === 'closed') { + return 0; + } + return controller._strategyHWM - controller._queueTotalSize; + } + // This is used in the implementation of TransformStream. + function ReadableStreamDefaultControllerHasBackpressure(controller) { + if (ReadableStreamDefaultControllerShouldCallPull(controller)) { + return false; + } + return true; + } + function ReadableStreamDefaultControllerCanCloseOrEnqueue(controller) { + const state = controller._controlledReadableStream._state; + if (!controller._closeRequested && state === 'readable') { + return true; + } + return false; + } + function SetUpReadableStreamDefaultController(stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, sizeAlgorithm) { + controller._controlledReadableStream = stream; + controller._queue = undefined; + controller._queueTotalSize = undefined; + ResetQueue(controller); + controller._started = false; + controller._closeRequested = false; + controller._pullAgain = false; + controller._pulling = false; + controller._strategySizeAlgorithm = sizeAlgorithm; + controller._strategyHWM = highWaterMark; + controller._pullAlgorithm = pullAlgorithm; + controller._cancelAlgorithm = cancelAlgorithm; + stream._readableStreamController = controller; + const startResult = startAlgorithm(); + uponPromise(promiseResolvedWith(startResult), () => { + controller._started = true; + ReadableStreamDefaultControllerCallPullIfNeeded(controller); + }, r => { + ReadableStreamDefaultControllerError(controller, r); + }); + } + function SetUpReadableStreamDefaultControllerFromUnderlyingSource(stream, underlyingSource, highWaterMark, sizeAlgorithm) { + const controller = Object.create(ReadableStreamDefaultController.prototype); + let startAlgorithm = () => undefined; + let pullAlgorithm = () => promiseResolvedWith(undefined); + let cancelAlgorithm = () => promiseResolvedWith(undefined); + if (underlyingSource.start !== undefined) { + startAlgorithm = () => underlyingSource.start(controller); + } + if (underlyingSource.pull !== undefined) { + pullAlgorithm = () => underlyingSource.pull(controller); + } + if (underlyingSource.cancel !== undefined) { + cancelAlgorithm = reason => underlyingSource.cancel(reason); + } + SetUpReadableStreamDefaultController(stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, sizeAlgorithm); + } + // Helper functions for the ReadableStreamDefaultController. + function defaultControllerBrandCheckException$1(name) { + return new TypeError(`ReadableStreamDefaultController.prototype.${name} can only be used on a ReadableStreamDefaultController`); + } + + function ReadableStreamTee(stream, cloneForBranch2) { + const reader = AcquireReadableStreamDefaultReader(stream); + let reading = false; + let canceled1 = false; + let canceled2 = false; + let reason1; + let reason2; + let branch1; + let branch2; + let resolveCancelPromise; + const cancelPromise = newPromise(resolve => { + resolveCancelPromise = resolve; + }); + function pullAlgorithm() { + if (reading) { + return promiseResolvedWith(undefined); + } + reading = true; + const readRequest = { + _chunkSteps: value => { + // This needs to be delayed a microtask because it takes at least a microtask to detect errors (using + // reader._closedPromise below), and we want errors in stream to error both branches immediately. We cannot let + // successful synchronously-available reads get ahead of asynchronously-available errors. + queueMicrotask(() => { + reading = false; + const value1 = value; + const value2 = value; + // There is no way to access the cloning code right now in the reference implementation. + // If we add one then we'll need an implementation for serializable objects. + // if (!canceled2 && cloneForBranch2) { + // value2 = StructuredDeserialize(StructuredSerialize(value2)); + // } + if (!canceled1) { + ReadableStreamDefaultControllerEnqueue(branch1._readableStreamController, value1); + } + if (!canceled2) { + ReadableStreamDefaultControllerEnqueue(branch2._readableStreamController, value2); + } + }); + }, + _closeSteps: () => { + reading = false; + if (!canceled1) { + ReadableStreamDefaultControllerClose(branch1._readableStreamController); + } + if (!canceled2) { + ReadableStreamDefaultControllerClose(branch2._readableStreamController); + } + if (!canceled1 || !canceled2) { + resolveCancelPromise(undefined); + } + }, + _errorSteps: () => { + reading = false; + } + }; + ReadableStreamDefaultReaderRead(reader, readRequest); + return promiseResolvedWith(undefined); + } + function cancel1Algorithm(reason) { + canceled1 = true; + reason1 = reason; + if (canceled2) { + const compositeReason = CreateArrayFromList([reason1, reason2]); + const cancelResult = ReadableStreamCancel(stream, compositeReason); + resolveCancelPromise(cancelResult); + } + return cancelPromise; + } + function cancel2Algorithm(reason) { + canceled2 = true; + reason2 = reason; + if (canceled1) { + const compositeReason = CreateArrayFromList([reason1, reason2]); + const cancelResult = ReadableStreamCancel(stream, compositeReason); + resolveCancelPromise(cancelResult); + } + return cancelPromise; + } + function startAlgorithm() { + // do nothing + } + branch1 = CreateReadableStream(startAlgorithm, pullAlgorithm, cancel1Algorithm); + branch2 = CreateReadableStream(startAlgorithm, pullAlgorithm, cancel2Algorithm); + uponRejection(reader._closedPromise, (r) => { + ReadableStreamDefaultControllerError(branch1._readableStreamController, r); + ReadableStreamDefaultControllerError(branch2._readableStreamController, r); + if (!canceled1 || !canceled2) { + resolveCancelPromise(undefined); + } + }); + return [branch1, branch2]; + } + + function convertUnderlyingDefaultOrByteSource(source, context) { + assertDictionary(source, context); + const original = source; + const autoAllocateChunkSize = original === null || original === void 0 ? void 0 : original.autoAllocateChunkSize; + const cancel = original === null || original === void 0 ? void 0 : original.cancel; + const pull = original === null || original === void 0 ? void 0 : original.pull; + const start = original === null || original === void 0 ? void 0 : original.start; + const type = original === null || original === void 0 ? void 0 : original.type; + return { + autoAllocateChunkSize: autoAllocateChunkSize === undefined ? + undefined : + convertUnsignedLongLongWithEnforceRange(autoAllocateChunkSize, `${context} has member 'autoAllocateChunkSize' that`), + cancel: cancel === undefined ? + undefined : + convertUnderlyingSourceCancelCallback(cancel, original, `${context} has member 'cancel' that`), + pull: pull === undefined ? + undefined : + convertUnderlyingSourcePullCallback(pull, original, `${context} has member 'pull' that`), + start: start === undefined ? + undefined : + convertUnderlyingSourceStartCallback(start, original, `${context} has member 'start' that`), + type: type === undefined ? undefined : convertReadableStreamType(type, `${context} has member 'type' that`) + }; + } + function convertUnderlyingSourceCancelCallback(fn, original, context) { + assertFunction(fn, context); + return (reason) => promiseCall(fn, original, [reason]); + } + function convertUnderlyingSourcePullCallback(fn, original, context) { + assertFunction(fn, context); + return (controller) => promiseCall(fn, original, [controller]); + } + function convertUnderlyingSourceStartCallback(fn, original, context) { + assertFunction(fn, context); + return (controller) => reflectCall(fn, original, [controller]); + } + function convertReadableStreamType(type, context) { + type = `${type}`; + if (type !== 'bytes') { + throw new TypeError(`${context} '${type}' is not a valid enumeration value for ReadableStreamType`); + } + return type; + } + + function convertReaderOptions(options, context) { + assertDictionary(options, context); + const mode = options === null || options === void 0 ? void 0 : options.mode; + return { + mode: mode === undefined ? undefined : convertReadableStreamReaderMode(mode, `${context} has member 'mode' that`) + }; + } + function convertReadableStreamReaderMode(mode, context) { + mode = `${mode}`; + if (mode !== 'byob') { + throw new TypeError(`${context} '${mode}' is not a valid enumeration value for ReadableStreamReaderMode`); + } + return mode; + } + + function convertIteratorOptions(options, context) { + assertDictionary(options, context); + const preventCancel = options === null || options === void 0 ? void 0 : options.preventCancel; + return { preventCancel: Boolean(preventCancel) }; + } + + function convertPipeOptions(options, context) { + assertDictionary(options, context); + const preventAbort = options === null || options === void 0 ? void 0 : options.preventAbort; + const preventCancel = options === null || options === void 0 ? void 0 : options.preventCancel; + const preventClose = options === null || options === void 0 ? void 0 : options.preventClose; + const signal = options === null || options === void 0 ? void 0 : options.signal; + if (signal !== undefined) { + assertAbortSignal(signal, `${context} has member 'signal' that`); + } + return { + preventAbort: Boolean(preventAbort), + preventCancel: Boolean(preventCancel), + preventClose: Boolean(preventClose), + signal + }; + } + function assertAbortSignal(signal, context) { + if (!isAbortSignal(signal)) { + throw new TypeError(`${context} is not an AbortSignal.`); + } + } + + function convertReadableWritablePair(pair, context) { + assertDictionary(pair, context); + const readable = pair === null || pair === void 0 ? void 0 : pair.readable; + assertRequiredField(readable, 'readable', 'ReadableWritablePair'); + assertReadableStream(readable, `${context} has member 'readable' that`); + const writable = pair === null || pair === void 0 ? void 0 : pair.writable; + assertRequiredField(writable, 'writable', 'ReadableWritablePair'); + assertWritableStream(writable, `${context} has member 'writable' that`); + return { readable, writable }; + } + + /** + * A readable stream represents a source of data, from which you can read. + * + * @public + */ + class ReadableStream$1 { + constructor(rawUnderlyingSource = {}, rawStrategy = {}) { + if (rawUnderlyingSource === undefined) { + rawUnderlyingSource = null; + } + else { + assertObject(rawUnderlyingSource, 'First parameter'); + } + const strategy = convertQueuingStrategy(rawStrategy, 'Second parameter'); + const underlyingSource = convertUnderlyingDefaultOrByteSource(rawUnderlyingSource, 'First parameter'); + InitializeReadableStream(this); + if (underlyingSource.type === 'bytes') { + if (strategy.size !== undefined) { + throw new RangeError('The strategy for a byte stream cannot have a size function'); + } + const highWaterMark = ExtractHighWaterMark(strategy, 0); + SetUpReadableByteStreamControllerFromUnderlyingSource(this, underlyingSource, highWaterMark); + } + else { + const sizeAlgorithm = ExtractSizeAlgorithm(strategy); + const highWaterMark = ExtractHighWaterMark(strategy, 1); + SetUpReadableStreamDefaultControllerFromUnderlyingSource(this, underlyingSource, highWaterMark, sizeAlgorithm); + } + } + /** + * Whether or not the readable stream is locked to a {@link ReadableStreamDefaultReader | reader}. + */ + get locked() { + if (!IsReadableStream(this)) { + throw streamBrandCheckException$1('locked'); + } + return IsReadableStreamLocked(this); + } + /** + * Cancels the stream, signaling a loss of interest in the stream by a consumer. + * + * The supplied `reason` argument will be given to the underlying source's {@link UnderlyingSource.cancel | cancel()} + * method, which might or might not use it. + */ + cancel(reason = undefined) { + if (!IsReadableStream(this)) { + return promiseRejectedWith(streamBrandCheckException$1('cancel')); + } + if (IsReadableStreamLocked(this)) { + return promiseRejectedWith(new TypeError('Cannot cancel a stream that already has a reader')); + } + return ReadableStreamCancel(this, reason); + } + getReader(rawOptions = undefined) { + if (!IsReadableStream(this)) { + throw streamBrandCheckException$1('getReader'); + } + const options = convertReaderOptions(rawOptions, 'First parameter'); + if (options.mode === undefined) { + return AcquireReadableStreamDefaultReader(this); + } + return AcquireReadableStreamBYOBReader(this); + } + pipeThrough(rawTransform, rawOptions = {}) { + if (!IsReadableStream(this)) { + throw streamBrandCheckException$1('pipeThrough'); + } + assertRequiredArgument(rawTransform, 1, 'pipeThrough'); + const transform = convertReadableWritablePair(rawTransform, 'First parameter'); + const options = convertPipeOptions(rawOptions, 'Second parameter'); + if (IsReadableStreamLocked(this)) { + throw new TypeError('ReadableStream.prototype.pipeThrough cannot be used on a locked ReadableStream'); + } + if (IsWritableStreamLocked(transform.writable)) { + throw new TypeError('ReadableStream.prototype.pipeThrough cannot be used on a locked WritableStream'); + } + const promise = ReadableStreamPipeTo(this, transform.writable, options.preventClose, options.preventAbort, options.preventCancel, options.signal); + setPromiseIsHandledToTrue(promise); + return transform.readable; + } + pipeTo(destination, rawOptions = {}) { + if (!IsReadableStream(this)) { + return promiseRejectedWith(streamBrandCheckException$1('pipeTo')); + } + if (destination === undefined) { + return promiseRejectedWith(`Parameter 1 is required in 'pipeTo'.`); + } + if (!IsWritableStream(destination)) { + return promiseRejectedWith(new TypeError(`ReadableStream.prototype.pipeTo's first argument must be a WritableStream`)); + } + let options; + try { + options = convertPipeOptions(rawOptions, 'Second parameter'); + } + catch (e) { + return promiseRejectedWith(e); + } + if (IsReadableStreamLocked(this)) { + return promiseRejectedWith(new TypeError('ReadableStream.prototype.pipeTo cannot be used on a locked ReadableStream')); + } + if (IsWritableStreamLocked(destination)) { + return promiseRejectedWith(new TypeError('ReadableStream.prototype.pipeTo cannot be used on a locked WritableStream')); + } + return ReadableStreamPipeTo(this, destination, options.preventClose, options.preventAbort, options.preventCancel, options.signal); + } + /** + * Tees this readable stream, returning a two-element array containing the two resulting branches as + * new {@link ReadableStream} instances. + * + * Teeing a stream will lock it, preventing any other consumer from acquiring a reader. + * To cancel the stream, cancel both of the resulting branches; a composite cancellation reason will then be + * propagated to the stream's underlying source. + * + * Note that the chunks seen in each branch will be the same object. If the chunks are not immutable, + * this could allow interference between the two branches. + */ + tee() { + if (!IsReadableStream(this)) { + throw streamBrandCheckException$1('tee'); + } + const branches = ReadableStreamTee(this); + return CreateArrayFromList(branches); + } + values(rawOptions = undefined) { + if (!IsReadableStream(this)) { + throw streamBrandCheckException$1('values'); + } + const options = convertIteratorOptions(rawOptions, 'First parameter'); + return AcquireReadableStreamAsyncIterator(this, options.preventCancel); + } + } + Object.defineProperties(ReadableStream$1.prototype, { + cancel: { enumerable: true }, + getReader: { enumerable: true }, + pipeThrough: { enumerable: true }, + pipeTo: { enumerable: true }, + tee: { enumerable: true }, + values: { enumerable: true }, + locked: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(ReadableStream$1.prototype, SymbolPolyfill.toStringTag, { + value: 'ReadableStream', + configurable: true + }); + } + if (typeof SymbolPolyfill.asyncIterator === 'symbol') { + Object.defineProperty(ReadableStream$1.prototype, SymbolPolyfill.asyncIterator, { + value: ReadableStream$1.prototype.values, + writable: true, + configurable: true + }); + } + // Abstract operations for the ReadableStream. + // Throws if and only if startAlgorithm throws. + function CreateReadableStream(startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark = 1, sizeAlgorithm = () => 1) { + const stream = Object.create(ReadableStream$1.prototype); + InitializeReadableStream(stream); + const controller = Object.create(ReadableStreamDefaultController.prototype); + SetUpReadableStreamDefaultController(stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, sizeAlgorithm); + return stream; + } + function InitializeReadableStream(stream) { + stream._state = 'readable'; + stream._reader = undefined; + stream._storedError = undefined; + stream._disturbed = false; + } + function IsReadableStream(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_readableStreamController')) { + return false; + } + return true; + } + function IsReadableStreamLocked(stream) { + if (stream._reader === undefined) { + return false; + } + return true; + } + // ReadableStream API exposed for controllers. + function ReadableStreamCancel(stream, reason) { + stream._disturbed = true; + if (stream._state === 'closed') { + return promiseResolvedWith(undefined); + } + if (stream._state === 'errored') { + return promiseRejectedWith(stream._storedError); + } + ReadableStreamClose(stream); + const sourceCancelPromise = stream._readableStreamController[CancelSteps](reason); + return transformPromiseWith(sourceCancelPromise, noop); + } + function ReadableStreamClose(stream) { + stream._state = 'closed'; + const reader = stream._reader; + if (reader === undefined) { + return; + } + defaultReaderClosedPromiseResolve(reader); + if (IsReadableStreamDefaultReader(reader)) { + reader._readRequests.forEach(readRequest => { + readRequest._closeSteps(); + }); + reader._readRequests = new SimpleQueue(); + } + } + function ReadableStreamError(stream, e) { + stream._state = 'errored'; + stream._storedError = e; + const reader = stream._reader; + if (reader === undefined) { + return; + } + defaultReaderClosedPromiseReject(reader, e); + if (IsReadableStreamDefaultReader(reader)) { + reader._readRequests.forEach(readRequest => { + readRequest._errorSteps(e); + }); + reader._readRequests = new SimpleQueue(); + } + else { + reader._readIntoRequests.forEach(readIntoRequest => { + readIntoRequest._errorSteps(e); + }); + reader._readIntoRequests = new SimpleQueue(); + } + } + // Helper functions for the ReadableStream. + function streamBrandCheckException$1(name) { + return new TypeError(`ReadableStream.prototype.${name} can only be used on a ReadableStream`); + } + + function convertQueuingStrategyInit(init, context) { + assertDictionary(init, context); + const highWaterMark = init === null || init === void 0 ? void 0 : init.highWaterMark; + assertRequiredField(highWaterMark, 'highWaterMark', 'QueuingStrategyInit'); + return { + highWaterMark: convertUnrestrictedDouble(highWaterMark) + }; + } + + const byteLengthSizeFunction = function size(chunk) { + return chunk.byteLength; + }; + /** + * A queuing strategy that counts the number of bytes in each chunk. + * + * @public + */ + class ByteLengthQueuingStrategy { + constructor(options) { + assertRequiredArgument(options, 1, 'ByteLengthQueuingStrategy'); + options = convertQueuingStrategyInit(options, 'First parameter'); + this._byteLengthQueuingStrategyHighWaterMark = options.highWaterMark; + } + /** + * Returns the high water mark provided to the constructor. + */ + get highWaterMark() { + if (!IsByteLengthQueuingStrategy(this)) { + throw byteLengthBrandCheckException('highWaterMark'); + } + return this._byteLengthQueuingStrategyHighWaterMark; + } + /** + * Measures the size of `chunk` by returning the value of its `byteLength` property. + */ + get size() { + if (!IsByteLengthQueuingStrategy(this)) { + throw byteLengthBrandCheckException('size'); + } + return byteLengthSizeFunction; + } + } + Object.defineProperties(ByteLengthQueuingStrategy.prototype, { + highWaterMark: { enumerable: true }, + size: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(ByteLengthQueuingStrategy.prototype, SymbolPolyfill.toStringTag, { + value: 'ByteLengthQueuingStrategy', + configurable: true + }); + } + // Helper functions for the ByteLengthQueuingStrategy. + function byteLengthBrandCheckException(name) { + return new TypeError(`ByteLengthQueuingStrategy.prototype.${name} can only be used on a ByteLengthQueuingStrategy`); + } + function IsByteLengthQueuingStrategy(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_byteLengthQueuingStrategyHighWaterMark')) { + return false; + } + return true; + } + + const countSizeFunction = function size() { + return 1; + }; + /** + * A queuing strategy that counts the number of chunks. + * + * @public + */ + class CountQueuingStrategy { + constructor(options) { + assertRequiredArgument(options, 1, 'CountQueuingStrategy'); + options = convertQueuingStrategyInit(options, 'First parameter'); + this._countQueuingStrategyHighWaterMark = options.highWaterMark; + } + /** + * Returns the high water mark provided to the constructor. + */ + get highWaterMark() { + if (!IsCountQueuingStrategy(this)) { + throw countBrandCheckException('highWaterMark'); + } + return this._countQueuingStrategyHighWaterMark; + } + /** + * Measures the size of `chunk` by always returning 1. + * This ensures that the total queue size is a count of the number of chunks in the queue. + */ + get size() { + if (!IsCountQueuingStrategy(this)) { + throw countBrandCheckException('size'); + } + return countSizeFunction; + } + } + Object.defineProperties(CountQueuingStrategy.prototype, { + highWaterMark: { enumerable: true }, + size: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(CountQueuingStrategy.prototype, SymbolPolyfill.toStringTag, { + value: 'CountQueuingStrategy', + configurable: true + }); + } + // Helper functions for the CountQueuingStrategy. + function countBrandCheckException(name) { + return new TypeError(`CountQueuingStrategy.prototype.${name} can only be used on a CountQueuingStrategy`); + } + function IsCountQueuingStrategy(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_countQueuingStrategyHighWaterMark')) { + return false; + } + return true; + } + + function convertTransformer(original, context) { + assertDictionary(original, context); + const flush = original === null || original === void 0 ? void 0 : original.flush; + const readableType = original === null || original === void 0 ? void 0 : original.readableType; + const start = original === null || original === void 0 ? void 0 : original.start; + const transform = original === null || original === void 0 ? void 0 : original.transform; + const writableType = original === null || original === void 0 ? void 0 : original.writableType; + return { + flush: flush === undefined ? + undefined : + convertTransformerFlushCallback(flush, original, `${context} has member 'flush' that`), + readableType, + start: start === undefined ? + undefined : + convertTransformerStartCallback(start, original, `${context} has member 'start' that`), + transform: transform === undefined ? + undefined : + convertTransformerTransformCallback(transform, original, `${context} has member 'transform' that`), + writableType + }; + } + function convertTransformerFlushCallback(fn, original, context) { + assertFunction(fn, context); + return (controller) => promiseCall(fn, original, [controller]); + } + function convertTransformerStartCallback(fn, original, context) { + assertFunction(fn, context); + return (controller) => reflectCall(fn, original, [controller]); + } + function convertTransformerTransformCallback(fn, original, context) { + assertFunction(fn, context); + return (chunk, controller) => promiseCall(fn, original, [chunk, controller]); + } + + // Class TransformStream + /** + * A transform stream consists of a pair of streams: a {@link WritableStream | writable stream}, + * known as its writable side, and a {@link ReadableStream | readable stream}, known as its readable side. + * In a manner specific to the transform stream in question, writes to the writable side result in new data being + * made available for reading from the readable side. + * + * @public + */ + class TransformStream$1 { + constructor(rawTransformer = {}, rawWritableStrategy = {}, rawReadableStrategy = {}) { + if (rawTransformer === undefined) { + rawTransformer = null; + } + const writableStrategy = convertQueuingStrategy(rawWritableStrategy, 'Second parameter'); + const readableStrategy = convertQueuingStrategy(rawReadableStrategy, 'Third parameter'); + const transformer = convertTransformer(rawTransformer, 'First parameter'); + if (transformer.readableType !== undefined) { + throw new RangeError('Invalid readableType specified'); + } + if (transformer.writableType !== undefined) { + throw new RangeError('Invalid writableType specified'); + } + const readableHighWaterMark = ExtractHighWaterMark(readableStrategy, 0); + const readableSizeAlgorithm = ExtractSizeAlgorithm(readableStrategy); + const writableHighWaterMark = ExtractHighWaterMark(writableStrategy, 1); + const writableSizeAlgorithm = ExtractSizeAlgorithm(writableStrategy); + let startPromise_resolve; + const startPromise = newPromise(resolve => { + startPromise_resolve = resolve; + }); + InitializeTransformStream(this, startPromise, writableHighWaterMark, writableSizeAlgorithm, readableHighWaterMark, readableSizeAlgorithm); + SetUpTransformStreamDefaultControllerFromTransformer(this, transformer); + if (transformer.start !== undefined) { + startPromise_resolve(transformer.start(this._transformStreamController)); + } + else { + startPromise_resolve(undefined); + } + } + /** + * The readable side of the transform stream. + */ + get readable() { + if (!IsTransformStream(this)) { + throw streamBrandCheckException('readable'); + } + return this._readable; + } + /** + * The writable side of the transform stream. + */ + get writable() { + if (!IsTransformStream(this)) { + throw streamBrandCheckException('writable'); + } + return this._writable; + } + } + Object.defineProperties(TransformStream$1.prototype, { + readable: { enumerable: true }, + writable: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(TransformStream$1.prototype, SymbolPolyfill.toStringTag, { + value: 'TransformStream', + configurable: true + }); + } + function InitializeTransformStream(stream, startPromise, writableHighWaterMark, writableSizeAlgorithm, readableHighWaterMark, readableSizeAlgorithm) { + function startAlgorithm() { + return startPromise; + } + function writeAlgorithm(chunk) { + return TransformStreamDefaultSinkWriteAlgorithm(stream, chunk); + } + function abortAlgorithm(reason) { + return TransformStreamDefaultSinkAbortAlgorithm(stream, reason); + } + function closeAlgorithm() { + return TransformStreamDefaultSinkCloseAlgorithm(stream); + } + stream._writable = CreateWritableStream(startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, writableHighWaterMark, writableSizeAlgorithm); + function pullAlgorithm() { + return TransformStreamDefaultSourcePullAlgorithm(stream); + } + function cancelAlgorithm(reason) { + TransformStreamErrorWritableAndUnblockWrite(stream, reason); + return promiseResolvedWith(undefined); + } + stream._readable = CreateReadableStream(startAlgorithm, pullAlgorithm, cancelAlgorithm, readableHighWaterMark, readableSizeAlgorithm); + // The [[backpressure]] slot is set to undefined so that it can be initialised by TransformStreamSetBackpressure. + stream._backpressure = undefined; + stream._backpressureChangePromise = undefined; + stream._backpressureChangePromise_resolve = undefined; + TransformStreamSetBackpressure(stream, true); + stream._transformStreamController = undefined; + } + function IsTransformStream(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_transformStreamController')) { + return false; + } + return true; + } + // This is a no-op if both sides are already errored. + function TransformStreamError(stream, e) { + ReadableStreamDefaultControllerError(stream._readable._readableStreamController, e); + TransformStreamErrorWritableAndUnblockWrite(stream, e); + } + function TransformStreamErrorWritableAndUnblockWrite(stream, e) { + TransformStreamDefaultControllerClearAlgorithms(stream._transformStreamController); + WritableStreamDefaultControllerErrorIfNeeded(stream._writable._writableStreamController, e); + if (stream._backpressure) { + // Pretend that pull() was called to permit any pending write() calls to complete. TransformStreamSetBackpressure() + // cannot be called from enqueue() or pull() once the ReadableStream is errored, so this will will be the final time + // _backpressure is set. + TransformStreamSetBackpressure(stream, false); + } + } + function TransformStreamSetBackpressure(stream, backpressure) { + // Passes also when called during construction. + if (stream._backpressureChangePromise !== undefined) { + stream._backpressureChangePromise_resolve(); + } + stream._backpressureChangePromise = newPromise(resolve => { + stream._backpressureChangePromise_resolve = resolve; + }); + stream._backpressure = backpressure; + } + // Class TransformStreamDefaultController + /** + * Allows control of the {@link ReadableStream} and {@link WritableStream} of the associated {@link TransformStream}. + * + * @public + */ + class TransformStreamDefaultController { + constructor() { + throw new TypeError('Illegal constructor'); + } + /** + * Returns the desired size to fill the readable side’s internal queue. It can be negative, if the queue is over-full. + */ + get desiredSize() { + if (!IsTransformStreamDefaultController(this)) { + throw defaultControllerBrandCheckException('desiredSize'); + } + const readableController = this._controlledTransformStream._readable._readableStreamController; + return ReadableStreamDefaultControllerGetDesiredSize(readableController); + } + enqueue(chunk = undefined) { + if (!IsTransformStreamDefaultController(this)) { + throw defaultControllerBrandCheckException('enqueue'); + } + TransformStreamDefaultControllerEnqueue(this, chunk); + } + /** + * Errors both the readable side and the writable side of the controlled transform stream, making all future + * interactions with it fail with the given error `e`. Any chunks queued for transformation will be discarded. + */ + error(reason = undefined) { + if (!IsTransformStreamDefaultController(this)) { + throw defaultControllerBrandCheckException('error'); + } + TransformStreamDefaultControllerError(this, reason); + } + /** + * Closes the readable side and errors the writable side of the controlled transform stream. This is useful when the + * transformer only needs to consume a portion of the chunks written to the writable side. + */ + terminate() { + if (!IsTransformStreamDefaultController(this)) { + throw defaultControllerBrandCheckException('terminate'); + } + TransformStreamDefaultControllerTerminate(this); + } + } + Object.defineProperties(TransformStreamDefaultController.prototype, { + enqueue: { enumerable: true }, + error: { enumerable: true }, + terminate: { enumerable: true }, + desiredSize: { enumerable: true } + }); + if (typeof SymbolPolyfill.toStringTag === 'symbol') { + Object.defineProperty(TransformStreamDefaultController.prototype, SymbolPolyfill.toStringTag, { + value: 'TransformStreamDefaultController', + configurable: true + }); + } + // Transform Stream Default Controller Abstract Operations + function IsTransformStreamDefaultController(x) { + if (!typeIsObject(x)) { + return false; + } + if (!Object.prototype.hasOwnProperty.call(x, '_controlledTransformStream')) { + return false; + } + return true; + } + function SetUpTransformStreamDefaultController(stream, controller, transformAlgorithm, flushAlgorithm) { + controller._controlledTransformStream = stream; + stream._transformStreamController = controller; + controller._transformAlgorithm = transformAlgorithm; + controller._flushAlgorithm = flushAlgorithm; + } + function SetUpTransformStreamDefaultControllerFromTransformer(stream, transformer) { + const controller = Object.create(TransformStreamDefaultController.prototype); + let transformAlgorithm = (chunk) => { + try { + TransformStreamDefaultControllerEnqueue(controller, chunk); + return promiseResolvedWith(undefined); + } + catch (transformResultE) { + return promiseRejectedWith(transformResultE); + } + }; + let flushAlgorithm = () => promiseResolvedWith(undefined); + if (transformer.transform !== undefined) { + transformAlgorithm = chunk => transformer.transform(chunk, controller); + } + if (transformer.flush !== undefined) { + flushAlgorithm = () => transformer.flush(controller); + } + SetUpTransformStreamDefaultController(stream, controller, transformAlgorithm, flushAlgorithm); + } + function TransformStreamDefaultControllerClearAlgorithms(controller) { + controller._transformAlgorithm = undefined; + controller._flushAlgorithm = undefined; + } + function TransformStreamDefaultControllerEnqueue(controller, chunk) { + const stream = controller._controlledTransformStream; + const readableController = stream._readable._readableStreamController; + if (!ReadableStreamDefaultControllerCanCloseOrEnqueue(readableController)) { + throw new TypeError('Readable side is not in a state that permits enqueue'); + } + // We throttle transform invocations based on the backpressure of the ReadableStream, but we still + // accept TransformStreamDefaultControllerEnqueue() calls. + try { + ReadableStreamDefaultControllerEnqueue(readableController, chunk); + } + catch (e) { + // This happens when readableStrategy.size() throws. + TransformStreamErrorWritableAndUnblockWrite(stream, e); + throw stream._readable._storedError; + } + const backpressure = ReadableStreamDefaultControllerHasBackpressure(readableController); + if (backpressure !== stream._backpressure) { + TransformStreamSetBackpressure(stream, true); + } + } + function TransformStreamDefaultControllerError(controller, e) { + TransformStreamError(controller._controlledTransformStream, e); + } + function TransformStreamDefaultControllerPerformTransform(controller, chunk) { + const transformPromise = controller._transformAlgorithm(chunk); + return transformPromiseWith(transformPromise, undefined, r => { + TransformStreamError(controller._controlledTransformStream, r); + throw r; + }); + } + function TransformStreamDefaultControllerTerminate(controller) { + const stream = controller._controlledTransformStream; + const readableController = stream._readable._readableStreamController; + ReadableStreamDefaultControllerClose(readableController); + const error = new TypeError('TransformStream terminated'); + TransformStreamErrorWritableAndUnblockWrite(stream, error); + } + // TransformStreamDefaultSink Algorithms + function TransformStreamDefaultSinkWriteAlgorithm(stream, chunk) { + const controller = stream._transformStreamController; + if (stream._backpressure) { + const backpressureChangePromise = stream._backpressureChangePromise; + return transformPromiseWith(backpressureChangePromise, () => { + const writable = stream._writable; + const state = writable._state; + if (state === 'erroring') { + throw writable._storedError; + } + return TransformStreamDefaultControllerPerformTransform(controller, chunk); + }); + } + return TransformStreamDefaultControllerPerformTransform(controller, chunk); + } + function TransformStreamDefaultSinkAbortAlgorithm(stream, reason) { + // abort() is not called synchronously, so it is possible for abort() to be called when the stream is already + // errored. + TransformStreamError(stream, reason); + return promiseResolvedWith(undefined); + } + function TransformStreamDefaultSinkCloseAlgorithm(stream) { + // stream._readable cannot change after construction, so caching it across a call to user code is safe. + const readable = stream._readable; + const controller = stream._transformStreamController; + const flushPromise = controller._flushAlgorithm(); + TransformStreamDefaultControllerClearAlgorithms(controller); + // Return a promise that is fulfilled with undefined on success. + return transformPromiseWith(flushPromise, () => { + if (readable._state === 'errored') { + throw readable._storedError; + } + ReadableStreamDefaultControllerClose(readable._readableStreamController); + }, r => { + TransformStreamError(stream, r); + throw readable._storedError; + }); + } + // TransformStreamDefaultSource Algorithms + function TransformStreamDefaultSourcePullAlgorithm(stream) { + // Invariant. Enforced by the promises returned by start() and pull(). + TransformStreamSetBackpressure(stream, false); + // Prevent the next pull() call until there is backpressure. + return stream._backpressureChangePromise; + } + // Helper functions for the TransformStreamDefaultController. + function defaultControllerBrandCheckException(name) { + return new TypeError(`TransformStreamDefaultController.prototype.${name} can only be used on a TransformStreamDefaultController`); + } + // Helper functions for the TransformStream. + function streamBrandCheckException(name) { + return new TypeError(`TransformStream.prototype.${name} can only be used on a TransformStream`); + } + + var ponyfill_es6 = /*#__PURE__*/Object.freeze({ + __proto__: null, + ByteLengthQueuingStrategy: ByteLengthQueuingStrategy, + CountQueuingStrategy: CountQueuingStrategy, + ReadableByteStreamController: ReadableByteStreamController, + ReadableStream: ReadableStream$1, + ReadableStreamBYOBReader: ReadableStreamBYOBReader, + ReadableStreamBYOBRequest: ReadableStreamBYOBRequest, + ReadableStreamDefaultController: ReadableStreamDefaultController, + ReadableStreamDefaultReader: ReadableStreamDefaultReader, + TransformStream: TransformStream$1, + TransformStreamDefaultController: TransformStreamDefaultController, + WritableStream: WritableStream$1, + WritableStreamDefaultController: WritableStreamDefaultController, + WritableStreamDefaultWriter: WritableStreamDefaultWriter + }); + + /*! ***************************************************************************** + Copyright (c) Microsoft Corporation. + + Permission to use, copy, modify, and/or distribute this software for any + purpose with or without fee is hereby granted. + + THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH + REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY + AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, + INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM + LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR + OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR + PERFORMANCE OF THIS SOFTWARE. + ***************************************************************************** */ + /* global Reflect, Promise */ + + var extendStatics = function(d, b) { + extendStatics = Object.setPrototypeOf || + ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || + function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; }; + return extendStatics(d, b); + }; + + function __extends(d, b) { + if (typeof b !== "function" && b !== null) + throw new TypeError("Class extends value " + String(b) + " is not a constructor or null"); + extendStatics(d, b); + function __() { this.constructor = d; } + d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); + } + + function assert$1(test) { + if (!test) { + throw new TypeError('Assertion failed'); + } + } + + function noop$1() { + return; + } + function typeIsObject$1(x) { + return (typeof x === 'object' && x !== null) || typeof x === 'function'; + } + + function isStreamConstructor(ctor) { + if (typeof ctor !== 'function') { + return false; + } + var startCalled = false; + try { + new ctor({ + start: function () { + startCalled = true; + } + }); + } + catch (e) { + // ignore + } + return startCalled; + } + function isReadableStream(readable) { + if (!typeIsObject$1(readable)) { + return false; + } + if (typeof readable.getReader !== 'function') { + return false; + } + return true; + } + function isReadableStreamConstructor(ctor) { + if (!isStreamConstructor(ctor)) { + return false; + } + if (!isReadableStream(new ctor())) { + return false; + } + return true; + } + function isWritableStream(writable) { + if (!typeIsObject$1(writable)) { + return false; + } + if (typeof writable.getWriter !== 'function') { + return false; + } + return true; + } + function isWritableStreamConstructor(ctor) { + if (!isStreamConstructor(ctor)) { + return false; + } + if (!isWritableStream(new ctor())) { + return false; + } + return true; + } + function isTransformStream(transform) { + if (!typeIsObject$1(transform)) { + return false; + } + if (!isReadableStream(transform.readable)) { + return false; + } + if (!isWritableStream(transform.writable)) { + return false; + } + return true; + } + function isTransformStreamConstructor(ctor) { + if (!isStreamConstructor(ctor)) { + return false; + } + if (!isTransformStream(new ctor())) { + return false; + } + return true; + } + function supportsByobReader(readable) { + try { + var reader = readable.getReader({ mode: 'byob' }); + reader.releaseLock(); + return true; + } + catch (_a) { + return false; + } + } + function supportsByteSource(ctor) { + try { + new ctor({ type: 'bytes' }); + return true; + } + catch (_a) { + return false; + } + } + + function createReadableStreamWrapper(ctor) { + assert$1(isReadableStreamConstructor(ctor)); + var byteSourceSupported = supportsByteSource(ctor); + return function (readable, _a) { + var _b = _a === void 0 ? {} : _a, type = _b.type; + type = parseReadableType(type); + if (type === 'bytes' && !byteSourceSupported) { + type = undefined; + } + if (readable.constructor === ctor) { + if (type !== 'bytes' || supportsByobReader(readable)) { + return readable; + } + } + if (type === 'bytes') { + var source = createWrappingReadableSource(readable, { type: type }); + return new ctor(source); + } + else { + var source = createWrappingReadableSource(readable); + return new ctor(source); + } + }; + } + function createWrappingReadableSource(readable, _a) { + var _b = _a === void 0 ? {} : _a, type = _b.type; + assert$1(isReadableStream(readable)); + assert$1(readable.locked === false); + type = parseReadableType(type); + var source; + if (type === 'bytes') { + source = new WrappingReadableByteStreamSource(readable); + } + else { + source = new WrappingReadableStreamDefaultSource(readable); + } + return source; + } + function parseReadableType(type) { + var typeString = String(type); + if (typeString === 'bytes') { + return typeString; + } + else if (type === undefined) { + return type; + } + else { + throw new RangeError('Invalid type is specified'); + } + } + var AbstractWrappingReadableStreamSource = /** @class */ (function () { + function AbstractWrappingReadableStreamSource(underlyingStream) { + this._underlyingReader = undefined; + this._readerMode = undefined; + this._readableStreamController = undefined; + this._pendingRead = undefined; + this._underlyingStream = underlyingStream; + // always keep a reader attached to detect close/error + this._attachDefaultReader(); + } + AbstractWrappingReadableStreamSource.prototype.start = function (controller) { + this._readableStreamController = controller; + }; + AbstractWrappingReadableStreamSource.prototype.cancel = function (reason) { + assert$1(this._underlyingReader !== undefined); + return this._underlyingReader.cancel(reason); + }; + AbstractWrappingReadableStreamSource.prototype._attachDefaultReader = function () { + if (this._readerMode === "default" /* DEFAULT */) { + return; + } + this._detachReader(); + var reader = this._underlyingStream.getReader(); + this._readerMode = "default" /* DEFAULT */; + this._attachReader(reader); + }; + AbstractWrappingReadableStreamSource.prototype._attachReader = function (reader) { + var _this = this; + assert$1(this._underlyingReader === undefined); + this._underlyingReader = reader; + var closed = this._underlyingReader.closed; + if (!closed) { + return; + } + closed + .then(function () { return _this._finishPendingRead(); }) + .then(function () { + if (reader === _this._underlyingReader) { + _this._readableStreamController.close(); + } + }, function (reason) { + if (reader === _this._underlyingReader) { + _this._readableStreamController.error(reason); + } + }) + .catch(noop$1); + }; + AbstractWrappingReadableStreamSource.prototype._detachReader = function () { + if (this._underlyingReader === undefined) { + return; + } + this._underlyingReader.releaseLock(); + this._underlyingReader = undefined; + this._readerMode = undefined; + }; + AbstractWrappingReadableStreamSource.prototype._pullWithDefaultReader = function () { + var _this = this; + this._attachDefaultReader(); + // TODO Backpressure? + var read = this._underlyingReader.read() + .then(function (result) { + var controller = _this._readableStreamController; + if (result.done) { + _this._tryClose(); + } + else { + controller.enqueue(result.value); + } + }); + this._setPendingRead(read); + return read; + }; + AbstractWrappingReadableStreamSource.prototype._tryClose = function () { + try { + this._readableStreamController.close(); + } + catch (_a) { + // already errored or closed + } + }; + AbstractWrappingReadableStreamSource.prototype._setPendingRead = function (readPromise) { + var _this = this; + var pendingRead; + var finishRead = function () { + if (_this._pendingRead === pendingRead) { + _this._pendingRead = undefined; + } + }; + this._pendingRead = pendingRead = readPromise.then(finishRead, finishRead); + }; + AbstractWrappingReadableStreamSource.prototype._finishPendingRead = function () { + var _this = this; + if (!this._pendingRead) { + return undefined; + } + var afterRead = function () { return _this._finishPendingRead(); }; + return this._pendingRead.then(afterRead, afterRead); + }; + return AbstractWrappingReadableStreamSource; + }()); + var WrappingReadableStreamDefaultSource = /** @class */ (function (_super) { + __extends(WrappingReadableStreamDefaultSource, _super); + function WrappingReadableStreamDefaultSource() { + return _super !== null && _super.apply(this, arguments) || this; + } + WrappingReadableStreamDefaultSource.prototype.pull = function () { + return this._pullWithDefaultReader(); + }; + return WrappingReadableStreamDefaultSource; + }(AbstractWrappingReadableStreamSource)); + function toUint8Array(view) { + return new Uint8Array(view.buffer, view.byteOffset, view.byteLength); + } + function copyArrayBufferView(from, to) { + var fromArray = toUint8Array(from); + var toArray = toUint8Array(to); + toArray.set(fromArray, 0); + } + var WrappingReadableByteStreamSource = /** @class */ (function (_super) { + __extends(WrappingReadableByteStreamSource, _super); + function WrappingReadableByteStreamSource(underlyingStream) { + var _this = this; + var supportsByob = supportsByobReader(underlyingStream); + _this = _super.call(this, underlyingStream) || this; + _this._supportsByob = supportsByob; + return _this; + } + Object.defineProperty(WrappingReadableByteStreamSource.prototype, "type", { + get: function () { + return 'bytes'; + }, + enumerable: false, + configurable: true + }); + WrappingReadableByteStreamSource.prototype._attachByobReader = function () { + if (this._readerMode === "byob" /* BYOB */) { + return; + } + assert$1(this._supportsByob); + this._detachReader(); + var reader = this._underlyingStream.getReader({ mode: 'byob' }); + this._readerMode = "byob" /* BYOB */; + this._attachReader(reader); + }; + WrappingReadableByteStreamSource.prototype.pull = function () { + if (this._supportsByob) { + var byobRequest = this._readableStreamController.byobRequest; + if (byobRequest) { + return this._pullWithByobRequest(byobRequest); + } + } + return this._pullWithDefaultReader(); + }; + WrappingReadableByteStreamSource.prototype._pullWithByobRequest = function (byobRequest) { + var _this = this; + this._attachByobReader(); + // reader.read(view) detaches the input view, therefore we cannot pass byobRequest.view directly + // create a separate buffer to read into, then copy that to byobRequest.view + var buffer = new Uint8Array(byobRequest.view.byteLength); + // TODO Backpressure? + var read = this._underlyingReader.read(buffer) + .then(function (result) { + _this._readableStreamController; + if (result.done) { + _this._tryClose(); + byobRequest.respond(0); + } + else { + copyArrayBufferView(result.value, byobRequest.view); + byobRequest.respond(result.value.byteLength); + } + }); + this._setPendingRead(read); + return read; + }; + return WrappingReadableByteStreamSource; + }(AbstractWrappingReadableStreamSource)); + + function createWritableStreamWrapper(ctor) { + assert$1(isWritableStreamConstructor(ctor)); + return function (writable) { + if (writable.constructor === ctor) { + return writable; + } + var sink = createWrappingWritableSink(writable); + return new ctor(sink); + }; + } + function createWrappingWritableSink(writable) { + assert$1(isWritableStream(writable)); + assert$1(writable.locked === false); + var writer = writable.getWriter(); + return new WrappingWritableStreamSink(writer); + } + var WrappingWritableStreamSink = /** @class */ (function () { + function WrappingWritableStreamSink(underlyingWriter) { + var _this = this; + this._writableStreamController = undefined; + this._pendingWrite = undefined; + this._state = "writable" /* WRITABLE */; + this._storedError = undefined; + this._underlyingWriter = underlyingWriter; + this._errorPromise = new Promise(function (resolve, reject) { + _this._errorPromiseReject = reject; + }); + this._errorPromise.catch(noop$1); + } + WrappingWritableStreamSink.prototype.start = function (controller) { + var _this = this; + this._writableStreamController = controller; + this._underlyingWriter.closed + .then(function () { + _this._state = "closed" /* CLOSED */; + }) + .catch(function (reason) { return _this._finishErroring(reason); }); + }; + WrappingWritableStreamSink.prototype.write = function (chunk) { + var _this = this; + var writer = this._underlyingWriter; + // Detect past errors + if (writer.desiredSize === null) { + return writer.ready; + } + var writeRequest = writer.write(chunk); + // Detect future errors + writeRequest.catch(function (reason) { return _this._finishErroring(reason); }); + writer.ready.catch(function (reason) { return _this._startErroring(reason); }); + // Reject write when errored + var write = Promise.race([writeRequest, this._errorPromise]); + this._setPendingWrite(write); + return write; + }; + WrappingWritableStreamSink.prototype.close = function () { + var _this = this; + if (this._pendingWrite === undefined) { + return this._underlyingWriter.close(); + } + return this._finishPendingWrite().then(function () { return _this.close(); }); + }; + WrappingWritableStreamSink.prototype.abort = function (reason) { + if (this._state === "errored" /* ERRORED */) { + return undefined; + } + var writer = this._underlyingWriter; + return writer.abort(reason); + }; + WrappingWritableStreamSink.prototype._setPendingWrite = function (writePromise) { + var _this = this; + var pendingWrite; + var finishWrite = function () { + if (_this._pendingWrite === pendingWrite) { + _this._pendingWrite = undefined; + } + }; + this._pendingWrite = pendingWrite = writePromise.then(finishWrite, finishWrite); + }; + WrappingWritableStreamSink.prototype._finishPendingWrite = function () { + var _this = this; + if (this._pendingWrite === undefined) { + return Promise.resolve(); + } + var afterWrite = function () { return _this._finishPendingWrite(); }; + return this._pendingWrite.then(afterWrite, afterWrite); + }; + WrappingWritableStreamSink.prototype._startErroring = function (reason) { + var _this = this; + if (this._state === "writable" /* WRITABLE */) { + this._state = "erroring" /* ERRORING */; + this._storedError = reason; + var afterWrite = function () { return _this._finishErroring(reason); }; + if (this._pendingWrite === undefined) { + afterWrite(); + } + else { + this._finishPendingWrite().then(afterWrite, afterWrite); + } + this._writableStreamController.error(reason); + } + }; + WrappingWritableStreamSink.prototype._finishErroring = function (reason) { + if (this._state === "writable" /* WRITABLE */) { + this._startErroring(reason); + } + if (this._state === "erroring" /* ERRORING */) { + this._state = "errored" /* ERRORED */; + this._errorPromiseReject(this._storedError); + } + }; + return WrappingWritableStreamSink; + }()); + + function createTransformStreamWrapper(ctor) { + assert$1(isTransformStreamConstructor(ctor)); + return function (transform) { + if (transform.constructor === ctor) { + return transform; + } + var transformer = createWrappingTransformer(transform); + return new ctor(transformer); + }; + } + function createWrappingTransformer(transform) { + assert$1(isTransformStream(transform)); + var readable = transform.readable, writable = transform.writable; + assert$1(readable.locked === false); + assert$1(writable.locked === false); + var reader = readable.getReader(); + var writer; + try { + writer = writable.getWriter(); + } + catch (e) { + reader.releaseLock(); // do not leak reader + throw e; + } + return new WrappingTransformStreamTransformer(reader, writer); + } + var WrappingTransformStreamTransformer = /** @class */ (function () { + function WrappingTransformStreamTransformer(reader, writer) { + var _this = this; + this._transformStreamController = undefined; + this._onRead = function (result) { + if (result.done) { + return; + } + _this._transformStreamController.enqueue(result.value); + return _this._reader.read().then(_this._onRead); + }; + this._onError = function (reason) { + _this._flushReject(reason); + _this._transformStreamController.error(reason); + _this._reader.cancel(reason).catch(noop$1); + _this._writer.abort(reason).catch(noop$1); + }; + this._onTerminate = function () { + _this._flushResolve(); + _this._transformStreamController.terminate(); + var error = new TypeError('TransformStream terminated'); + _this._writer.abort(error).catch(noop$1); + }; + this._reader = reader; + this._writer = writer; + this._flushPromise = new Promise(function (resolve, reject) { + _this._flushResolve = resolve; + _this._flushReject = reject; + }); + } + WrappingTransformStreamTransformer.prototype.start = function (controller) { + this._transformStreamController = controller; + this._reader.read() + .then(this._onRead) + .then(this._onTerminate, this._onError); + var readerClosed = this._reader.closed; + if (readerClosed) { + readerClosed + .then(this._onTerminate, this._onError); + } + }; + WrappingTransformStreamTransformer.prototype.transform = function (chunk) { + return this._writer.write(chunk); + }; + WrappingTransformStreamTransformer.prototype.flush = function () { + var _this = this; + return this._writer.close() + .then(function () { return _this._flushPromise; }); + }; + return WrappingTransformStreamTransformer; + }()); + + var webStreamsAdapter = /*#__PURE__*/Object.freeze({ + __proto__: null, + createReadableStreamWrapper: createReadableStreamWrapper, + createTransformStreamWrapper: createTransformStreamWrapper, + createWrappingReadableSource: createWrappingReadableSource, + createWrappingTransformer: createWrappingTransformer, + createWrappingWritableSink: createWrappingWritableSink, + createWritableStreamWrapper: createWritableStreamWrapper + }); + + var bn = createCommonjsModule(function (module) { + (function (module, exports) { + + // Utils + function assert (val, msg) { + if (!val) throw new Error(msg || 'Assertion failed'); + } + + // Could use `inherits` module, but don't want to move from single file + // architecture yet. + function inherits (ctor, superCtor) { + ctor.super_ = superCtor; + var TempCtor = function () {}; + TempCtor.prototype = superCtor.prototype; + ctor.prototype = new TempCtor(); + ctor.prototype.constructor = ctor; + } + + // BN + + function BN (number, base, endian) { + if (BN.isBN(number)) { + return number; + } + + this.negative = 0; + this.words = null; + this.length = 0; + + // Reduction context + this.red = null; + + if (number !== null) { + if (base === 'le' || base === 'be') { + endian = base; + base = 10; + } + + this._init(number || 0, base || 10, endian || 'be'); + } + } + if (typeof module === 'object') { + module.exports = BN; + } else { + exports.BN = BN; + } + + BN.BN = BN; + BN.wordSize = 26; + + var Buffer; + try { + Buffer = void('buffer').Buffer; + } catch (e) { + } + + BN.isBN = function isBN (num) { + if (num instanceof BN) { + return true; + } + + return num !== null && typeof num === 'object' && + num.constructor.wordSize === BN.wordSize && Array.isArray(num.words); + }; + + BN.max = function max (left, right) { + if (left.cmp(right) > 0) return left; + return right; + }; + + BN.min = function min (left, right) { + if (left.cmp(right) < 0) return left; + return right; + }; + + BN.prototype._init = function init (number, base, endian) { + if (typeof number === 'number') { + return this._initNumber(number, base, endian); + } + + if (typeof number === 'object') { + return this._initArray(number, base, endian); + } + + if (base === 'hex') { + base = 16; + } + assert(base === (base | 0) && base >= 2 && base <= 36); + + number = number.toString().replace(/\s+/g, ''); + var start = 0; + if (number[0] === '-') { + start++; + } + + if (base === 16) { + this._parseHex(number, start); + } else { + this._parseBase(number, base, start); + } + + if (number[0] === '-') { + this.negative = 1; + } + + this.strip(); + + if (endian !== 'le') return; + + this._initArray(this.toArray(), base, endian); + }; + + BN.prototype._initNumber = function _initNumber (number, base, endian) { + if (number < 0) { + this.negative = 1; + number = -number; + } + if (number < 0x4000000) { + this.words = [ number & 0x3ffffff ]; + this.length = 1; + } else if (number < 0x10000000000000) { + this.words = [ + number & 0x3ffffff, + (number / 0x4000000) & 0x3ffffff + ]; + this.length = 2; + } else { + assert(number < 0x20000000000000); // 2 ^ 53 (unsafe) + this.words = [ + number & 0x3ffffff, + (number / 0x4000000) & 0x3ffffff, + 1 + ]; + this.length = 3; + } + + if (endian !== 'le') return; + + // Reverse the bytes + this._initArray(this.toArray(), base, endian); + }; + + BN.prototype._initArray = function _initArray (number, base, endian) { + // Perhaps a Uint8Array + assert(typeof number.length === 'number'); + if (number.length <= 0) { + this.words = [ 0 ]; + this.length = 1; + return this; + } + + this.length = Math.ceil(number.length / 3); + this.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + this.words[i] = 0; + } + + var j, w; + var off = 0; + if (endian === 'be') { + for (i = number.length - 1, j = 0; i >= 0; i -= 3) { + w = number[i] | (number[i - 1] << 8) | (number[i - 2] << 16); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + } else if (endian === 'le') { + for (i = 0, j = 0; i < number.length; i += 3) { + w = number[i] | (number[i + 1] << 8) | (number[i + 2] << 16); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + } + return this.strip(); + }; + + function parseHex (str, start, end) { + var r = 0; + var len = Math.min(str.length, end); + for (var i = start; i < len; i++) { + var c = str.charCodeAt(i) - 48; + + r <<= 4; + + // 'a' - 'f' + if (c >= 49 && c <= 54) { + r |= c - 49 + 0xa; + + // 'A' - 'F' + } else if (c >= 17 && c <= 22) { + r |= c - 17 + 0xa; + + // '0' - '9' + } else { + r |= c & 0xf; + } + } + return r; + } + + BN.prototype._parseHex = function _parseHex (number, start) { + // Create possibly bigger array to ensure that it fits the number + this.length = Math.ceil((number.length - start) / 6); + this.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + this.words[i] = 0; + } + + var j, w; + // Scan 24-bit chunks and add them to the number + var off = 0; + for (i = number.length - 6, j = 0; i >= start; i -= 6) { + w = parseHex(number, i, i + 6); + this.words[j] |= (w << off) & 0x3ffffff; + // NOTE: `0x3fffff` is intentional here, 26bits max shift + 24bit hex limb + this.words[j + 1] |= w >>> (26 - off) & 0x3fffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + if (i + 6 !== start) { + w = parseHex(number, start, i + 6); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] |= w >>> (26 - off) & 0x3fffff; + } + this.strip(); + }; + + function parseBase (str, start, end, mul) { + var r = 0; + var len = Math.min(str.length, end); + for (var i = start; i < len; i++) { + var c = str.charCodeAt(i) - 48; + + r *= mul; + + // 'a' + if (c >= 49) { + r += c - 49 + 0xa; + + // 'A' + } else if (c >= 17) { + r += c - 17 + 0xa; + + // '0' - '9' + } else { + r += c; + } + } + return r; + } + + BN.prototype._parseBase = function _parseBase (number, base, start) { + // Initialize as zero + this.words = [ 0 ]; + this.length = 1; + + // Find length of limb in base + for (var limbLen = 0, limbPow = 1; limbPow <= 0x3ffffff; limbPow *= base) { + limbLen++; + } + limbLen--; + limbPow = (limbPow / base) | 0; + + var total = number.length - start; + var mod = total % limbLen; + var end = Math.min(total, total - mod) + start; + + var word = 0; + for (var i = start; i < end; i += limbLen) { + word = parseBase(number, i, i + limbLen, base); + + this.imuln(limbPow); + if (this.words[0] + word < 0x4000000) { + this.words[0] += word; + } else { + this._iaddn(word); + } + } + + if (mod !== 0) { + var pow = 1; + word = parseBase(number, i, number.length, base); + + for (i = 0; i < mod; i++) { + pow *= base; + } + + this.imuln(pow); + if (this.words[0] + word < 0x4000000) { + this.words[0] += word; + } else { + this._iaddn(word); + } + } + }; + + BN.prototype.copy = function copy (dest) { + dest.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + dest.words[i] = this.words[i]; + } + dest.length = this.length; + dest.negative = this.negative; + dest.red = this.red; + }; + + BN.prototype.clone = function clone () { + var r = new BN(null); + this.copy(r); + return r; + }; + + BN.prototype._expand = function _expand (size) { + while (this.length < size) { + this.words[this.length++] = 0; + } + return this; + }; + + // Remove leading `0` from `this` + BN.prototype.strip = function strip () { + while (this.length > 1 && this.words[this.length - 1] === 0) { + this.length--; + } + return this._normSign(); + }; + + BN.prototype._normSign = function _normSign () { + // -0 = 0 + if (this.length === 1 && this.words[0] === 0) { + this.negative = 0; + } + return this; + }; + + BN.prototype.inspect = function inspect () { + return (this.red ? ''; + }; + + /* + + var zeros = []; + var groupSizes = []; + var groupBases = []; + + var s = ''; + var i = -1; + while (++i < BN.wordSize) { + zeros[i] = s; + s += '0'; + } + groupSizes[0] = 0; + groupSizes[1] = 0; + groupBases[0] = 0; + groupBases[1] = 0; + var base = 2 - 1; + while (++base < 36 + 1) { + var groupSize = 0; + var groupBase = 1; + while (groupBase < (1 << BN.wordSize) / base) { + groupBase *= base; + groupSize += 1; + } + groupSizes[base] = groupSize; + groupBases[base] = groupBase; + } + + */ + + var zeros = [ + '', + '0', + '00', + '000', + '0000', + '00000', + '000000', + '0000000', + '00000000', + '000000000', + '0000000000', + '00000000000', + '000000000000', + '0000000000000', + '00000000000000', + '000000000000000', + '0000000000000000', + '00000000000000000', + '000000000000000000', + '0000000000000000000', + '00000000000000000000', + '000000000000000000000', + '0000000000000000000000', + '00000000000000000000000', + '000000000000000000000000', + '0000000000000000000000000' + ]; + + var groupSizes = [ + 0, 0, + 25, 16, 12, 11, 10, 9, 8, + 8, 7, 7, 7, 7, 6, 6, + 6, 6, 6, 6, 6, 5, 5, + 5, 5, 5, 5, 5, 5, 5, + 5, 5, 5, 5, 5, 5, 5 + ]; + + var groupBases = [ + 0, 0, + 33554432, 43046721, 16777216, 48828125, 60466176, 40353607, 16777216, + 43046721, 10000000, 19487171, 35831808, 62748517, 7529536, 11390625, + 16777216, 24137569, 34012224, 47045881, 64000000, 4084101, 5153632, + 6436343, 7962624, 9765625, 11881376, 14348907, 17210368, 20511149, + 24300000, 28629151, 33554432, 39135393, 45435424, 52521875, 60466176 + ]; + + BN.prototype.toString = function toString (base, padding) { + base = base || 10; + padding = padding | 0 || 1; + + var out; + if (base === 16 || base === 'hex') { + out = ''; + var off = 0; + var carry = 0; + for (var i = 0; i < this.length; i++) { + var w = this.words[i]; + var word = (((w << off) | carry) & 0xffffff).toString(16); + carry = (w >>> (24 - off)) & 0xffffff; + if (carry !== 0 || i !== this.length - 1) { + out = zeros[6 - word.length] + word + out; + } else { + out = word + out; + } + off += 2; + if (off >= 26) { + off -= 26; + i--; + } + } + if (carry !== 0) { + out = carry.toString(16) + out; + } + while (out.length % padding !== 0) { + out = '0' + out; + } + if (this.negative !== 0) { + out = '-' + out; + } + return out; + } + + if (base === (base | 0) && base >= 2 && base <= 36) { + // var groupSize = Math.floor(BN.wordSize * Math.LN2 / Math.log(base)); + var groupSize = groupSizes[base]; + // var groupBase = Math.pow(base, groupSize); + var groupBase = groupBases[base]; + out = ''; + var c = this.clone(); + c.negative = 0; + while (!c.isZero()) { + var r = c.modn(groupBase).toString(base); + c = c.idivn(groupBase); + + if (!c.isZero()) { + out = zeros[groupSize - r.length] + r + out; + } else { + out = r + out; + } + } + if (this.isZero()) { + out = '0' + out; + } + while (out.length % padding !== 0) { + out = '0' + out; + } + if (this.negative !== 0) { + out = '-' + out; + } + return out; + } + + assert(false, 'Base should be between 2 and 36'); + }; + + BN.prototype.toNumber = function toNumber () { + var ret = this.words[0]; + if (this.length === 2) { + ret += this.words[1] * 0x4000000; + } else if (this.length === 3 && this.words[2] === 0x01) { + // NOTE: at this stage it is known that the top bit is set + ret += 0x10000000000000 + (this.words[1] * 0x4000000); + } else if (this.length > 2) { + assert(false, 'Number can only safely store up to 53 bits'); + } + return (this.negative !== 0) ? -ret : ret; + }; + + BN.prototype.toJSON = function toJSON () { + return this.toString(16); + }; + + BN.prototype.toBuffer = function toBuffer (endian, length) { + assert(typeof Buffer !== 'undefined'); + return this.toArrayLike(Buffer, endian, length); + }; + + BN.prototype.toArray = function toArray (endian, length) { + return this.toArrayLike(Array, endian, length); + }; + + BN.prototype.toArrayLike = function toArrayLike (ArrayType, endian, length) { + var byteLength = this.byteLength(); + var reqLength = length || Math.max(1, byteLength); + assert(byteLength <= reqLength, 'byte array longer than desired length'); + assert(reqLength > 0, 'Requested array length <= 0'); + + this.strip(); + var littleEndian = endian === 'le'; + var res = new ArrayType(reqLength); + + var b, i; + var q = this.clone(); + if (!littleEndian) { + // Assume big-endian + for (i = 0; i < reqLength - byteLength; i++) { + res[i] = 0; + } + + for (i = 0; !q.isZero(); i++) { + b = q.andln(0xff); + q.iushrn(8); + + res[reqLength - i - 1] = b; + } + } else { + for (i = 0; !q.isZero(); i++) { + b = q.andln(0xff); + q.iushrn(8); + + res[i] = b; + } + + for (; i < reqLength; i++) { + res[i] = 0; + } + } + + return res; + }; + + if (Math.clz32) { + BN.prototype._countBits = function _countBits (w) { + return 32 - Math.clz32(w); + }; + } else { + BN.prototype._countBits = function _countBits (w) { + var t = w; + var r = 0; + if (t >= 0x1000) { + r += 13; + t >>>= 13; + } + if (t >= 0x40) { + r += 7; + t >>>= 7; + } + if (t >= 0x8) { + r += 4; + t >>>= 4; + } + if (t >= 0x02) { + r += 2; + t >>>= 2; + } + return r + t; + }; + } + + BN.prototype._zeroBits = function _zeroBits (w) { + // Short-cut + if (w === 0) return 26; + + var t = w; + var r = 0; + if ((t & 0x1fff) === 0) { + r += 13; + t >>>= 13; + } + if ((t & 0x7f) === 0) { + r += 7; + t >>>= 7; + } + if ((t & 0xf) === 0) { + r += 4; + t >>>= 4; + } + if ((t & 0x3) === 0) { + r += 2; + t >>>= 2; + } + if ((t & 0x1) === 0) { + r++; + } + return r; + }; + + // Return number of used bits in a BN + BN.prototype.bitLength = function bitLength () { + var w = this.words[this.length - 1]; + var hi = this._countBits(w); + return (this.length - 1) * 26 + hi; + }; + + function toBitArray (num) { + var w = new Array(num.bitLength()); + + for (var bit = 0; bit < w.length; bit++) { + var off = (bit / 26) | 0; + var wbit = bit % 26; + + w[bit] = (num.words[off] & (1 << wbit)) >>> wbit; + } + + return w; + } + + // Number of trailing zero bits + BN.prototype.zeroBits = function zeroBits () { + if (this.isZero()) return 0; + + var r = 0; + for (var i = 0; i < this.length; i++) { + var b = this._zeroBits(this.words[i]); + r += b; + if (b !== 26) break; + } + return r; + }; + + BN.prototype.byteLength = function byteLength () { + return Math.ceil(this.bitLength() / 8); + }; + + BN.prototype.toTwos = function toTwos (width) { + if (this.negative !== 0) { + return this.abs().inotn(width).iaddn(1); + } + return this.clone(); + }; + + BN.prototype.fromTwos = function fromTwos (width) { + if (this.testn(width - 1)) { + return this.notn(width).iaddn(1).ineg(); + } + return this.clone(); + }; + + BN.prototype.isNeg = function isNeg () { + return this.negative !== 0; + }; + + // Return negative clone of `this` + BN.prototype.neg = function neg () { + return this.clone().ineg(); + }; + + BN.prototype.ineg = function ineg () { + if (!this.isZero()) { + this.negative ^= 1; + } + + return this; + }; + + // Or `num` with `this` in-place + BN.prototype.iuor = function iuor (num) { + while (this.length < num.length) { + this.words[this.length++] = 0; + } + + for (var i = 0; i < num.length; i++) { + this.words[i] = this.words[i] | num.words[i]; + } + + return this.strip(); + }; + + BN.prototype.ior = function ior (num) { + assert((this.negative | num.negative) === 0); + return this.iuor(num); + }; + + // Or `num` with `this` + BN.prototype.or = function or (num) { + if (this.length > num.length) return this.clone().ior(num); + return num.clone().ior(this); + }; + + BN.prototype.uor = function uor (num) { + if (this.length > num.length) return this.clone().iuor(num); + return num.clone().iuor(this); + }; + + // And `num` with `this` in-place + BN.prototype.iuand = function iuand (num) { + // b = min-length(num, this) + var b; + if (this.length > num.length) { + b = num; + } else { + b = this; + } + + for (var i = 0; i < b.length; i++) { + this.words[i] = this.words[i] & num.words[i]; + } + + this.length = b.length; + + return this.strip(); + }; + + BN.prototype.iand = function iand (num) { + assert((this.negative | num.negative) === 0); + return this.iuand(num); + }; + + // And `num` with `this` + BN.prototype.and = function and (num) { + if (this.length > num.length) return this.clone().iand(num); + return num.clone().iand(this); + }; + + BN.prototype.uand = function uand (num) { + if (this.length > num.length) return this.clone().iuand(num); + return num.clone().iuand(this); + }; + + // Xor `num` with `this` in-place + BN.prototype.iuxor = function iuxor (num) { + // a.length > b.length + var a; + var b; + if (this.length > num.length) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + for (var i = 0; i < b.length; i++) { + this.words[i] = a.words[i] ^ b.words[i]; + } + + if (this !== a) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + this.length = a.length; + + return this.strip(); + }; + + BN.prototype.ixor = function ixor (num) { + assert((this.negative | num.negative) === 0); + return this.iuxor(num); + }; + + // Xor `num` with `this` + BN.prototype.xor = function xor (num) { + if (this.length > num.length) return this.clone().ixor(num); + return num.clone().ixor(this); + }; + + BN.prototype.uxor = function uxor (num) { + if (this.length > num.length) return this.clone().iuxor(num); + return num.clone().iuxor(this); + }; + + // Not ``this`` with ``width`` bitwidth + BN.prototype.inotn = function inotn (width) { + assert(typeof width === 'number' && width >= 0); + + var bytesNeeded = Math.ceil(width / 26) | 0; + var bitsLeft = width % 26; + + // Extend the buffer with leading zeroes + this._expand(bytesNeeded); + + if (bitsLeft > 0) { + bytesNeeded--; + } + + // Handle complete words + for (var i = 0; i < bytesNeeded; i++) { + this.words[i] = ~this.words[i] & 0x3ffffff; + } + + // Handle the residue + if (bitsLeft > 0) { + this.words[i] = ~this.words[i] & (0x3ffffff >> (26 - bitsLeft)); + } + + // And remove leading zeroes + return this.strip(); + }; + + BN.prototype.notn = function notn (width) { + return this.clone().inotn(width); + }; + + // Set `bit` of `this` + BN.prototype.setn = function setn (bit, val) { + assert(typeof bit === 'number' && bit >= 0); + + var off = (bit / 26) | 0; + var wbit = bit % 26; + + this._expand(off + 1); + + if (val) { + this.words[off] = this.words[off] | (1 << wbit); + } else { + this.words[off] = this.words[off] & ~(1 << wbit); + } + + return this.strip(); + }; + + // Add `num` to `this` in-place + BN.prototype.iadd = function iadd (num) { + var r; + + // negative + positive + if (this.negative !== 0 && num.negative === 0) { + this.negative = 0; + r = this.isub(num); + this.negative ^= 1; + return this._normSign(); + + // positive + negative + } else if (this.negative === 0 && num.negative !== 0) { + num.negative = 0; + r = this.isub(num); + num.negative = 1; + return r._normSign(); + } + + // a.length > b.length + var a, b; + if (this.length > num.length) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + var carry = 0; + for (var i = 0; i < b.length; i++) { + r = (a.words[i] | 0) + (b.words[i] | 0) + carry; + this.words[i] = r & 0x3ffffff; + carry = r >>> 26; + } + for (; carry !== 0 && i < a.length; i++) { + r = (a.words[i] | 0) + carry; + this.words[i] = r & 0x3ffffff; + carry = r >>> 26; + } + + this.length = a.length; + if (carry !== 0) { + this.words[this.length] = carry; + this.length++; + // Copy the rest of the words + } else if (a !== this) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + return this; + }; + + // Add `num` to `this` + BN.prototype.add = function add (num) { + var res; + if (num.negative !== 0 && this.negative === 0) { + num.negative = 0; + res = this.sub(num); + num.negative ^= 1; + return res; + } else if (num.negative === 0 && this.negative !== 0) { + this.negative = 0; + res = num.sub(this); + this.negative = 1; + return res; + } + + if (this.length > num.length) return this.clone().iadd(num); + + return num.clone().iadd(this); + }; + + // Subtract `num` from `this` in-place + BN.prototype.isub = function isub (num) { + // this - (-num) = this + num + if (num.negative !== 0) { + num.negative = 0; + var r = this.iadd(num); + num.negative = 1; + return r._normSign(); + + // -this - num = -(this + num) + } else if (this.negative !== 0) { + this.negative = 0; + this.iadd(num); + this.negative = 1; + return this._normSign(); + } + + // At this point both numbers are positive + var cmp = this.cmp(num); + + // Optimization - zeroify + if (cmp === 0) { + this.negative = 0; + this.length = 1; + this.words[0] = 0; + return this; + } + + // a > b + var a, b; + if (cmp > 0) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + var carry = 0; + for (var i = 0; i < b.length; i++) { + r = (a.words[i] | 0) - (b.words[i] | 0) + carry; + carry = r >> 26; + this.words[i] = r & 0x3ffffff; + } + for (; carry !== 0 && i < a.length; i++) { + r = (a.words[i] | 0) + carry; + carry = r >> 26; + this.words[i] = r & 0x3ffffff; + } + + // Copy rest of the words + if (carry === 0 && i < a.length && a !== this) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + this.length = Math.max(this.length, i); + + if (a !== this) { + this.negative = 1; + } + + return this.strip(); + }; + + // Subtract `num` from `this` + BN.prototype.sub = function sub (num) { + return this.clone().isub(num); + }; + + function smallMulTo (self, num, out) { + out.negative = num.negative ^ self.negative; + var len = (self.length + num.length) | 0; + out.length = len; + len = (len - 1) | 0; + + // Peel one iteration (compiler can't do it, because of code complexity) + var a = self.words[0] | 0; + var b = num.words[0] | 0; + var r = a * b; + + var lo = r & 0x3ffffff; + var carry = (r / 0x4000000) | 0; + out.words[0] = lo; + + for (var k = 1; k < len; k++) { + // Sum all words with the same `i + j = k` and accumulate `ncarry`, + // note that ncarry could be >= 0x3ffffff + var ncarry = carry >>> 26; + var rword = carry & 0x3ffffff; + var maxJ = Math.min(k, num.length - 1); + for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { + var i = (k - j) | 0; + a = self.words[i] | 0; + b = num.words[j] | 0; + r = a * b + rword; + ncarry += (r / 0x4000000) | 0; + rword = r & 0x3ffffff; + } + out.words[k] = rword | 0; + carry = ncarry | 0; + } + if (carry !== 0) { + out.words[k] = carry | 0; + } else { + out.length--; + } + + return out.strip(); + } + + // TODO(indutny): it may be reasonable to omit it for users who don't need + // to work with 256-bit numbers, otherwise it gives 20% improvement for 256-bit + // multiplication (like elliptic secp256k1). + var comb10MulTo = function comb10MulTo (self, num, out) { + var a = self.words; + var b = num.words; + var o = out.words; + var c = 0; + var lo; + var mid; + var hi; + var a0 = a[0] | 0; + var al0 = a0 & 0x1fff; + var ah0 = a0 >>> 13; + var a1 = a[1] | 0; + var al1 = a1 & 0x1fff; + var ah1 = a1 >>> 13; + var a2 = a[2] | 0; + var al2 = a2 & 0x1fff; + var ah2 = a2 >>> 13; + var a3 = a[3] | 0; + var al3 = a3 & 0x1fff; + var ah3 = a3 >>> 13; + var a4 = a[4] | 0; + var al4 = a4 & 0x1fff; + var ah4 = a4 >>> 13; + var a5 = a[5] | 0; + var al5 = a5 & 0x1fff; + var ah5 = a5 >>> 13; + var a6 = a[6] | 0; + var al6 = a6 & 0x1fff; + var ah6 = a6 >>> 13; + var a7 = a[7] | 0; + var al7 = a7 & 0x1fff; + var ah7 = a7 >>> 13; + var a8 = a[8] | 0; + var al8 = a8 & 0x1fff; + var ah8 = a8 >>> 13; + var a9 = a[9] | 0; + var al9 = a9 & 0x1fff; + var ah9 = a9 >>> 13; + var b0 = b[0] | 0; + var bl0 = b0 & 0x1fff; + var bh0 = b0 >>> 13; + var b1 = b[1] | 0; + var bl1 = b1 & 0x1fff; + var bh1 = b1 >>> 13; + var b2 = b[2] | 0; + var bl2 = b2 & 0x1fff; + var bh2 = b2 >>> 13; + var b3 = b[3] | 0; + var bl3 = b3 & 0x1fff; + var bh3 = b3 >>> 13; + var b4 = b[4] | 0; + var bl4 = b4 & 0x1fff; + var bh4 = b4 >>> 13; + var b5 = b[5] | 0; + var bl5 = b5 & 0x1fff; + var bh5 = b5 >>> 13; + var b6 = b[6] | 0; + var bl6 = b6 & 0x1fff; + var bh6 = b6 >>> 13; + var b7 = b[7] | 0; + var bl7 = b7 & 0x1fff; + var bh7 = b7 >>> 13; + var b8 = b[8] | 0; + var bl8 = b8 & 0x1fff; + var bh8 = b8 >>> 13; + var b9 = b[9] | 0; + var bl9 = b9 & 0x1fff; + var bh9 = b9 >>> 13; + + out.negative = self.negative ^ num.negative; + out.length = 19; + /* k = 0 */ + lo = Math.imul(al0, bl0); + mid = Math.imul(al0, bh0); + mid = (mid + Math.imul(ah0, bl0)) | 0; + hi = Math.imul(ah0, bh0); + var w0 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w0 >>> 26)) | 0; + w0 &= 0x3ffffff; + /* k = 1 */ + lo = Math.imul(al1, bl0); + mid = Math.imul(al1, bh0); + mid = (mid + Math.imul(ah1, bl0)) | 0; + hi = Math.imul(ah1, bh0); + lo = (lo + Math.imul(al0, bl1)) | 0; + mid = (mid + Math.imul(al0, bh1)) | 0; + mid = (mid + Math.imul(ah0, bl1)) | 0; + hi = (hi + Math.imul(ah0, bh1)) | 0; + var w1 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w1 >>> 26)) | 0; + w1 &= 0x3ffffff; + /* k = 2 */ + lo = Math.imul(al2, bl0); + mid = Math.imul(al2, bh0); + mid = (mid + Math.imul(ah2, bl0)) | 0; + hi = Math.imul(ah2, bh0); + lo = (lo + Math.imul(al1, bl1)) | 0; + mid = (mid + Math.imul(al1, bh1)) | 0; + mid = (mid + Math.imul(ah1, bl1)) | 0; + hi = (hi + Math.imul(ah1, bh1)) | 0; + lo = (lo + Math.imul(al0, bl2)) | 0; + mid = (mid + Math.imul(al0, bh2)) | 0; + mid = (mid + Math.imul(ah0, bl2)) | 0; + hi = (hi + Math.imul(ah0, bh2)) | 0; + var w2 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w2 >>> 26)) | 0; + w2 &= 0x3ffffff; + /* k = 3 */ + lo = Math.imul(al3, bl0); + mid = Math.imul(al3, bh0); + mid = (mid + Math.imul(ah3, bl0)) | 0; + hi = Math.imul(ah3, bh0); + lo = (lo + Math.imul(al2, bl1)) | 0; + mid = (mid + Math.imul(al2, bh1)) | 0; + mid = (mid + Math.imul(ah2, bl1)) | 0; + hi = (hi + Math.imul(ah2, bh1)) | 0; + lo = (lo + Math.imul(al1, bl2)) | 0; + mid = (mid + Math.imul(al1, bh2)) | 0; + mid = (mid + Math.imul(ah1, bl2)) | 0; + hi = (hi + Math.imul(ah1, bh2)) | 0; + lo = (lo + Math.imul(al0, bl3)) | 0; + mid = (mid + Math.imul(al0, bh3)) | 0; + mid = (mid + Math.imul(ah0, bl3)) | 0; + hi = (hi + Math.imul(ah0, bh3)) | 0; + var w3 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w3 >>> 26)) | 0; + w3 &= 0x3ffffff; + /* k = 4 */ + lo = Math.imul(al4, bl0); + mid = Math.imul(al4, bh0); + mid = (mid + Math.imul(ah4, bl0)) | 0; + hi = Math.imul(ah4, bh0); + lo = (lo + Math.imul(al3, bl1)) | 0; + mid = (mid + Math.imul(al3, bh1)) | 0; + mid = (mid + Math.imul(ah3, bl1)) | 0; + hi = (hi + Math.imul(ah3, bh1)) | 0; + lo = (lo + Math.imul(al2, bl2)) | 0; + mid = (mid + Math.imul(al2, bh2)) | 0; + mid = (mid + Math.imul(ah2, bl2)) | 0; + hi = (hi + Math.imul(ah2, bh2)) | 0; + lo = (lo + Math.imul(al1, bl3)) | 0; + mid = (mid + Math.imul(al1, bh3)) | 0; + mid = (mid + Math.imul(ah1, bl3)) | 0; + hi = (hi + Math.imul(ah1, bh3)) | 0; + lo = (lo + Math.imul(al0, bl4)) | 0; + mid = (mid + Math.imul(al0, bh4)) | 0; + mid = (mid + Math.imul(ah0, bl4)) | 0; + hi = (hi + Math.imul(ah0, bh4)) | 0; + var w4 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w4 >>> 26)) | 0; + w4 &= 0x3ffffff; + /* k = 5 */ + lo = Math.imul(al5, bl0); + mid = Math.imul(al5, bh0); + mid = (mid + Math.imul(ah5, bl0)) | 0; + hi = Math.imul(ah5, bh0); + lo = (lo + Math.imul(al4, bl1)) | 0; + mid = (mid + Math.imul(al4, bh1)) | 0; + mid = (mid + Math.imul(ah4, bl1)) | 0; + hi = (hi + Math.imul(ah4, bh1)) | 0; + lo = (lo + Math.imul(al3, bl2)) | 0; + mid = (mid + Math.imul(al3, bh2)) | 0; + mid = (mid + Math.imul(ah3, bl2)) | 0; + hi = (hi + Math.imul(ah3, bh2)) | 0; + lo = (lo + Math.imul(al2, bl3)) | 0; + mid = (mid + Math.imul(al2, bh3)) | 0; + mid = (mid + Math.imul(ah2, bl3)) | 0; + hi = (hi + Math.imul(ah2, bh3)) | 0; + lo = (lo + Math.imul(al1, bl4)) | 0; + mid = (mid + Math.imul(al1, bh4)) | 0; + mid = (mid + Math.imul(ah1, bl4)) | 0; + hi = (hi + Math.imul(ah1, bh4)) | 0; + lo = (lo + Math.imul(al0, bl5)) | 0; + mid = (mid + Math.imul(al0, bh5)) | 0; + mid = (mid + Math.imul(ah0, bl5)) | 0; + hi = (hi + Math.imul(ah0, bh5)) | 0; + var w5 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w5 >>> 26)) | 0; + w5 &= 0x3ffffff; + /* k = 6 */ + lo = Math.imul(al6, bl0); + mid = Math.imul(al6, bh0); + mid = (mid + Math.imul(ah6, bl0)) | 0; + hi = Math.imul(ah6, bh0); + lo = (lo + Math.imul(al5, bl1)) | 0; + mid = (mid + Math.imul(al5, bh1)) | 0; + mid = (mid + Math.imul(ah5, bl1)) | 0; + hi = (hi + Math.imul(ah5, bh1)) | 0; + lo = (lo + Math.imul(al4, bl2)) | 0; + mid = (mid + Math.imul(al4, bh2)) | 0; + mid = (mid + Math.imul(ah4, bl2)) | 0; + hi = (hi + Math.imul(ah4, bh2)) | 0; + lo = (lo + Math.imul(al3, bl3)) | 0; + mid = (mid + Math.imul(al3, bh3)) | 0; + mid = (mid + Math.imul(ah3, bl3)) | 0; + hi = (hi + Math.imul(ah3, bh3)) | 0; + lo = (lo + Math.imul(al2, bl4)) | 0; + mid = (mid + Math.imul(al2, bh4)) | 0; + mid = (mid + Math.imul(ah2, bl4)) | 0; + hi = (hi + Math.imul(ah2, bh4)) | 0; + lo = (lo + Math.imul(al1, bl5)) | 0; + mid = (mid + Math.imul(al1, bh5)) | 0; + mid = (mid + Math.imul(ah1, bl5)) | 0; + hi = (hi + Math.imul(ah1, bh5)) | 0; + lo = (lo + Math.imul(al0, bl6)) | 0; + mid = (mid + Math.imul(al0, bh6)) | 0; + mid = (mid + Math.imul(ah0, bl6)) | 0; + hi = (hi + Math.imul(ah0, bh6)) | 0; + var w6 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w6 >>> 26)) | 0; + w6 &= 0x3ffffff; + /* k = 7 */ + lo = Math.imul(al7, bl0); + mid = Math.imul(al7, bh0); + mid = (mid + Math.imul(ah7, bl0)) | 0; + hi = Math.imul(ah7, bh0); + lo = (lo + Math.imul(al6, bl1)) | 0; + mid = (mid + Math.imul(al6, bh1)) | 0; + mid = (mid + Math.imul(ah6, bl1)) | 0; + hi = (hi + Math.imul(ah6, bh1)) | 0; + lo = (lo + Math.imul(al5, bl2)) | 0; + mid = (mid + Math.imul(al5, bh2)) | 0; + mid = (mid + Math.imul(ah5, bl2)) | 0; + hi = (hi + Math.imul(ah5, bh2)) | 0; + lo = (lo + Math.imul(al4, bl3)) | 0; + mid = (mid + Math.imul(al4, bh3)) | 0; + mid = (mid + Math.imul(ah4, bl3)) | 0; + hi = (hi + Math.imul(ah4, bh3)) | 0; + lo = (lo + Math.imul(al3, bl4)) | 0; + mid = (mid + Math.imul(al3, bh4)) | 0; + mid = (mid + Math.imul(ah3, bl4)) | 0; + hi = (hi + Math.imul(ah3, bh4)) | 0; + lo = (lo + Math.imul(al2, bl5)) | 0; + mid = (mid + Math.imul(al2, bh5)) | 0; + mid = (mid + Math.imul(ah2, bl5)) | 0; + hi = (hi + Math.imul(ah2, bh5)) | 0; + lo = (lo + Math.imul(al1, bl6)) | 0; + mid = (mid + Math.imul(al1, bh6)) | 0; + mid = (mid + Math.imul(ah1, bl6)) | 0; + hi = (hi + Math.imul(ah1, bh6)) | 0; + lo = (lo + Math.imul(al0, bl7)) | 0; + mid = (mid + Math.imul(al0, bh7)) | 0; + mid = (mid + Math.imul(ah0, bl7)) | 0; + hi = (hi + Math.imul(ah0, bh7)) | 0; + var w7 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w7 >>> 26)) | 0; + w7 &= 0x3ffffff; + /* k = 8 */ + lo = Math.imul(al8, bl0); + mid = Math.imul(al8, bh0); + mid = (mid + Math.imul(ah8, bl0)) | 0; + hi = Math.imul(ah8, bh0); + lo = (lo + Math.imul(al7, bl1)) | 0; + mid = (mid + Math.imul(al7, bh1)) | 0; + mid = (mid + Math.imul(ah7, bl1)) | 0; + hi = (hi + Math.imul(ah7, bh1)) | 0; + lo = (lo + Math.imul(al6, bl2)) | 0; + mid = (mid + Math.imul(al6, bh2)) | 0; + mid = (mid + Math.imul(ah6, bl2)) | 0; + hi = (hi + Math.imul(ah6, bh2)) | 0; + lo = (lo + Math.imul(al5, bl3)) | 0; + mid = (mid + Math.imul(al5, bh3)) | 0; + mid = (mid + Math.imul(ah5, bl3)) | 0; + hi = (hi + Math.imul(ah5, bh3)) | 0; + lo = (lo + Math.imul(al4, bl4)) | 0; + mid = (mid + Math.imul(al4, bh4)) | 0; + mid = (mid + Math.imul(ah4, bl4)) | 0; + hi = (hi + Math.imul(ah4, bh4)) | 0; + lo = (lo + Math.imul(al3, bl5)) | 0; + mid = (mid + Math.imul(al3, bh5)) | 0; + mid = (mid + Math.imul(ah3, bl5)) | 0; + hi = (hi + Math.imul(ah3, bh5)) | 0; + lo = (lo + Math.imul(al2, bl6)) | 0; + mid = (mid + Math.imul(al2, bh6)) | 0; + mid = (mid + Math.imul(ah2, bl6)) | 0; + hi = (hi + Math.imul(ah2, bh6)) | 0; + lo = (lo + Math.imul(al1, bl7)) | 0; + mid = (mid + Math.imul(al1, bh7)) | 0; + mid = (mid + Math.imul(ah1, bl7)) | 0; + hi = (hi + Math.imul(ah1, bh7)) | 0; + lo = (lo + Math.imul(al0, bl8)) | 0; + mid = (mid + Math.imul(al0, bh8)) | 0; + mid = (mid + Math.imul(ah0, bl8)) | 0; + hi = (hi + Math.imul(ah0, bh8)) | 0; + var w8 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w8 >>> 26)) | 0; + w8 &= 0x3ffffff; + /* k = 9 */ + lo = Math.imul(al9, bl0); + mid = Math.imul(al9, bh0); + mid = (mid + Math.imul(ah9, bl0)) | 0; + hi = Math.imul(ah9, bh0); + lo = (lo + Math.imul(al8, bl1)) | 0; + mid = (mid + Math.imul(al8, bh1)) | 0; + mid = (mid + Math.imul(ah8, bl1)) | 0; + hi = (hi + Math.imul(ah8, bh1)) | 0; + lo = (lo + Math.imul(al7, bl2)) | 0; + mid = (mid + Math.imul(al7, bh2)) | 0; + mid = (mid + Math.imul(ah7, bl2)) | 0; + hi = (hi + Math.imul(ah7, bh2)) | 0; + lo = (lo + Math.imul(al6, bl3)) | 0; + mid = (mid + Math.imul(al6, bh3)) | 0; + mid = (mid + Math.imul(ah6, bl3)) | 0; + hi = (hi + Math.imul(ah6, bh3)) | 0; + lo = (lo + Math.imul(al5, bl4)) | 0; + mid = (mid + Math.imul(al5, bh4)) | 0; + mid = (mid + Math.imul(ah5, bl4)) | 0; + hi = (hi + Math.imul(ah5, bh4)) | 0; + lo = (lo + Math.imul(al4, bl5)) | 0; + mid = (mid + Math.imul(al4, bh5)) | 0; + mid = (mid + Math.imul(ah4, bl5)) | 0; + hi = (hi + Math.imul(ah4, bh5)) | 0; + lo = (lo + Math.imul(al3, bl6)) | 0; + mid = (mid + Math.imul(al3, bh6)) | 0; + mid = (mid + Math.imul(ah3, bl6)) | 0; + hi = (hi + Math.imul(ah3, bh6)) | 0; + lo = (lo + Math.imul(al2, bl7)) | 0; + mid = (mid + Math.imul(al2, bh7)) | 0; + mid = (mid + Math.imul(ah2, bl7)) | 0; + hi = (hi + Math.imul(ah2, bh7)) | 0; + lo = (lo + Math.imul(al1, bl8)) | 0; + mid = (mid + Math.imul(al1, bh8)) | 0; + mid = (mid + Math.imul(ah1, bl8)) | 0; + hi = (hi + Math.imul(ah1, bh8)) | 0; + lo = (lo + Math.imul(al0, bl9)) | 0; + mid = (mid + Math.imul(al0, bh9)) | 0; + mid = (mid + Math.imul(ah0, bl9)) | 0; + hi = (hi + Math.imul(ah0, bh9)) | 0; + var w9 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w9 >>> 26)) | 0; + w9 &= 0x3ffffff; + /* k = 10 */ + lo = Math.imul(al9, bl1); + mid = Math.imul(al9, bh1); + mid = (mid + Math.imul(ah9, bl1)) | 0; + hi = Math.imul(ah9, bh1); + lo = (lo + Math.imul(al8, bl2)) | 0; + mid = (mid + Math.imul(al8, bh2)) | 0; + mid = (mid + Math.imul(ah8, bl2)) | 0; + hi = (hi + Math.imul(ah8, bh2)) | 0; + lo = (lo + Math.imul(al7, bl3)) | 0; + mid = (mid + Math.imul(al7, bh3)) | 0; + mid = (mid + Math.imul(ah7, bl3)) | 0; + hi = (hi + Math.imul(ah7, bh3)) | 0; + lo = (lo + Math.imul(al6, bl4)) | 0; + mid = (mid + Math.imul(al6, bh4)) | 0; + mid = (mid + Math.imul(ah6, bl4)) | 0; + hi = (hi + Math.imul(ah6, bh4)) | 0; + lo = (lo + Math.imul(al5, bl5)) | 0; + mid = (mid + Math.imul(al5, bh5)) | 0; + mid = (mid + Math.imul(ah5, bl5)) | 0; + hi = (hi + Math.imul(ah5, bh5)) | 0; + lo = (lo + Math.imul(al4, bl6)) | 0; + mid = (mid + Math.imul(al4, bh6)) | 0; + mid = (mid + Math.imul(ah4, bl6)) | 0; + hi = (hi + Math.imul(ah4, bh6)) | 0; + lo = (lo + Math.imul(al3, bl7)) | 0; + mid = (mid + Math.imul(al3, bh7)) | 0; + mid = (mid + Math.imul(ah3, bl7)) | 0; + hi = (hi + Math.imul(ah3, bh7)) | 0; + lo = (lo + Math.imul(al2, bl8)) | 0; + mid = (mid + Math.imul(al2, bh8)) | 0; + mid = (mid + Math.imul(ah2, bl8)) | 0; + hi = (hi + Math.imul(ah2, bh8)) | 0; + lo = (lo + Math.imul(al1, bl9)) | 0; + mid = (mid + Math.imul(al1, bh9)) | 0; + mid = (mid + Math.imul(ah1, bl9)) | 0; + hi = (hi + Math.imul(ah1, bh9)) | 0; + var w10 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w10 >>> 26)) | 0; + w10 &= 0x3ffffff; + /* k = 11 */ + lo = Math.imul(al9, bl2); + mid = Math.imul(al9, bh2); + mid = (mid + Math.imul(ah9, bl2)) | 0; + hi = Math.imul(ah9, bh2); + lo = (lo + Math.imul(al8, bl3)) | 0; + mid = (mid + Math.imul(al8, bh3)) | 0; + mid = (mid + Math.imul(ah8, bl3)) | 0; + hi = (hi + Math.imul(ah8, bh3)) | 0; + lo = (lo + Math.imul(al7, bl4)) | 0; + mid = (mid + Math.imul(al7, bh4)) | 0; + mid = (mid + Math.imul(ah7, bl4)) | 0; + hi = (hi + Math.imul(ah7, bh4)) | 0; + lo = (lo + Math.imul(al6, bl5)) | 0; + mid = (mid + Math.imul(al6, bh5)) | 0; + mid = (mid + Math.imul(ah6, bl5)) | 0; + hi = (hi + Math.imul(ah6, bh5)) | 0; + lo = (lo + Math.imul(al5, bl6)) | 0; + mid = (mid + Math.imul(al5, bh6)) | 0; + mid = (mid + Math.imul(ah5, bl6)) | 0; + hi = (hi + Math.imul(ah5, bh6)) | 0; + lo = (lo + Math.imul(al4, bl7)) | 0; + mid = (mid + Math.imul(al4, bh7)) | 0; + mid = (mid + Math.imul(ah4, bl7)) | 0; + hi = (hi + Math.imul(ah4, bh7)) | 0; + lo = (lo + Math.imul(al3, bl8)) | 0; + mid = (mid + Math.imul(al3, bh8)) | 0; + mid = (mid + Math.imul(ah3, bl8)) | 0; + hi = (hi + Math.imul(ah3, bh8)) | 0; + lo = (lo + Math.imul(al2, bl9)) | 0; + mid = (mid + Math.imul(al2, bh9)) | 0; + mid = (mid + Math.imul(ah2, bl9)) | 0; + hi = (hi + Math.imul(ah2, bh9)) | 0; + var w11 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w11 >>> 26)) | 0; + w11 &= 0x3ffffff; + /* k = 12 */ + lo = Math.imul(al9, bl3); + mid = Math.imul(al9, bh3); + mid = (mid + Math.imul(ah9, bl3)) | 0; + hi = Math.imul(ah9, bh3); + lo = (lo + Math.imul(al8, bl4)) | 0; + mid = (mid + Math.imul(al8, bh4)) | 0; + mid = (mid + Math.imul(ah8, bl4)) | 0; + hi = (hi + Math.imul(ah8, bh4)) | 0; + lo = (lo + Math.imul(al7, bl5)) | 0; + mid = (mid + Math.imul(al7, bh5)) | 0; + mid = (mid + Math.imul(ah7, bl5)) | 0; + hi = (hi + Math.imul(ah7, bh5)) | 0; + lo = (lo + Math.imul(al6, bl6)) | 0; + mid = (mid + Math.imul(al6, bh6)) | 0; + mid = (mid + Math.imul(ah6, bl6)) | 0; + hi = (hi + Math.imul(ah6, bh6)) | 0; + lo = (lo + Math.imul(al5, bl7)) | 0; + mid = (mid + Math.imul(al5, bh7)) | 0; + mid = (mid + Math.imul(ah5, bl7)) | 0; + hi = (hi + Math.imul(ah5, bh7)) | 0; + lo = (lo + Math.imul(al4, bl8)) | 0; + mid = (mid + Math.imul(al4, bh8)) | 0; + mid = (mid + Math.imul(ah4, bl8)) | 0; + hi = (hi + Math.imul(ah4, bh8)) | 0; + lo = (lo + Math.imul(al3, bl9)) | 0; + mid = (mid + Math.imul(al3, bh9)) | 0; + mid = (mid + Math.imul(ah3, bl9)) | 0; + hi = (hi + Math.imul(ah3, bh9)) | 0; + var w12 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w12 >>> 26)) | 0; + w12 &= 0x3ffffff; + /* k = 13 */ + lo = Math.imul(al9, bl4); + mid = Math.imul(al9, bh4); + mid = (mid + Math.imul(ah9, bl4)) | 0; + hi = Math.imul(ah9, bh4); + lo = (lo + Math.imul(al8, bl5)) | 0; + mid = (mid + Math.imul(al8, bh5)) | 0; + mid = (mid + Math.imul(ah8, bl5)) | 0; + hi = (hi + Math.imul(ah8, bh5)) | 0; + lo = (lo + Math.imul(al7, bl6)) | 0; + mid = (mid + Math.imul(al7, bh6)) | 0; + mid = (mid + Math.imul(ah7, bl6)) | 0; + hi = (hi + Math.imul(ah7, bh6)) | 0; + lo = (lo + Math.imul(al6, bl7)) | 0; + mid = (mid + Math.imul(al6, bh7)) | 0; + mid = (mid + Math.imul(ah6, bl7)) | 0; + hi = (hi + Math.imul(ah6, bh7)) | 0; + lo = (lo + Math.imul(al5, bl8)) | 0; + mid = (mid + Math.imul(al5, bh8)) | 0; + mid = (mid + Math.imul(ah5, bl8)) | 0; + hi = (hi + Math.imul(ah5, bh8)) | 0; + lo = (lo + Math.imul(al4, bl9)) | 0; + mid = (mid + Math.imul(al4, bh9)) | 0; + mid = (mid + Math.imul(ah4, bl9)) | 0; + hi = (hi + Math.imul(ah4, bh9)) | 0; + var w13 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w13 >>> 26)) | 0; + w13 &= 0x3ffffff; + /* k = 14 */ + lo = Math.imul(al9, bl5); + mid = Math.imul(al9, bh5); + mid = (mid + Math.imul(ah9, bl5)) | 0; + hi = Math.imul(ah9, bh5); + lo = (lo + Math.imul(al8, bl6)) | 0; + mid = (mid + Math.imul(al8, bh6)) | 0; + mid = (mid + Math.imul(ah8, bl6)) | 0; + hi = (hi + Math.imul(ah8, bh6)) | 0; + lo = (lo + Math.imul(al7, bl7)) | 0; + mid = (mid + Math.imul(al7, bh7)) | 0; + mid = (mid + Math.imul(ah7, bl7)) | 0; + hi = (hi + Math.imul(ah7, bh7)) | 0; + lo = (lo + Math.imul(al6, bl8)) | 0; + mid = (mid + Math.imul(al6, bh8)) | 0; + mid = (mid + Math.imul(ah6, bl8)) | 0; + hi = (hi + Math.imul(ah6, bh8)) | 0; + lo = (lo + Math.imul(al5, bl9)) | 0; + mid = (mid + Math.imul(al5, bh9)) | 0; + mid = (mid + Math.imul(ah5, bl9)) | 0; + hi = (hi + Math.imul(ah5, bh9)) | 0; + var w14 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w14 >>> 26)) | 0; + w14 &= 0x3ffffff; + /* k = 15 */ + lo = Math.imul(al9, bl6); + mid = Math.imul(al9, bh6); + mid = (mid + Math.imul(ah9, bl6)) | 0; + hi = Math.imul(ah9, bh6); + lo = (lo + Math.imul(al8, bl7)) | 0; + mid = (mid + Math.imul(al8, bh7)) | 0; + mid = (mid + Math.imul(ah8, bl7)) | 0; + hi = (hi + Math.imul(ah8, bh7)) | 0; + lo = (lo + Math.imul(al7, bl8)) | 0; + mid = (mid + Math.imul(al7, bh8)) | 0; + mid = (mid + Math.imul(ah7, bl8)) | 0; + hi = (hi + Math.imul(ah7, bh8)) | 0; + lo = (lo + Math.imul(al6, bl9)) | 0; + mid = (mid + Math.imul(al6, bh9)) | 0; + mid = (mid + Math.imul(ah6, bl9)) | 0; + hi = (hi + Math.imul(ah6, bh9)) | 0; + var w15 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w15 >>> 26)) | 0; + w15 &= 0x3ffffff; + /* k = 16 */ + lo = Math.imul(al9, bl7); + mid = Math.imul(al9, bh7); + mid = (mid + Math.imul(ah9, bl7)) | 0; + hi = Math.imul(ah9, bh7); + lo = (lo + Math.imul(al8, bl8)) | 0; + mid = (mid + Math.imul(al8, bh8)) | 0; + mid = (mid + Math.imul(ah8, bl8)) | 0; + hi = (hi + Math.imul(ah8, bh8)) | 0; + lo = (lo + Math.imul(al7, bl9)) | 0; + mid = (mid + Math.imul(al7, bh9)) | 0; + mid = (mid + Math.imul(ah7, bl9)) | 0; + hi = (hi + Math.imul(ah7, bh9)) | 0; + var w16 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w16 >>> 26)) | 0; + w16 &= 0x3ffffff; + /* k = 17 */ + lo = Math.imul(al9, bl8); + mid = Math.imul(al9, bh8); + mid = (mid + Math.imul(ah9, bl8)) | 0; + hi = Math.imul(ah9, bh8); + lo = (lo + Math.imul(al8, bl9)) | 0; + mid = (mid + Math.imul(al8, bh9)) | 0; + mid = (mid + Math.imul(ah8, bl9)) | 0; + hi = (hi + Math.imul(ah8, bh9)) | 0; + var w17 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w17 >>> 26)) | 0; + w17 &= 0x3ffffff; + /* k = 18 */ + lo = Math.imul(al9, bl9); + mid = Math.imul(al9, bh9); + mid = (mid + Math.imul(ah9, bl9)) | 0; + hi = Math.imul(ah9, bh9); + var w18 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w18 >>> 26)) | 0; + w18 &= 0x3ffffff; + o[0] = w0; + o[1] = w1; + o[2] = w2; + o[3] = w3; + o[4] = w4; + o[5] = w5; + o[6] = w6; + o[7] = w7; + o[8] = w8; + o[9] = w9; + o[10] = w10; + o[11] = w11; + o[12] = w12; + o[13] = w13; + o[14] = w14; + o[15] = w15; + o[16] = w16; + o[17] = w17; + o[18] = w18; + if (c !== 0) { + o[19] = c; + out.length++; + } + return out; + }; + + // Polyfill comb + if (!Math.imul) { + comb10MulTo = smallMulTo; + } + + function bigMulTo (self, num, out) { + out.negative = num.negative ^ self.negative; + out.length = self.length + num.length; + + var carry = 0; + var hncarry = 0; + for (var k = 0; k < out.length - 1; k++) { + // Sum all words with the same `i + j = k` and accumulate `ncarry`, + // note that ncarry could be >= 0x3ffffff + var ncarry = hncarry; + hncarry = 0; + var rword = carry & 0x3ffffff; + var maxJ = Math.min(k, num.length - 1); + for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { + var i = k - j; + var a = self.words[i] | 0; + var b = num.words[j] | 0; + var r = a * b; + + var lo = r & 0x3ffffff; + ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0; + lo = (lo + rword) | 0; + rword = lo & 0x3ffffff; + ncarry = (ncarry + (lo >>> 26)) | 0; + + hncarry += ncarry >>> 26; + ncarry &= 0x3ffffff; + } + out.words[k] = rword; + carry = ncarry; + ncarry = hncarry; + } + if (carry !== 0) { + out.words[k] = carry; + } else { + out.length--; + } + + return out.strip(); + } + + function jumboMulTo (self, num, out) { + var fftm = new FFTM(); + return fftm.mulp(self, num, out); + } + + BN.prototype.mulTo = function mulTo (num, out) { + var res; + var len = this.length + num.length; + if (this.length === 10 && num.length === 10) { + res = comb10MulTo(this, num, out); + } else if (len < 63) { + res = smallMulTo(this, num, out); + } else if (len < 1024) { + res = bigMulTo(this, num, out); + } else { + res = jumboMulTo(this, num, out); + } + + return res; + }; + + // Cooley-Tukey algorithm for FFT + // slightly revisited to rely on looping instead of recursion + + function FFTM (x, y) { + this.x = x; + this.y = y; + } + + FFTM.prototype.makeRBT = function makeRBT (N) { + var t = new Array(N); + var l = BN.prototype._countBits(N) - 1; + for (var i = 0; i < N; i++) { + t[i] = this.revBin(i, l, N); + } + + return t; + }; + + // Returns binary-reversed representation of `x` + FFTM.prototype.revBin = function revBin (x, l, N) { + if (x === 0 || x === N - 1) return x; + + var rb = 0; + for (var i = 0; i < l; i++) { + rb |= (x & 1) << (l - i - 1); + x >>= 1; + } + + return rb; + }; + + // Performs "tweedling" phase, therefore 'emulating' + // behaviour of the recursive algorithm + FFTM.prototype.permute = function permute (rbt, rws, iws, rtws, itws, N) { + for (var i = 0; i < N; i++) { + rtws[i] = rws[rbt[i]]; + itws[i] = iws[rbt[i]]; + } + }; + + FFTM.prototype.transform = function transform (rws, iws, rtws, itws, N, rbt) { + this.permute(rbt, rws, iws, rtws, itws, N); + + for (var s = 1; s < N; s <<= 1) { + var l = s << 1; + + var rtwdf = Math.cos(2 * Math.PI / l); + var itwdf = Math.sin(2 * Math.PI / l); + + for (var p = 0; p < N; p += l) { + var rtwdf_ = rtwdf; + var itwdf_ = itwdf; + + for (var j = 0; j < s; j++) { + var re = rtws[p + j]; + var ie = itws[p + j]; + + var ro = rtws[p + j + s]; + var io = itws[p + j + s]; + + var rx = rtwdf_ * ro - itwdf_ * io; + + io = rtwdf_ * io + itwdf_ * ro; + ro = rx; + + rtws[p + j] = re + ro; + itws[p + j] = ie + io; + + rtws[p + j + s] = re - ro; + itws[p + j + s] = ie - io; + + /* jshint maxdepth : false */ + if (j !== l) { + rx = rtwdf * rtwdf_ - itwdf * itwdf_; + + itwdf_ = rtwdf * itwdf_ + itwdf * rtwdf_; + rtwdf_ = rx; + } + } + } + } + }; + + FFTM.prototype.guessLen13b = function guessLen13b (n, m) { + var N = Math.max(m, n) | 1; + var odd = N & 1; + var i = 0; + for (N = N / 2 | 0; N; N = N >>> 1) { + i++; + } + + return 1 << i + 1 + odd; + }; + + FFTM.prototype.conjugate = function conjugate (rws, iws, N) { + if (N <= 1) return; + + for (var i = 0; i < N / 2; i++) { + var t = rws[i]; + + rws[i] = rws[N - i - 1]; + rws[N - i - 1] = t; + + t = iws[i]; + + iws[i] = -iws[N - i - 1]; + iws[N - i - 1] = -t; + } + }; + + FFTM.prototype.normalize13b = function normalize13b (ws, N) { + var carry = 0; + for (var i = 0; i < N / 2; i++) { + var w = Math.round(ws[2 * i + 1] / N) * 0x2000 + + Math.round(ws[2 * i] / N) + + carry; + + ws[i] = w & 0x3ffffff; + + if (w < 0x4000000) { + carry = 0; + } else { + carry = w / 0x4000000 | 0; + } + } + + return ws; + }; + + FFTM.prototype.convert13b = function convert13b (ws, len, rws, N) { + var carry = 0; + for (var i = 0; i < len; i++) { + carry = carry + (ws[i] | 0); + + rws[2 * i] = carry & 0x1fff; carry = carry >>> 13; + rws[2 * i + 1] = carry & 0x1fff; carry = carry >>> 13; + } + + // Pad with zeroes + for (i = 2 * len; i < N; ++i) { + rws[i] = 0; + } + + assert(carry === 0); + assert((carry & ~0x1fff) === 0); + }; + + FFTM.prototype.stub = function stub (N) { + var ph = new Array(N); + for (var i = 0; i < N; i++) { + ph[i] = 0; + } + + return ph; + }; + + FFTM.prototype.mulp = function mulp (x, y, out) { + var N = 2 * this.guessLen13b(x.length, y.length); + + var rbt = this.makeRBT(N); + + var _ = this.stub(N); + + var rws = new Array(N); + var rwst = new Array(N); + var iwst = new Array(N); + + var nrws = new Array(N); + var nrwst = new Array(N); + var niwst = new Array(N); + + var rmws = out.words; + rmws.length = N; + + this.convert13b(x.words, x.length, rws, N); + this.convert13b(y.words, y.length, nrws, N); + + this.transform(rws, _, rwst, iwst, N, rbt); + this.transform(nrws, _, nrwst, niwst, N, rbt); + + for (var i = 0; i < N; i++) { + var rx = rwst[i] * nrwst[i] - iwst[i] * niwst[i]; + iwst[i] = rwst[i] * niwst[i] + iwst[i] * nrwst[i]; + rwst[i] = rx; + } + + this.conjugate(rwst, iwst, N); + this.transform(rwst, iwst, rmws, _, N, rbt); + this.conjugate(rmws, _, N); + this.normalize13b(rmws, N); + + out.negative = x.negative ^ y.negative; + out.length = x.length + y.length; + return out.strip(); + }; + + // Multiply `this` by `num` + BN.prototype.mul = function mul (num) { + var out = new BN(null); + out.words = new Array(this.length + num.length); + return this.mulTo(num, out); + }; + + // Multiply employing FFT + BN.prototype.mulf = function mulf (num) { + var out = new BN(null); + out.words = new Array(this.length + num.length); + return jumboMulTo(this, num, out); + }; + + // In-place Multiplication + BN.prototype.imul = function imul (num) { + return this.clone().mulTo(num, this); + }; + + BN.prototype.imuln = function imuln (num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + + // Carry + var carry = 0; + for (var i = 0; i < this.length; i++) { + var w = (this.words[i] | 0) * num; + var lo = (w & 0x3ffffff) + (carry & 0x3ffffff); + carry >>= 26; + carry += (w / 0x4000000) | 0; + // NOTE: lo is 27bit maximum + carry += lo >>> 26; + this.words[i] = lo & 0x3ffffff; + } + + if (carry !== 0) { + this.words[i] = carry; + this.length++; + } + + return this; + }; + + BN.prototype.muln = function muln (num) { + return this.clone().imuln(num); + }; + + // `this` * `this` + BN.prototype.sqr = function sqr () { + return this.mul(this); + }; + + // `this` * `this` in-place + BN.prototype.isqr = function isqr () { + return this.imul(this.clone()); + }; + + // Math.pow(`this`, `num`) + BN.prototype.pow = function pow (num) { + var w = toBitArray(num); + if (w.length === 0) return new BN(1); + + // Skip leading zeroes + var res = this; + for (var i = 0; i < w.length; i++, res = res.sqr()) { + if (w[i] !== 0) break; + } + + if (++i < w.length) { + for (var q = res.sqr(); i < w.length; i++, q = q.sqr()) { + if (w[i] === 0) continue; + + res = res.mul(q); + } + } + + return res; + }; + + // Shift-left in-place + BN.prototype.iushln = function iushln (bits) { + assert(typeof bits === 'number' && bits >= 0); + var r = bits % 26; + var s = (bits - r) / 26; + var carryMask = (0x3ffffff >>> (26 - r)) << (26 - r); + var i; + + if (r !== 0) { + var carry = 0; + + for (i = 0; i < this.length; i++) { + var newCarry = this.words[i] & carryMask; + var c = ((this.words[i] | 0) - newCarry) << r; + this.words[i] = c | carry; + carry = newCarry >>> (26 - r); + } + + if (carry) { + this.words[i] = carry; + this.length++; + } + } + + if (s !== 0) { + for (i = this.length - 1; i >= 0; i--) { + this.words[i + s] = this.words[i]; + } + + for (i = 0; i < s; i++) { + this.words[i] = 0; + } + + this.length += s; + } + + return this.strip(); + }; + + BN.prototype.ishln = function ishln (bits) { + // TODO(indutny): implement me + assert(this.negative === 0); + return this.iushln(bits); + }; + + // Shift-right in-place + // NOTE: `hint` is a lowest bit before trailing zeroes + // NOTE: if `extended` is present - it will be filled with destroyed bits + BN.prototype.iushrn = function iushrn (bits, hint, extended) { + assert(typeof bits === 'number' && bits >= 0); + var h; + if (hint) { + h = (hint - (hint % 26)) / 26; + } else { + h = 0; + } + + var r = bits % 26; + var s = Math.min((bits - r) / 26, this.length); + var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); + var maskedWords = extended; + + h -= s; + h = Math.max(0, h); + + // Extended mode, copy masked part + if (maskedWords) { + for (var i = 0; i < s; i++) { + maskedWords.words[i] = this.words[i]; + } + maskedWords.length = s; + } + + if (s === 0) ; else if (this.length > s) { + this.length -= s; + for (i = 0; i < this.length; i++) { + this.words[i] = this.words[i + s]; + } + } else { + this.words[0] = 0; + this.length = 1; + } + + var carry = 0; + for (i = this.length - 1; i >= 0 && (carry !== 0 || i >= h); i--) { + var word = this.words[i] | 0; + this.words[i] = (carry << (26 - r)) | (word >>> r); + carry = word & mask; + } + + // Push carried bits as a mask + if (maskedWords && carry !== 0) { + maskedWords.words[maskedWords.length++] = carry; + } + + if (this.length === 0) { + this.words[0] = 0; + this.length = 1; + } + + return this.strip(); + }; + + BN.prototype.ishrn = function ishrn (bits, hint, extended) { + // TODO(indutny): implement me + assert(this.negative === 0); + return this.iushrn(bits, hint, extended); + }; + + // Shift-left + BN.prototype.shln = function shln (bits) { + return this.clone().ishln(bits); + }; + + BN.prototype.ushln = function ushln (bits) { + return this.clone().iushln(bits); + }; + + // Shift-right + BN.prototype.shrn = function shrn (bits) { + return this.clone().ishrn(bits); + }; + + BN.prototype.ushrn = function ushrn (bits) { + return this.clone().iushrn(bits); + }; + + // Test if n bit is set + BN.prototype.testn = function testn (bit) { + assert(typeof bit === 'number' && bit >= 0); + var r = bit % 26; + var s = (bit - r) / 26; + var q = 1 << r; + + // Fast case: bit is much higher than all existing words + if (this.length <= s) return false; + + // Check bit and return + var w = this.words[s]; + + return !!(w & q); + }; + + // Return only lowers bits of number (in-place) + BN.prototype.imaskn = function imaskn (bits) { + assert(typeof bits === 'number' && bits >= 0); + var r = bits % 26; + var s = (bits - r) / 26; + + assert(this.negative === 0, 'imaskn works only with positive numbers'); + + if (this.length <= s) { + return this; + } + + if (r !== 0) { + s++; + } + this.length = Math.min(s, this.length); + + if (r !== 0) { + var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); + this.words[this.length - 1] &= mask; + } + + return this.strip(); + }; + + // Return only lowers bits of number + BN.prototype.maskn = function maskn (bits) { + return this.clone().imaskn(bits); + }; + + // Add plain number `num` to `this` + BN.prototype.iaddn = function iaddn (num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + if (num < 0) return this.isubn(-num); + + // Possible sign change + if (this.negative !== 0) { + if (this.length === 1 && (this.words[0] | 0) < num) { + this.words[0] = num - (this.words[0] | 0); + this.negative = 0; + return this; + } + + this.negative = 0; + this.isubn(num); + this.negative = 1; + return this; + } + + // Add without checks + return this._iaddn(num); + }; + + BN.prototype._iaddn = function _iaddn (num) { + this.words[0] += num; + + // Carry + for (var i = 0; i < this.length && this.words[i] >= 0x4000000; i++) { + this.words[i] -= 0x4000000; + if (i === this.length - 1) { + this.words[i + 1] = 1; + } else { + this.words[i + 1]++; + } + } + this.length = Math.max(this.length, i + 1); + + return this; + }; + + // Subtract plain number `num` from `this` + BN.prototype.isubn = function isubn (num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + if (num < 0) return this.iaddn(-num); + + if (this.negative !== 0) { + this.negative = 0; + this.iaddn(num); + this.negative = 1; + return this; + } + + this.words[0] -= num; + + if (this.length === 1 && this.words[0] < 0) { + this.words[0] = -this.words[0]; + this.negative = 1; + } else { + // Carry + for (var i = 0; i < this.length && this.words[i] < 0; i++) { + this.words[i] += 0x4000000; + this.words[i + 1] -= 1; + } + } + + return this.strip(); + }; + + BN.prototype.addn = function addn (num) { + return this.clone().iaddn(num); + }; + + BN.prototype.subn = function subn (num) { + return this.clone().isubn(num); + }; + + BN.prototype.iabs = function iabs () { + this.negative = 0; + + return this; + }; + + BN.prototype.abs = function abs () { + return this.clone().iabs(); + }; + + BN.prototype._ishlnsubmul = function _ishlnsubmul (num, mul, shift) { + var len = num.length + shift; + var i; + + this._expand(len); + + var w; + var carry = 0; + for (i = 0; i < num.length; i++) { + w = (this.words[i + shift] | 0) + carry; + var right = (num.words[i] | 0) * mul; + w -= right & 0x3ffffff; + carry = (w >> 26) - ((right / 0x4000000) | 0); + this.words[i + shift] = w & 0x3ffffff; + } + for (; i < this.length - shift; i++) { + w = (this.words[i + shift] | 0) + carry; + carry = w >> 26; + this.words[i + shift] = w & 0x3ffffff; + } + + if (carry === 0) return this.strip(); + + // Subtraction overflow + assert(carry === -1); + carry = 0; + for (i = 0; i < this.length; i++) { + w = -(this.words[i] | 0) + carry; + carry = w >> 26; + this.words[i] = w & 0x3ffffff; + } + this.negative = 1; + + return this.strip(); + }; + + BN.prototype._wordDiv = function _wordDiv (num, mode) { + var shift = this.length - num.length; + + var a = this.clone(); + var b = num; + + // Normalize + var bhi = b.words[b.length - 1] | 0; + var bhiBits = this._countBits(bhi); + shift = 26 - bhiBits; + if (shift !== 0) { + b = b.ushln(shift); + a.iushln(shift); + bhi = b.words[b.length - 1] | 0; + } + + // Initialize quotient + var m = a.length - b.length; + var q; + + if (mode !== 'mod') { + q = new BN(null); + q.length = m + 1; + q.words = new Array(q.length); + for (var i = 0; i < q.length; i++) { + q.words[i] = 0; + } + } + + var diff = a.clone()._ishlnsubmul(b, 1, m); + if (diff.negative === 0) { + a = diff; + if (q) { + q.words[m] = 1; + } + } + + for (var j = m - 1; j >= 0; j--) { + var qj = (a.words[b.length + j] | 0) * 0x4000000 + + (a.words[b.length + j - 1] | 0); + + // NOTE: (qj / bhi) is (0x3ffffff * 0x4000000 + 0x3ffffff) / 0x2000000 max + // (0x7ffffff) + qj = Math.min((qj / bhi) | 0, 0x3ffffff); + + a._ishlnsubmul(b, qj, j); + while (a.negative !== 0) { + qj--; + a.negative = 0; + a._ishlnsubmul(b, 1, j); + if (!a.isZero()) { + a.negative ^= 1; + } + } + if (q) { + q.words[j] = qj; + } + } + if (q) { + q.strip(); + } + a.strip(); + + // Denormalize + if (mode !== 'div' && shift !== 0) { + a.iushrn(shift); + } + + return { + div: q || null, + mod: a + }; + }; + + // NOTE: 1) `mode` can be set to `mod` to request mod only, + // to `div` to request div only, or be absent to + // request both div & mod + // 2) `positive` is true if unsigned mod is requested + BN.prototype.divmod = function divmod (num, mode, positive) { + assert(!num.isZero()); + + if (this.isZero()) { + return { + div: new BN(0), + mod: new BN(0) + }; + } + + var div, mod, res; + if (this.negative !== 0 && num.negative === 0) { + res = this.neg().divmod(num, mode); + + if (mode !== 'mod') { + div = res.div.neg(); + } + + if (mode !== 'div') { + mod = res.mod.neg(); + if (positive && mod.negative !== 0) { + mod.iadd(num); + } + } + + return { + div: div, + mod: mod + }; + } + + if (this.negative === 0 && num.negative !== 0) { + res = this.divmod(num.neg(), mode); + + if (mode !== 'mod') { + div = res.div.neg(); + } + + return { + div: div, + mod: res.mod + }; + } + + if ((this.negative & num.negative) !== 0) { + res = this.neg().divmod(num.neg(), mode); + + if (mode !== 'div') { + mod = res.mod.neg(); + if (positive && mod.negative !== 0) { + mod.isub(num); + } + } + + return { + div: res.div, + mod: mod + }; + } + + // Both numbers are positive at this point + + // Strip both numbers to approximate shift value + if (num.length > this.length || this.cmp(num) < 0) { + return { + div: new BN(0), + mod: this + }; + } + + // Very short reduction + if (num.length === 1) { + if (mode === 'div') { + return { + div: this.divn(num.words[0]), + mod: null + }; + } + + if (mode === 'mod') { + return { + div: null, + mod: new BN(this.modn(num.words[0])) + }; + } + + return { + div: this.divn(num.words[0]), + mod: new BN(this.modn(num.words[0])) + }; + } + + return this._wordDiv(num, mode); + }; + + // Find `this` / `num` + BN.prototype.div = function div (num) { + return this.divmod(num, 'div', false).div; + }; + + // Find `this` % `num` + BN.prototype.mod = function mod (num) { + return this.divmod(num, 'mod', false).mod; + }; + + BN.prototype.umod = function umod (num) { + return this.divmod(num, 'mod', true).mod; + }; + + // Find Round(`this` / `num`) + BN.prototype.divRound = function divRound (num) { + var dm = this.divmod(num); + + // Fast case - exact division + if (dm.mod.isZero()) return dm.div; + + var mod = dm.div.negative !== 0 ? dm.mod.isub(num) : dm.mod; + + var half = num.ushrn(1); + var r2 = num.andln(1); + var cmp = mod.cmp(half); + + // Round down + if (cmp < 0 || r2 === 1 && cmp === 0) return dm.div; + + // Round up + return dm.div.negative !== 0 ? dm.div.isubn(1) : dm.div.iaddn(1); + }; + + BN.prototype.modn = function modn (num) { + assert(num <= 0x3ffffff); + var p = (1 << 26) % num; + + var acc = 0; + for (var i = this.length - 1; i >= 0; i--) { + acc = (p * acc + (this.words[i] | 0)) % num; + } + + return acc; + }; + + // In-place division by number + BN.prototype.idivn = function idivn (num) { + assert(num <= 0x3ffffff); + + var carry = 0; + for (var i = this.length - 1; i >= 0; i--) { + var w = (this.words[i] | 0) + carry * 0x4000000; + this.words[i] = (w / num) | 0; + carry = w % num; + } + + return this.strip(); + }; + + BN.prototype.divn = function divn (num) { + return this.clone().idivn(num); + }; + + BN.prototype.egcd = function egcd (p) { + assert(p.negative === 0); + assert(!p.isZero()); + + var x = this; + var y = p.clone(); + + if (x.negative !== 0) { + x = x.umod(p); + } else { + x = x.clone(); + } + + // A * x + B * y = x + var A = new BN(1); + var B = new BN(0); + + // C * x + D * y = y + var C = new BN(0); + var D = new BN(1); + + var g = 0; + + while (x.isEven() && y.isEven()) { + x.iushrn(1); + y.iushrn(1); + ++g; + } + + var yp = y.clone(); + var xp = x.clone(); + + while (!x.isZero()) { + for (var i = 0, im = 1; (x.words[0] & im) === 0 && i < 26; ++i, im <<= 1); + if (i > 0) { + x.iushrn(i); + while (i-- > 0) { + if (A.isOdd() || B.isOdd()) { + A.iadd(yp); + B.isub(xp); + } + + A.iushrn(1); + B.iushrn(1); + } + } + + for (var j = 0, jm = 1; (y.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); + if (j > 0) { + y.iushrn(j); + while (j-- > 0) { + if (C.isOdd() || D.isOdd()) { + C.iadd(yp); + D.isub(xp); + } + + C.iushrn(1); + D.iushrn(1); + } + } + + if (x.cmp(y) >= 0) { + x.isub(y); + A.isub(C); + B.isub(D); + } else { + y.isub(x); + C.isub(A); + D.isub(B); + } + } + + return { + a: C, + b: D, + gcd: y.iushln(g) + }; + }; + + // This is reduced incarnation of the binary EEA + // above, designated to invert members of the + // _prime_ fields F(p) at a maximal speed + BN.prototype._invmp = function _invmp (p) { + assert(p.negative === 0); + assert(!p.isZero()); + + var a = this; + var b = p.clone(); + + if (a.negative !== 0) { + a = a.umod(p); + } else { + a = a.clone(); + } + + var x1 = new BN(1); + var x2 = new BN(0); + + var delta = b.clone(); + + while (a.cmpn(1) > 0 && b.cmpn(1) > 0) { + for (var i = 0, im = 1; (a.words[0] & im) === 0 && i < 26; ++i, im <<= 1); + if (i > 0) { + a.iushrn(i); + while (i-- > 0) { + if (x1.isOdd()) { + x1.iadd(delta); + } + + x1.iushrn(1); + } + } + + for (var j = 0, jm = 1; (b.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); + if (j > 0) { + b.iushrn(j); + while (j-- > 0) { + if (x2.isOdd()) { + x2.iadd(delta); + } + + x2.iushrn(1); + } + } + + if (a.cmp(b) >= 0) { + a.isub(b); + x1.isub(x2); + } else { + b.isub(a); + x2.isub(x1); + } + } + + var res; + if (a.cmpn(1) === 0) { + res = x1; + } else { + res = x2; + } + + if (res.cmpn(0) < 0) { + res.iadd(p); + } + + return res; + }; + + BN.prototype.gcd = function gcd (num) { + if (this.isZero()) return num.abs(); + if (num.isZero()) return this.abs(); + + var a = this.clone(); + var b = num.clone(); + a.negative = 0; + b.negative = 0; + + // Remove common factor of two + for (var shift = 0; a.isEven() && b.isEven(); shift++) { + a.iushrn(1); + b.iushrn(1); + } + + do { + while (a.isEven()) { + a.iushrn(1); + } + while (b.isEven()) { + b.iushrn(1); + } + + var r = a.cmp(b); + if (r < 0) { + // Swap `a` and `b` to make `a` always bigger than `b` + var t = a; + a = b; + b = t; + } else if (r === 0 || b.cmpn(1) === 0) { + break; + } + + a.isub(b); + } while (true); + + return b.iushln(shift); + }; + + // Invert number in the field F(num) + BN.prototype.invm = function invm (num) { + return this.egcd(num).a.umod(num); + }; + + BN.prototype.isEven = function isEven () { + return (this.words[0] & 1) === 0; + }; + + BN.prototype.isOdd = function isOdd () { + return (this.words[0] & 1) === 1; + }; + + // And first word and num + BN.prototype.andln = function andln (num) { + return this.words[0] & num; + }; + + // Increment at the bit position in-line + BN.prototype.bincn = function bincn (bit) { + assert(typeof bit === 'number'); + var r = bit % 26; + var s = (bit - r) / 26; + var q = 1 << r; + + // Fast case: bit is much higher than all existing words + if (this.length <= s) { + this._expand(s + 1); + this.words[s] |= q; + return this; + } + + // Add bit and propagate, if needed + var carry = q; + for (var i = s; carry !== 0 && i < this.length; i++) { + var w = this.words[i] | 0; + w += carry; + carry = w >>> 26; + w &= 0x3ffffff; + this.words[i] = w; + } + if (carry !== 0) { + this.words[i] = carry; + this.length++; + } + return this; + }; + + BN.prototype.isZero = function isZero () { + return this.length === 1 && this.words[0] === 0; + }; + + BN.prototype.cmpn = function cmpn (num) { + var negative = num < 0; + + if (this.negative !== 0 && !negative) return -1; + if (this.negative === 0 && negative) return 1; + + this.strip(); + + var res; + if (this.length > 1) { + res = 1; + } else { + if (negative) { + num = -num; + } + + assert(num <= 0x3ffffff, 'Number is too big'); + + var w = this.words[0] | 0; + res = w === num ? 0 : w < num ? -1 : 1; + } + if (this.negative !== 0) return -res | 0; + return res; + }; + + // Compare two numbers and return: + // 1 - if `this` > `num` + // 0 - if `this` == `num` + // -1 - if `this` < `num` + BN.prototype.cmp = function cmp (num) { + if (this.negative !== 0 && num.negative === 0) return -1; + if (this.negative === 0 && num.negative !== 0) return 1; + + var res = this.ucmp(num); + if (this.negative !== 0) return -res | 0; + return res; + }; + + // Unsigned comparison + BN.prototype.ucmp = function ucmp (num) { + // At this point both numbers have the same sign + if (this.length > num.length) return 1; + if (this.length < num.length) return -1; + + var res = 0; + for (var i = this.length - 1; i >= 0; i--) { + var a = this.words[i] | 0; + var b = num.words[i] | 0; + + if (a === b) continue; + if (a < b) { + res = -1; + } else if (a > b) { + res = 1; + } + break; + } + return res; + }; + + BN.prototype.gtn = function gtn (num) { + return this.cmpn(num) === 1; + }; + + BN.prototype.gt = function gt (num) { + return this.cmp(num) === 1; + }; + + BN.prototype.gten = function gten (num) { + return this.cmpn(num) >= 0; + }; + + BN.prototype.gte = function gte (num) { + return this.cmp(num) >= 0; + }; + + BN.prototype.ltn = function ltn (num) { + return this.cmpn(num) === -1; + }; + + BN.prototype.lt = function lt (num) { + return this.cmp(num) === -1; + }; + + BN.prototype.lten = function lten (num) { + return this.cmpn(num) <= 0; + }; + + BN.prototype.lte = function lte (num) { + return this.cmp(num) <= 0; + }; + + BN.prototype.eqn = function eqn (num) { + return this.cmpn(num) === 0; + }; + + BN.prototype.eq = function eq (num) { + return this.cmp(num) === 0; + }; + + // + // A reduce context, could be using montgomery or something better, depending + // on the `m` itself. + // + BN.red = function red (num) { + return new Red(num); + }; + + BN.prototype.toRed = function toRed (ctx) { + assert(!this.red, 'Already a number in reduction context'); + assert(this.negative === 0, 'red works only with positives'); + return ctx.convertTo(this)._forceRed(ctx); + }; + + BN.prototype.fromRed = function fromRed () { + assert(this.red, 'fromRed works only with numbers in reduction context'); + return this.red.convertFrom(this); + }; + + BN.prototype._forceRed = function _forceRed (ctx) { + this.red = ctx; + return this; + }; + + BN.prototype.forceRed = function forceRed (ctx) { + assert(!this.red, 'Already a number in reduction context'); + return this._forceRed(ctx); + }; + + BN.prototype.redAdd = function redAdd (num) { + assert(this.red, 'redAdd works only with red numbers'); + return this.red.add(this, num); + }; + + BN.prototype.redIAdd = function redIAdd (num) { + assert(this.red, 'redIAdd works only with red numbers'); + return this.red.iadd(this, num); + }; + + BN.prototype.redSub = function redSub (num) { + assert(this.red, 'redSub works only with red numbers'); + return this.red.sub(this, num); + }; + + BN.prototype.redISub = function redISub (num) { + assert(this.red, 'redISub works only with red numbers'); + return this.red.isub(this, num); + }; + + BN.prototype.redShl = function redShl (num) { + assert(this.red, 'redShl works only with red numbers'); + return this.red.shl(this, num); + }; + + BN.prototype.redMul = function redMul (num) { + assert(this.red, 'redMul works only with red numbers'); + this.red._verify2(this, num); + return this.red.mul(this, num); + }; + + BN.prototype.redIMul = function redIMul (num) { + assert(this.red, 'redMul works only with red numbers'); + this.red._verify2(this, num); + return this.red.imul(this, num); + }; + + BN.prototype.redSqr = function redSqr () { + assert(this.red, 'redSqr works only with red numbers'); + this.red._verify1(this); + return this.red.sqr(this); + }; + + BN.prototype.redISqr = function redISqr () { + assert(this.red, 'redISqr works only with red numbers'); + this.red._verify1(this); + return this.red.isqr(this); + }; + + // Square root over p + BN.prototype.redSqrt = function redSqrt () { + assert(this.red, 'redSqrt works only with red numbers'); + this.red._verify1(this); + return this.red.sqrt(this); + }; + + BN.prototype.redInvm = function redInvm () { + assert(this.red, 'redInvm works only with red numbers'); + this.red._verify1(this); + return this.red.invm(this); + }; + + // Return negative clone of `this` % `red modulo` + BN.prototype.redNeg = function redNeg () { + assert(this.red, 'redNeg works only with red numbers'); + this.red._verify1(this); + return this.red.neg(this); + }; + + BN.prototype.redPow = function redPow (num) { + assert(this.red && !num.red, 'redPow(normalNum)'); + this.red._verify1(this); + return this.red.pow(this, num); + }; + + // Prime numbers with efficient reduction + var primes = { + k256: null, + p224: null, + p192: null, + p25519: null + }; + + // Pseudo-Mersenne prime + function MPrime (name, p) { + // P = 2 ^ N - K + this.name = name; + this.p = new BN(p, 16); + this.n = this.p.bitLength(); + this.k = new BN(1).iushln(this.n).isub(this.p); + + this.tmp = this._tmp(); + } + + MPrime.prototype._tmp = function _tmp () { + var tmp = new BN(null); + tmp.words = new Array(Math.ceil(this.n / 13)); + return tmp; + }; + + MPrime.prototype.ireduce = function ireduce (num) { + // Assumes that `num` is less than `P^2` + // num = HI * (2 ^ N - K) + HI * K + LO = HI * K + LO (mod P) + var r = num; + var rlen; + + do { + this.split(r, this.tmp); + r = this.imulK(r); + r = r.iadd(this.tmp); + rlen = r.bitLength(); + } while (rlen > this.n); + + var cmp = rlen < this.n ? -1 : r.ucmp(this.p); + if (cmp === 0) { + r.words[0] = 0; + r.length = 1; + } else if (cmp > 0) { + r.isub(this.p); + } else { + r.strip(); + } + + return r; + }; + + MPrime.prototype.split = function split (input, out) { + input.iushrn(this.n, 0, out); + }; + + MPrime.prototype.imulK = function imulK (num) { + return num.imul(this.k); + }; + + function K256 () { + MPrime.call( + this, + 'k256', + 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f'); + } + inherits(K256, MPrime); + + K256.prototype.split = function split (input, output) { + // 256 = 9 * 26 + 22 + var mask = 0x3fffff; + + var outLen = Math.min(input.length, 9); + for (var i = 0; i < outLen; i++) { + output.words[i] = input.words[i]; + } + output.length = outLen; + + if (input.length <= 9) { + input.words[0] = 0; + input.length = 1; + return; + } + + // Shift by 9 limbs + var prev = input.words[9]; + output.words[output.length++] = prev & mask; + + for (i = 10; i < input.length; i++) { + var next = input.words[i] | 0; + input.words[i - 10] = ((next & mask) << 4) | (prev >>> 22); + prev = next; + } + prev >>>= 22; + input.words[i - 10] = prev; + if (prev === 0 && input.length > 10) { + input.length -= 10; + } else { + input.length -= 9; + } + }; + + K256.prototype.imulK = function imulK (num) { + // K = 0x1000003d1 = [ 0x40, 0x3d1 ] + num.words[num.length] = 0; + num.words[num.length + 1] = 0; + num.length += 2; + + // bounded at: 0x40 * 0x3ffffff + 0x3d0 = 0x100000390 + var lo = 0; + for (var i = 0; i < num.length; i++) { + var w = num.words[i] | 0; + lo += w * 0x3d1; + num.words[i] = lo & 0x3ffffff; + lo = w * 0x40 + ((lo / 0x4000000) | 0); + } + + // Fast length reduction + if (num.words[num.length - 1] === 0) { + num.length--; + if (num.words[num.length - 1] === 0) { + num.length--; + } + } + return num; + }; + + function P224 () { + MPrime.call( + this, + 'p224', + 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001'); + } + inherits(P224, MPrime); + + function P192 () { + MPrime.call( + this, + 'p192', + 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff'); + } + inherits(P192, MPrime); + + function P25519 () { + // 2 ^ 255 - 19 + MPrime.call( + this, + '25519', + '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed'); + } + inherits(P25519, MPrime); + + P25519.prototype.imulK = function imulK (num) { + // K = 0x13 + var carry = 0; + for (var i = 0; i < num.length; i++) { + var hi = (num.words[i] | 0) * 0x13 + carry; + var lo = hi & 0x3ffffff; + hi >>>= 26; + + num.words[i] = lo; + carry = hi; + } + if (carry !== 0) { + num.words[num.length++] = carry; + } + return num; + }; + + // Exported mostly for testing purposes, use plain name instead + BN._prime = function prime (name) { + // Cached version of prime + if (primes[name]) return primes[name]; + + var prime; + if (name === 'k256') { + prime = new K256(); + } else if (name === 'p224') { + prime = new P224(); + } else if (name === 'p192') { + prime = new P192(); + } else if (name === 'p25519') { + prime = new P25519(); + } else { + throw new Error('Unknown prime ' + name); + } + primes[name] = prime; + + return prime; + }; + + // + // Base reduction engine + // + function Red (m) { + if (typeof m === 'string') { + var prime = BN._prime(m); + this.m = prime.p; + this.prime = prime; + } else { + assert(m.gtn(1), 'modulus must be greater than 1'); + this.m = m; + this.prime = null; + } + } + + Red.prototype._verify1 = function _verify1 (a) { + assert(a.negative === 0, 'red works only with positives'); + assert(a.red, 'red works only with red numbers'); + }; + + Red.prototype._verify2 = function _verify2 (a, b) { + assert((a.negative | b.negative) === 0, 'red works only with positives'); + assert(a.red && a.red === b.red, + 'red works only with red numbers'); + }; + + Red.prototype.imod = function imod (a) { + if (this.prime) return this.prime.ireduce(a)._forceRed(this); + return a.umod(this.m)._forceRed(this); + }; + + Red.prototype.neg = function neg (a) { + if (a.isZero()) { + return a.clone(); + } + + return this.m.sub(a)._forceRed(this); + }; + + Red.prototype.add = function add (a, b) { + this._verify2(a, b); + + var res = a.add(b); + if (res.cmp(this.m) >= 0) { + res.isub(this.m); + } + return res._forceRed(this); + }; + + Red.prototype.iadd = function iadd (a, b) { + this._verify2(a, b); + + var res = a.iadd(b); + if (res.cmp(this.m) >= 0) { + res.isub(this.m); + } + return res; + }; + + Red.prototype.sub = function sub (a, b) { + this._verify2(a, b); + + var res = a.sub(b); + if (res.cmpn(0) < 0) { + res.iadd(this.m); + } + return res._forceRed(this); + }; + + Red.prototype.isub = function isub (a, b) { + this._verify2(a, b); + + var res = a.isub(b); + if (res.cmpn(0) < 0) { + res.iadd(this.m); + } + return res; + }; + + Red.prototype.shl = function shl (a, num) { + this._verify1(a); + return this.imod(a.ushln(num)); + }; + + Red.prototype.imul = function imul (a, b) { + this._verify2(a, b); + return this.imod(a.imul(b)); + }; + + Red.prototype.mul = function mul (a, b) { + this._verify2(a, b); + return this.imod(a.mul(b)); + }; + + Red.prototype.isqr = function isqr (a) { + return this.imul(a, a.clone()); + }; + + Red.prototype.sqr = function sqr (a) { + return this.mul(a, a); + }; + + Red.prototype.sqrt = function sqrt (a) { + if (a.isZero()) return a.clone(); + + var mod3 = this.m.andln(3); + assert(mod3 % 2 === 1); + + // Fast case + if (mod3 === 3) { + var pow = this.m.add(new BN(1)).iushrn(2); + return this.pow(a, pow); + } + + // Tonelli-Shanks algorithm (Totally unoptimized and slow) + // + // Find Q and S, that Q * 2 ^ S = (P - 1) + var q = this.m.subn(1); + var s = 0; + while (!q.isZero() && q.andln(1) === 0) { + s++; + q.iushrn(1); + } + assert(!q.isZero()); + + var one = new BN(1).toRed(this); + var nOne = one.redNeg(); + + // Find quadratic non-residue + // NOTE: Max is such because of generalized Riemann hypothesis. + var lpow = this.m.subn(1).iushrn(1); + var z = this.m.bitLength(); + z = new BN(2 * z * z).toRed(this); + + while (this.pow(z, lpow).cmp(nOne) !== 0) { + z.redIAdd(nOne); + } + + var c = this.pow(z, q); + var r = this.pow(a, q.addn(1).iushrn(1)); + var t = this.pow(a, q); + var m = s; + while (t.cmp(one) !== 0) { + var tmp = t; + for (var i = 0; tmp.cmp(one) !== 0; i++) { + tmp = tmp.redSqr(); + } + assert(i < m); + var b = this.pow(c, new BN(1).iushln(m - i - 1)); + + r = r.redMul(b); + c = b.redSqr(); + t = t.redMul(c); + m = i; + } + + return r; + }; + + Red.prototype.invm = function invm (a) { + var inv = a._invmp(this.m); + if (inv.negative !== 0) { + inv.negative = 0; + return this.imod(inv).redNeg(); + } else { + return this.imod(inv); + } + }; + + Red.prototype.pow = function pow (a, num) { + if (num.isZero()) return new BN(1).toRed(this); + if (num.cmpn(1) === 0) return a.clone(); + + var windowSize = 4; + var wnd = new Array(1 << windowSize); + wnd[0] = new BN(1).toRed(this); + wnd[1] = a; + for (var i = 2; i < wnd.length; i++) { + wnd[i] = this.mul(wnd[i - 1], a); + } + + var res = wnd[0]; + var current = 0; + var currentLen = 0; + var start = num.bitLength() % 26; + if (start === 0) { + start = 26; + } + + for (i = num.length - 1; i >= 0; i--) { + var word = num.words[i]; + for (var j = start - 1; j >= 0; j--) { + var bit = (word >> j) & 1; + if (res !== wnd[0]) { + res = this.sqr(res); + } + + if (bit === 0 && current === 0) { + currentLen = 0; + continue; + } + + current <<= 1; + current |= bit; + currentLen++; + if (currentLen !== windowSize && (i !== 0 || j !== 0)) continue; + + res = this.mul(res, wnd[current]); + currentLen = 0; + current = 0; + } + start = 26; + } + + return res; + }; + + Red.prototype.convertTo = function convertTo (num) { + var r = num.umod(this.m); + + return r === num ? r.clone() : r; + }; + + Red.prototype.convertFrom = function convertFrom (num) { + var res = num.clone(); + res.red = null; + return res; + }; + + // + // Montgomery method engine + // + + BN.mont = function mont (num) { + return new Mont(num); + }; + + function Mont (m) { + Red.call(this, m); + + this.shift = this.m.bitLength(); + if (this.shift % 26 !== 0) { + this.shift += 26 - (this.shift % 26); + } + + this.r = new BN(1).iushln(this.shift); + this.r2 = this.imod(this.r.sqr()); + this.rinv = this.r._invmp(this.m); + + this.minv = this.rinv.mul(this.r).isubn(1).div(this.m); + this.minv = this.minv.umod(this.r); + this.minv = this.r.sub(this.minv); + } + inherits(Mont, Red); + + Mont.prototype.convertTo = function convertTo (num) { + return this.imod(num.ushln(this.shift)); + }; + + Mont.prototype.convertFrom = function convertFrom (num) { + var r = this.imod(num.mul(this.rinv)); + r.red = null; + return r; + }; + + Mont.prototype.imul = function imul (a, b) { + if (a.isZero() || b.isZero()) { + a.words[0] = 0; + a.length = 1; + return a; + } + + var t = a.imul(b); + var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); + var u = t.isub(c).iushrn(this.shift); + var res = u; + + if (u.cmp(this.m) >= 0) { + res = u.isub(this.m); + } else if (u.cmpn(0) < 0) { + res = u.iadd(this.m); + } + + return res._forceRed(this); + }; + + Mont.prototype.mul = function mul (a, b) { + if (a.isZero() || b.isZero()) return new BN(0)._forceRed(this); + + var t = a.mul(b); + var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); + var u = t.isub(c).iushrn(this.shift); + var res = u; + if (u.cmp(this.m) >= 0) { + res = u.isub(this.m); + } else if (u.cmpn(0) < 0) { + res = u.iadd(this.m); + } + + return res._forceRed(this); + }; + + Mont.prototype.invm = function invm (a) { + // (AR)^-1 * R^2 = (A^-1 * R^-1) * R^2 = A^-1 * R + var res = this.imod(a._invmp(this.m).mul(this.r2)); + return res._forceRed(this); + }; + })(module, commonjsGlobal); + }); + + var bn$1 = /*#__PURE__*/Object.freeze({ + __proto__: null, + 'default': bn, + __moduleExports: bn + }); + + /** + * @fileoverview + * BigInteger implementation of basic operations + * Wrapper of bn.js library (wwww.github.com/indutny/bn.js) + * @module biginteger/bn + * @private + */ + + /** + * @private + */ + class BigInteger$1 { + /** + * Get a BigInteger (input must be big endian for strings and arrays) + * @param {Number|String|Uint8Array} n - Value to convert + * @throws {Error} on undefined input + */ + constructor(n) { + if (n === undefined) { + throw new Error('Invalid BigInteger input'); + } + + this.value = new bn(n); + } + + clone() { + const clone = new BigInteger$1(null); + this.value.copy(clone.value); + return clone; + } + + /** + * BigInteger increment in place + */ + iinc() { + this.value.iadd(new bn(1)); + return this; + } + + /** + * BigInteger increment + * @returns {BigInteger} this + 1. + */ + inc() { + return this.clone().iinc(); + } + + /** + * BigInteger decrement in place + */ + idec() { + this.value.isub(new bn(1)); + return this; + } + + /** + * BigInteger decrement + * @returns {BigInteger} this - 1. + */ + dec() { + return this.clone().idec(); + } + + + /** + * BigInteger addition in place + * @param {BigInteger} x - Value to add + */ + iadd(x) { + this.value.iadd(x.value); + return this; + } + + /** + * BigInteger addition + * @param {BigInteger} x - Value to add + * @returns {BigInteger} this + x. + */ + add(x) { + return this.clone().iadd(x); + } + + /** + * BigInteger subtraction in place + * @param {BigInteger} x - Value to subtract + */ + isub(x) { + this.value.isub(x.value); + return this; + } + + /** + * BigInteger subtraction + * @param {BigInteger} x - Value to subtract + * @returns {BigInteger} this - x. + */ + sub(x) { + return this.clone().isub(x); + } + + /** + * BigInteger multiplication in place + * @param {BigInteger} x - Value to multiply + */ + imul(x) { + this.value.imul(x.value); + return this; + } + + /** + * BigInteger multiplication + * @param {BigInteger} x - Value to multiply + * @returns {BigInteger} this * x. + */ + mul(x) { + return this.clone().imul(x); + } + + /** + * Compute value modulo m, in place + * @param {BigInteger} m - Modulo + */ + imod(m) { + this.value = this.value.umod(m.value); + return this; + } + + /** + * Compute value modulo m + * @param {BigInteger} m - Modulo + * @returns {BigInteger} this mod m. + */ + mod(m) { + return this.clone().imod(m); + } + + /** + * Compute modular exponentiation + * Much faster than this.exp(e).mod(n) + * @param {BigInteger} e - Exponent + * @param {BigInteger} n - Modulo + * @returns {BigInteger} this ** e mod n. + */ + modExp(e, n) { + // We use either Montgomery or normal reduction context + // Montgomery requires coprime n and R (montogmery multiplier) + // bn.js picks R as power of 2, so n must be odd + const nred = n.isEven() ? bn.red(n.value) : bn.mont(n.value); + const x = this.clone(); + x.value = x.value.toRed(nred).redPow(e.value).fromRed(); + return x; + } + + /** + * Compute the inverse of this value modulo n + * Note: this and and n must be relatively prime + * @param {BigInteger} n - Modulo + * @returns {BigInteger} x such that this*x = 1 mod n + * @throws {Error} if the inverse does not exist + */ + modInv(n) { + // invm returns a wrong result if the inverse does not exist + if (!this.gcd(n).isOne()) { + throw new Error('Inverse does not exist'); + } + return new BigInteger$1(this.value.invm(n.value)); + } + + /** + * Compute greatest common divisor between this and n + * @param {BigInteger} n - Operand + * @returns {BigInteger} gcd + */ + gcd(n) { + return new BigInteger$1(this.value.gcd(n.value)); + } + + /** + * Shift this to the left by x, in place + * @param {BigInteger} x - Shift value + */ + ileftShift(x) { + this.value.ishln(x.value.toNumber()); + return this; + } + + /** + * Shift this to the left by x + * @param {BigInteger} x - Shift value + * @returns {BigInteger} this << x. + */ + leftShift(x) { + return this.clone().ileftShift(x); + } + + /** + * Shift this to the right by x, in place + * @param {BigInteger} x - Shift value + */ + irightShift(x) { + this.value.ishrn(x.value.toNumber()); + return this; + } + + /** + * Shift this to the right by x + * @param {BigInteger} x - Shift value + * @returns {BigInteger} this >> x. + */ + rightShift(x) { + return this.clone().irightShift(x); + } + + /** + * Whether this value is equal to x + * @param {BigInteger} x + * @returns {Boolean} + */ + equal(x) { + return this.value.eq(x.value); + } + + /** + * Whether this value is less than x + * @param {BigInteger} x + * @returns {Boolean} + */ + lt(x) { + return this.value.lt(x.value); + } + + /** + * Whether this value is less than or equal to x + * @param {BigInteger} x + * @returns {Boolean} + */ + lte(x) { + return this.value.lte(x.value); + } + + /** + * Whether this value is greater than x + * @param {BigInteger} x + * @returns {Boolean} + */ + gt(x) { + return this.value.gt(x.value); + } + + /** + * Whether this value is greater than or equal to x + * @param {BigInteger} x + * @returns {Boolean} + */ + gte(x) { + return this.value.gte(x.value); + } + + isZero() { + return this.value.isZero(); + } + + isOne() { + return this.value.eq(new bn(1)); + } + + isNegative() { + return this.value.isNeg(); + } + + isEven() { + return this.value.isEven(); + } + + abs() { + const res = this.clone(); + res.value = res.value.abs(); + return res; + } + + /** + * Get this value as a string + * @returns {String} this value. + */ + toString() { + return this.value.toString(); + } + + /** + * Get this value as an exact Number (max 53 bits) + * Fails if this value is too large + * @returns {Number} + */ + toNumber() { + return this.value.toNumber(); + } + + /** + * Get value of i-th bit + * @param {Number} i - Bit index + * @returns {Number} Bit value. + */ + getBit(i) { + return this.value.testn(i) ? 1 : 0; + } + + /** + * Compute bit length + * @returns {Number} Bit length. + */ + bitLength() { + return this.value.bitLength(); + } + + /** + * Compute byte length + * @returns {Number} Byte length. + */ + byteLength() { + return this.value.byteLength(); + } + + /** + * Get Uint8Array representation of this number + * @param {String} endian - Endianess of output array (defaults to 'be') + * @param {Number} length - Of output array + * @returns {Uint8Array} + */ + toUint8Array(endian = 'be', length) { + return this.value.toArrayLike(Uint8Array, endian, length); + } + } + + var bn_interface = /*#__PURE__*/Object.freeze({ + __proto__: null, + 'default': BigInteger$1 + }); + + var utils_1 = createCommonjsModule(function (module, exports) { + + var utils = exports; + + function toArray(msg, enc) { + if (Array.isArray(msg)) + return msg.slice(); + if (!msg) + return []; + var res = []; + if (typeof msg !== 'string') { + for (var i = 0; i < msg.length; i++) + res[i] = msg[i] | 0; + return res; + } + if (enc === 'hex') { + msg = msg.replace(/[^a-z0-9]+/ig, ''); + if (msg.length % 2 !== 0) + msg = '0' + msg; + for (var i = 0; i < msg.length; i += 2) + res.push(parseInt(msg[i] + msg[i + 1], 16)); + } else { + for (var i = 0; i < msg.length; i++) { + var c = msg.charCodeAt(i); + var hi = c >> 8; + var lo = c & 0xff; + if (hi) + res.push(hi, lo); + else + res.push(lo); + } + } + return res; + } + utils.toArray = toArray; + + function zero2(word) { + if (word.length === 1) + return '0' + word; + else + return word; + } + utils.zero2 = zero2; + + function toHex(msg) { + var res = ''; + for (var i = 0; i < msg.length; i++) + res += zero2(msg[i].toString(16)); + return res; + } + utils.toHex = toHex; + + utils.encode = function encode(arr, enc) { + if (enc === 'hex') + return toHex(arr); + else + return arr; + }; + }); + + var utils_1$1 = createCommonjsModule(function (module, exports) { + + var utils = exports; + + + + + utils.assert = minimalisticAssert; + utils.toArray = utils_1.toArray; + utils.zero2 = utils_1.zero2; + utils.toHex = utils_1.toHex; + utils.encode = utils_1.encode; + + // Represent num in a w-NAF form + function getNAF(num, w) { + var naf = []; + var ws = 1 << (w + 1); + var k = num.clone(); + while (k.cmpn(1) >= 0) { + var z; + if (k.isOdd()) { + var mod = k.andln(ws - 1); + if (mod > (ws >> 1) - 1) + z = (ws >> 1) - mod; + else + z = mod; + k.isubn(z); + } else { + z = 0; + } + naf.push(z); + + // Optimization, shift by word if possible + var shift = (k.cmpn(0) !== 0 && k.andln(ws - 1) === 0) ? (w + 1) : 1; + for (var i = 1; i < shift; i++) + naf.push(0); + k.iushrn(shift); + } + + return naf; + } + utils.getNAF = getNAF; + + // Represent k1, k2 in a Joint Sparse Form + function getJSF(k1, k2) { + var jsf = [ + [], + [] + ]; + + k1 = k1.clone(); + k2 = k2.clone(); + var d1 = 0; + var d2 = 0; + while (k1.cmpn(-d1) > 0 || k2.cmpn(-d2) > 0) { + + // First phase + var m14 = (k1.andln(3) + d1) & 3; + var m24 = (k2.andln(3) + d2) & 3; + if (m14 === 3) + m14 = -1; + if (m24 === 3) + m24 = -1; + var u1; + if ((m14 & 1) === 0) { + u1 = 0; + } else { + var m8 = (k1.andln(7) + d1) & 7; + if ((m8 === 3 || m8 === 5) && m24 === 2) + u1 = -m14; + else + u1 = m14; + } + jsf[0].push(u1); + + var u2; + if ((m24 & 1) === 0) { + u2 = 0; + } else { + var m8 = (k2.andln(7) + d2) & 7; + if ((m8 === 3 || m8 === 5) && m14 === 2) + u2 = -m24; + else + u2 = m24; + } + jsf[1].push(u2); + + // Second phase + if (2 * d1 === u1 + 1) + d1 = 1 - d1; + if (2 * d2 === u2 + 1) + d2 = 1 - d2; + k1.iushrn(1); + k2.iushrn(1); + } + + return jsf; + } + utils.getJSF = getJSF; + + function cachedProperty(obj, name, computer) { + var key = '_' + name; + obj.prototype[name] = function cachedProperty() { + return this[key] !== undefined ? this[key] : + this[key] = computer.call(this); + }; + } + utils.cachedProperty = cachedProperty; + + function parseBytes(bytes) { + return typeof bytes === 'string' ? utils.toArray(bytes, 'hex') : + bytes; + } + utils.parseBytes = parseBytes; + + function intFromLE(bytes) { + return new bn(bytes, 'hex', 'le'); + } + utils.intFromLE = intFromLE; + }); + + var r$1; + + var brorand = function rand(len) { + if (!r$1) + r$1 = new Rand(null); + + return r$1.generate(len); + }; + + function Rand(rand) { + this.rand = rand; + } + var Rand_1 = Rand; + + Rand.prototype.generate = function generate(len) { + return this._rand(len); + }; + + // Emulate crypto API using randy + Rand.prototype._rand = function _rand(n) { + if (this.rand.getBytes) + return this.rand.getBytes(n); + + var res = new Uint8Array(n); + for (var i = 0; i < res.length; i++) + res[i] = this.rand.getByte(); + return res; + }; + + if (typeof self === 'object') { + if (self.crypto && self.crypto.getRandomValues) { + // Modern browsers + Rand.prototype._rand = function _rand(n) { + var arr = new Uint8Array(n); + self.crypto.getRandomValues(arr); + return arr; + }; + } else if (self.msCrypto && self.msCrypto.getRandomValues) { + // IE + Rand.prototype._rand = function _rand(n) { + var arr = new Uint8Array(n); + self.msCrypto.getRandomValues(arr); + return arr; + }; + + // Safari's WebWorkers do not have `crypto` + } else if (typeof window === 'object') { + // Old junk + Rand.prototype._rand = function() { + throw new Error('Not implemented yet'); + }; + } + } else { + // Node.js or Web worker with no crypto support + try { + var crypto$2 = void('crypto'); + if (typeof crypto$2.randomBytes !== 'function') + throw new Error('Not supported'); + + Rand.prototype._rand = function _rand(n) { + return crypto$2.randomBytes(n); + }; + } catch (e) { + } + } + brorand.Rand = Rand_1; + + var getNAF = utils_1$1.getNAF; + var getJSF = utils_1$1.getJSF; + var assert$2 = utils_1$1.assert; + + function BaseCurve(type, conf) { + this.type = type; + this.p = new bn(conf.p, 16); + + // Use Montgomery, when there is no fast reduction for the prime + this.red = conf.prime ? bn.red(conf.prime) : bn.mont(this.p); + + // Useful for many curves + this.zero = new bn(0).toRed(this.red); + this.one = new bn(1).toRed(this.red); + this.two = new bn(2).toRed(this.red); + + // Curve configuration, optional + this.n = conf.n && new bn(conf.n, 16); + this.g = conf.g && this.pointFromJSON(conf.g, conf.gRed); + + // Temporary arrays + this._wnafT1 = new Array(4); + this._wnafT2 = new Array(4); + this._wnafT3 = new Array(4); + this._wnafT4 = new Array(4); + + // Generalized Greg Maxwell's trick + var adjustCount = this.n && this.p.div(this.n); + if (!adjustCount || adjustCount.cmpn(100) > 0) { + this.redN = null; + } else { + this._maxwellTrick = true; + this.redN = this.n.toRed(this.red); + } + } + var base = BaseCurve; + + BaseCurve.prototype.point = function point() { + throw new Error('Not implemented'); + }; + + BaseCurve.prototype.validate = function validate() { + throw new Error('Not implemented'); + }; + + BaseCurve.prototype._fixedNafMul = function _fixedNafMul(p, k) { + assert$2(p.precomputed); + var doubles = p._getDoubles(); + + var naf = getNAF(k, 1); + var I = (1 << (doubles.step + 1)) - (doubles.step % 2 === 0 ? 2 : 1); + I /= 3; + + // Translate into more windowed form + var repr = []; + for (var j = 0; j < naf.length; j += doubles.step) { + var nafW = 0; + for (var k = j + doubles.step - 1; k >= j; k--) + nafW = (nafW << 1) + naf[k]; + repr.push(nafW); + } + + var a = this.jpoint(null, null, null); + var b = this.jpoint(null, null, null); + for (var i = I; i > 0; i--) { + for (var j = 0; j < repr.length; j++) { + var nafW = repr[j]; + if (nafW === i) + b = b.mixedAdd(doubles.points[j]); + else if (nafW === -i) + b = b.mixedAdd(doubles.points[j].neg()); + } + a = a.add(b); + } + return a.toP(); + }; + + BaseCurve.prototype._wnafMul = function _wnafMul(p, k) { + var w = 4; + + // Precompute window + var nafPoints = p._getNAFPoints(w); + w = nafPoints.wnd; + var wnd = nafPoints.points; + + // Get NAF form + var naf = getNAF(k, w); + + // Add `this`*(N+1) for every w-NAF index + var acc = this.jpoint(null, null, null); + for (var i = naf.length - 1; i >= 0; i--) { + // Count zeroes + for (var k = 0; i >= 0 && naf[i] === 0; i--) + k++; + if (i >= 0) + k++; + acc = acc.dblp(k); + + if (i < 0) + break; + var z = naf[i]; + assert$2(z !== 0); + if (p.type === 'affine') { + // J +- P + if (z > 0) + acc = acc.mixedAdd(wnd[(z - 1) >> 1]); + else + acc = acc.mixedAdd(wnd[(-z - 1) >> 1].neg()); + } else { + // J +- J + if (z > 0) + acc = acc.add(wnd[(z - 1) >> 1]); + else + acc = acc.add(wnd[(-z - 1) >> 1].neg()); + } + } + return p.type === 'affine' ? acc.toP() : acc; + }; + + BaseCurve.prototype._wnafMulAdd = function _wnafMulAdd(defW, + points, + coeffs, + len, + jacobianResult) { + var wndWidth = this._wnafT1; + var wnd = this._wnafT2; + var naf = this._wnafT3; + + // Fill all arrays + var max = 0; + for (var i = 0; i < len; i++) { + var p = points[i]; + var nafPoints = p._getNAFPoints(defW); + wndWidth[i] = nafPoints.wnd; + wnd[i] = nafPoints.points; + } + + // Comb small window NAFs + for (var i = len - 1; i >= 1; i -= 2) { + var a = i - 1; + var b = i; + if (wndWidth[a] !== 1 || wndWidth[b] !== 1) { + naf[a] = getNAF(coeffs[a], wndWidth[a]); + naf[b] = getNAF(coeffs[b], wndWidth[b]); + max = Math.max(naf[a].length, max); + max = Math.max(naf[b].length, max); + continue; + } + + var comb = [ + points[a], /* 1 */ + null, /* 3 */ + null, /* 5 */ + points[b] /* 7 */ + ]; + + // Try to avoid Projective points, if possible + if (points[a].y.cmp(points[b].y) === 0) { + comb[1] = points[a].add(points[b]); + comb[2] = points[a].toJ().mixedAdd(points[b].neg()); + } else if (points[a].y.cmp(points[b].y.redNeg()) === 0) { + comb[1] = points[a].toJ().mixedAdd(points[b]); + comb[2] = points[a].add(points[b].neg()); + } else { + comb[1] = points[a].toJ().mixedAdd(points[b]); + comb[2] = points[a].toJ().mixedAdd(points[b].neg()); + } + + var index = [ + -3, /* -1 -1 */ + -1, /* -1 0 */ + -5, /* -1 1 */ + -7, /* 0 -1 */ + 0, /* 0 0 */ + 7, /* 0 1 */ + 5, /* 1 -1 */ + 1, /* 1 0 */ + 3 /* 1 1 */ + ]; + + var jsf = getJSF(coeffs[a], coeffs[b]); + max = Math.max(jsf[0].length, max); + naf[a] = new Array(max); + naf[b] = new Array(max); + for (var j = 0; j < max; j++) { + var ja = jsf[0][j] | 0; + var jb = jsf[1][j] | 0; + + naf[a][j] = index[(ja + 1) * 3 + (jb + 1)]; + naf[b][j] = 0; + wnd[a] = comb; + } + } + + var acc = this.jpoint(null, null, null); + var tmp = this._wnafT4; + for (var i = max; i >= 0; i--) { + var k = 0; + + while (i >= 0) { + var zero = true; + for (var j = 0; j < len; j++) { + tmp[j] = naf[j][i] | 0; + if (tmp[j] !== 0) + zero = false; + } + if (!zero) + break; + k++; + i--; + } + if (i >= 0) + k++; + acc = acc.dblp(k); + if (i < 0) + break; + + for (var j = 0; j < len; j++) { + var z = tmp[j]; + var p; + if (z === 0) + continue; + else if (z > 0) + p = wnd[j][(z - 1) >> 1]; + else if (z < 0) + p = wnd[j][(-z - 1) >> 1].neg(); + + if (p.type === 'affine') + acc = acc.mixedAdd(p); + else + acc = acc.add(p); + } + } + // Zeroify references + for (var i = 0; i < len; i++) + wnd[i] = null; + + if (jacobianResult) + return acc; + else + return acc.toP(); + }; + + function BasePoint(curve, type) { + this.curve = curve; + this.type = type; + this.precomputed = null; + } + BaseCurve.BasePoint = BasePoint; + + BasePoint.prototype.eq = function eq(/*other*/) { + throw new Error('Not implemented'); + }; + + BasePoint.prototype.validate = function validate() { + return this.curve.validate(this); + }; + + BaseCurve.prototype.decodePoint = function decodePoint(bytes, enc) { + bytes = utils_1$1.toArray(bytes, enc); + + var len = this.p.byteLength(); + + // uncompressed, hybrid-odd, hybrid-even + if ((bytes[0] === 0x04 || bytes[0] === 0x06 || bytes[0] === 0x07) && + bytes.length - 1 === 2 * len) { + if (bytes[0] === 0x06) + assert$2(bytes[bytes.length - 1] % 2 === 0); + else if (bytes[0] === 0x07) + assert$2(bytes[bytes.length - 1] % 2 === 1); + + var res = this.point(bytes.slice(1, 1 + len), + bytes.slice(1 + len, 1 + 2 * len)); + + return res; + } else if ((bytes[0] === 0x02 || bytes[0] === 0x03) && + bytes.length - 1 === len) { + return this.pointFromX(bytes.slice(1, 1 + len), bytes[0] === 0x03); + } + throw new Error('Unknown point format'); + }; + + BasePoint.prototype.encodeCompressed = function encodeCompressed(enc) { + return this.encode(enc, true); + }; + + BasePoint.prototype._encode = function _encode(compact) { + var len = this.curve.p.byteLength(); + var x = this.getX().toArray('be', len); + + if (compact) + return [ this.getY().isEven() ? 0x02 : 0x03 ].concat(x); + + return [ 0x04 ].concat(x, this.getY().toArray('be', len)) ; + }; + + BasePoint.prototype.encode = function encode(enc, compact) { + return utils_1$1.encode(this._encode(compact), enc); + }; + + BasePoint.prototype.precompute = function precompute(power) { + if (this.precomputed) + return this; + + var precomputed = { + doubles: null, + naf: null, + beta: null + }; + precomputed.naf = this._getNAFPoints(8); + precomputed.doubles = this._getDoubles(4, power); + precomputed.beta = this._getBeta(); + this.precomputed = precomputed; + + return this; + }; + + BasePoint.prototype._hasDoubles = function _hasDoubles(k) { + if (!this.precomputed) + return false; + + var doubles = this.precomputed.doubles; + if (!doubles) + return false; + + return doubles.points.length >= Math.ceil((k.bitLength() + 1) / doubles.step); + }; + + BasePoint.prototype._getDoubles = function _getDoubles(step, power) { + if (this.precomputed && this.precomputed.doubles) + return this.precomputed.doubles; + + var doubles = [ this ]; + var acc = this; + for (var i = 0; i < power; i += step) { + for (var j = 0; j < step; j++) + acc = acc.dbl(); + doubles.push(acc); + } + return { + step: step, + points: doubles + }; + }; + + BasePoint.prototype._getNAFPoints = function _getNAFPoints(wnd) { + if (this.precomputed && this.precomputed.naf) + return this.precomputed.naf; + + var res = [ this ]; + var max = (1 << wnd) - 1; + var dbl = max === 1 ? null : this.dbl(); + for (var i = 1; i < max; i++) + res[i] = res[i - 1].add(dbl); + return { + wnd: wnd, + points: res + }; + }; + + BasePoint.prototype._getBeta = function _getBeta() { + return null; + }; + + BasePoint.prototype.dblp = function dblp(k) { + var r = this; + for (var i = 0; i < k; i++) + r = r.dbl(); + return r; + }; + + var assert$3 = utils_1$1.assert; + + function ShortCurve(conf) { + base.call(this, 'short', conf); + + this.a = new bn(conf.a, 16).toRed(this.red); + this.b = new bn(conf.b, 16).toRed(this.red); + this.tinv = this.two.redInvm(); + + this.zeroA = this.a.fromRed().cmpn(0) === 0; + this.threeA = this.a.fromRed().sub(this.p).cmpn(-3) === 0; + + // If the curve is endomorphic, precalculate beta and lambda + this.endo = this._getEndomorphism(conf); + this._endoWnafT1 = new Array(4); + this._endoWnafT2 = new Array(4); + } + inherits_browser(ShortCurve, base); + var short_1 = ShortCurve; + + ShortCurve.prototype._getEndomorphism = function _getEndomorphism(conf) { + // No efficient endomorphism + if (!this.zeroA || !this.g || !this.n || this.p.modn(3) !== 1) + return; + + // Compute beta and lambda, that lambda * P = (beta * Px; Py) + var beta; + var lambda; + if (conf.beta) { + beta = new bn(conf.beta, 16).toRed(this.red); + } else { + var betas = this._getEndoRoots(this.p); + // Choose the smallest beta + beta = betas[0].cmp(betas[1]) < 0 ? betas[0] : betas[1]; + beta = beta.toRed(this.red); + } + if (conf.lambda) { + lambda = new bn(conf.lambda, 16); + } else { + // Choose the lambda that is matching selected beta + var lambdas = this._getEndoRoots(this.n); + if (this.g.mul(lambdas[0]).x.cmp(this.g.x.redMul(beta)) === 0) { + lambda = lambdas[0]; + } else { + lambda = lambdas[1]; + assert$3(this.g.mul(lambda).x.cmp(this.g.x.redMul(beta)) === 0); + } + } + + // Get basis vectors, used for balanced length-two representation + var basis; + if (conf.basis) { + basis = conf.basis.map(function(vec) { + return { + a: new bn(vec.a, 16), + b: new bn(vec.b, 16) + }; + }); + } else { + basis = this._getEndoBasis(lambda); + } + + return { + beta: beta, + lambda: lambda, + basis: basis + }; + }; + + ShortCurve.prototype._getEndoRoots = function _getEndoRoots(num) { + // Find roots of for x^2 + x + 1 in F + // Root = (-1 +- Sqrt(-3)) / 2 + // + var red = num === this.p ? this.red : bn.mont(num); + var tinv = new bn(2).toRed(red).redInvm(); + var ntinv = tinv.redNeg(); + + var s = new bn(3).toRed(red).redNeg().redSqrt().redMul(tinv); + + var l1 = ntinv.redAdd(s).fromRed(); + var l2 = ntinv.redSub(s).fromRed(); + return [ l1, l2 ]; + }; + + ShortCurve.prototype._getEndoBasis = function _getEndoBasis(lambda) { + // aprxSqrt >= sqrt(this.n) + var aprxSqrt = this.n.ushrn(Math.floor(this.n.bitLength() / 2)); + + // 3.74 + // Run EGCD, until r(L + 1) < aprxSqrt + var u = lambda; + var v = this.n.clone(); + var x1 = new bn(1); + var y1 = new bn(0); + var x2 = new bn(0); + var y2 = new bn(1); + + // NOTE: all vectors are roots of: a + b * lambda = 0 (mod n) + var a0; + var b0; + // First vector + var a1; + var b1; + // Second vector + var a2; + var b2; + + var prevR; + var i = 0; + var r; + var x; + while (u.cmpn(0) !== 0) { + var q = v.div(u); + r = v.sub(q.mul(u)); + x = x2.sub(q.mul(x1)); + var y = y2.sub(q.mul(y1)); + + if (!a1 && r.cmp(aprxSqrt) < 0) { + a0 = prevR.neg(); + b0 = x1; + a1 = r.neg(); + b1 = x; + } else if (a1 && ++i === 2) { + break; + } + prevR = r; + + v = u; + u = r; + x2 = x1; + x1 = x; + y2 = y1; + y1 = y; + } + a2 = r.neg(); + b2 = x; + + var len1 = a1.sqr().add(b1.sqr()); + var len2 = a2.sqr().add(b2.sqr()); + if (len2.cmp(len1) >= 0) { + a2 = a0; + b2 = b0; + } + + // Normalize signs + if (a1.negative) { + a1 = a1.neg(); + b1 = b1.neg(); + } + if (a2.negative) { + a2 = a2.neg(); + b2 = b2.neg(); + } + + return [ + { a: a1, b: b1 }, + { a: a2, b: b2 } + ]; + }; + + ShortCurve.prototype._endoSplit = function _endoSplit(k) { + var basis = this.endo.basis; + var v1 = basis[0]; + var v2 = basis[1]; + + var c1 = v2.b.mul(k).divRound(this.n); + var c2 = v1.b.neg().mul(k).divRound(this.n); + + var p1 = c1.mul(v1.a); + var p2 = c2.mul(v2.a); + var q1 = c1.mul(v1.b); + var q2 = c2.mul(v2.b); + + // Calculate answer + var k1 = k.sub(p1).sub(p2); + var k2 = q1.add(q2).neg(); + return { k1: k1, k2: k2 }; + }; + + ShortCurve.prototype.pointFromX = function pointFromX(x, odd) { + x = new bn(x, 16); + if (!x.red) + x = x.toRed(this.red); + + var y2 = x.redSqr().redMul(x).redIAdd(x.redMul(this.a)).redIAdd(this.b); + var y = y2.redSqrt(); + if (y.redSqr().redSub(y2).cmp(this.zero) !== 0) + throw new Error('invalid point'); + + // XXX Is there any way to tell if the number is odd without converting it + // to non-red form? + var isOdd = y.fromRed().isOdd(); + if (odd && !isOdd || !odd && isOdd) + y = y.redNeg(); + + return this.point(x, y); + }; + + ShortCurve.prototype.validate = function validate(point) { + if (point.inf) + return true; + + var x = point.x; + var y = point.y; + + var ax = this.a.redMul(x); + var rhs = x.redSqr().redMul(x).redIAdd(ax).redIAdd(this.b); + return y.redSqr().redISub(rhs).cmpn(0) === 0; + }; + + ShortCurve.prototype._endoWnafMulAdd = + function _endoWnafMulAdd(points, coeffs, jacobianResult) { + var npoints = this._endoWnafT1; + var ncoeffs = this._endoWnafT2; + for (var i = 0; i < points.length; i++) { + var split = this._endoSplit(coeffs[i]); + var p = points[i]; + var beta = p._getBeta(); + + if (split.k1.negative) { + split.k1.ineg(); + p = p.neg(true); + } + if (split.k2.negative) { + split.k2.ineg(); + beta = beta.neg(true); + } + + npoints[i * 2] = p; + npoints[i * 2 + 1] = beta; + ncoeffs[i * 2] = split.k1; + ncoeffs[i * 2 + 1] = split.k2; + } + var res = this._wnafMulAdd(1, npoints, ncoeffs, i * 2, jacobianResult); + + // Clean-up references to points and coefficients + for (var j = 0; j < i * 2; j++) { + npoints[j] = null; + ncoeffs[j] = null; + } + return res; + }; + + function Point(curve, x, y, isRed) { + base.BasePoint.call(this, curve, 'affine'); + if (x === null && y === null) { + this.x = null; + this.y = null; + this.inf = true; + } else { + this.x = new bn(x, 16); + this.y = new bn(y, 16); + // Force redgomery representation when loading from JSON + if (isRed) { + this.x.forceRed(this.curve.red); + this.y.forceRed(this.curve.red); + } + if (!this.x.red) + this.x = this.x.toRed(this.curve.red); + if (!this.y.red) + this.y = this.y.toRed(this.curve.red); + this.inf = false; + } + } + inherits_browser(Point, base.BasePoint); + + ShortCurve.prototype.point = function point(x, y, isRed) { + return new Point(this, x, y, isRed); + }; + + ShortCurve.prototype.pointFromJSON = function pointFromJSON(obj, red) { + return Point.fromJSON(this, obj, red); + }; + + Point.prototype._getBeta = function _getBeta() { + if (!this.curve.endo) + return; + + var pre = this.precomputed; + if (pre && pre.beta) + return pre.beta; + + var beta = this.curve.point(this.x.redMul(this.curve.endo.beta), this.y); + if (pre) { + var curve = this.curve; + var endoMul = function(p) { + return curve.point(p.x.redMul(curve.endo.beta), p.y); + }; + pre.beta = beta; + beta.precomputed = { + beta: null, + naf: pre.naf && { + wnd: pre.naf.wnd, + points: pre.naf.points.map(endoMul) + }, + doubles: pre.doubles && { + step: pre.doubles.step, + points: pre.doubles.points.map(endoMul) + } + }; + } + return beta; + }; + + Point.prototype.toJSON = function toJSON() { + if (!this.precomputed) + return [ this.x, this.y ]; + + return [ this.x, this.y, this.precomputed && { + doubles: this.precomputed.doubles && { + step: this.precomputed.doubles.step, + points: this.precomputed.doubles.points.slice(1) + }, + naf: this.precomputed.naf && { + wnd: this.precomputed.naf.wnd, + points: this.precomputed.naf.points.slice(1) + } + } ]; + }; + + Point.fromJSON = function fromJSON(curve, obj, red) { + if (typeof obj === 'string') + obj = JSON.parse(obj); + var res = curve.point(obj[0], obj[1], red); + if (!obj[2]) + return res; + + function obj2point(obj) { + return curve.point(obj[0], obj[1], red); + } + + var pre = obj[2]; + res.precomputed = { + beta: null, + doubles: pre.doubles && { + step: pre.doubles.step, + points: [ res ].concat(pre.doubles.points.map(obj2point)) + }, + naf: pre.naf && { + wnd: pre.naf.wnd, + points: [ res ].concat(pre.naf.points.map(obj2point)) + } + }; + return res; + }; + + Point.prototype.inspect = function inspect() { + if (this.isInfinity()) + return ''; + return ''; + }; + + Point.prototype.isInfinity = function isInfinity() { + return this.inf; + }; + + Point.prototype.add = function add(p) { + // O + P = P + if (this.inf) + return p; + + // P + O = P + if (p.inf) + return this; + + // P + P = 2P + if (this.eq(p)) + return this.dbl(); + + // P + (-P) = O + if (this.neg().eq(p)) + return this.curve.point(null, null); + + // P + Q = O + if (this.x.cmp(p.x) === 0) + return this.curve.point(null, null); + + var c = this.y.redSub(p.y); + if (c.cmpn(0) !== 0) + c = c.redMul(this.x.redSub(p.x).redInvm()); + var nx = c.redSqr().redISub(this.x).redISub(p.x); + var ny = c.redMul(this.x.redSub(nx)).redISub(this.y); + return this.curve.point(nx, ny); + }; + + Point.prototype.dbl = function dbl() { + if (this.inf) + return this; + + // 2P = O + var ys1 = this.y.redAdd(this.y); + if (ys1.cmpn(0) === 0) + return this.curve.point(null, null); + + var a = this.curve.a; + + var x2 = this.x.redSqr(); + var dyinv = ys1.redInvm(); + var c = x2.redAdd(x2).redIAdd(x2).redIAdd(a).redMul(dyinv); + + var nx = c.redSqr().redISub(this.x.redAdd(this.x)); + var ny = c.redMul(this.x.redSub(nx)).redISub(this.y); + return this.curve.point(nx, ny); + }; + + Point.prototype.getX = function getX() { + return this.x.fromRed(); + }; + + Point.prototype.getY = function getY() { + return this.y.fromRed(); + }; + + Point.prototype.mul = function mul(k) { + k = new bn(k, 16); + if (this.isInfinity()) + return this; + else if (this._hasDoubles(k)) + return this.curve._fixedNafMul(this, k); + else if (this.curve.endo) + return this.curve._endoWnafMulAdd([ this ], [ k ]); + else + return this.curve._wnafMul(this, k); + }; + + Point.prototype.mulAdd = function mulAdd(k1, p2, k2) { + var points = [ this, p2 ]; + var coeffs = [ k1, k2 ]; + if (this.curve.endo) + return this.curve._endoWnafMulAdd(points, coeffs); + else + return this.curve._wnafMulAdd(1, points, coeffs, 2); + }; + + Point.prototype.jmulAdd = function jmulAdd(k1, p2, k2) { + var points = [ this, p2 ]; + var coeffs = [ k1, k2 ]; + if (this.curve.endo) + return this.curve._endoWnafMulAdd(points, coeffs, true); + else + return this.curve._wnafMulAdd(1, points, coeffs, 2, true); + }; + + Point.prototype.eq = function eq(p) { + return this === p || + this.inf === p.inf && + (this.inf || this.x.cmp(p.x) === 0 && this.y.cmp(p.y) === 0); + }; + + Point.prototype.neg = function neg(_precompute) { + if (this.inf) + return this; + + var res = this.curve.point(this.x, this.y.redNeg()); + if (_precompute && this.precomputed) { + var pre = this.precomputed; + var negate = function(p) { + return p.neg(); + }; + res.precomputed = { + naf: pre.naf && { + wnd: pre.naf.wnd, + points: pre.naf.points.map(negate) + }, + doubles: pre.doubles && { + step: pre.doubles.step, + points: pre.doubles.points.map(negate) + } + }; + } + return res; + }; + + Point.prototype.toJ = function toJ() { + if (this.inf) + return this.curve.jpoint(null, null, null); + + var res = this.curve.jpoint(this.x, this.y, this.curve.one); + return res; + }; + + function JPoint(curve, x, y, z) { + base.BasePoint.call(this, curve, 'jacobian'); + if (x === null && y === null && z === null) { + this.x = this.curve.one; + this.y = this.curve.one; + this.z = new bn(0); + } else { + this.x = new bn(x, 16); + this.y = new bn(y, 16); + this.z = new bn(z, 16); + } + if (!this.x.red) + this.x = this.x.toRed(this.curve.red); + if (!this.y.red) + this.y = this.y.toRed(this.curve.red); + if (!this.z.red) + this.z = this.z.toRed(this.curve.red); + + this.zOne = this.z === this.curve.one; + } + inherits_browser(JPoint, base.BasePoint); + + ShortCurve.prototype.jpoint = function jpoint(x, y, z) { + return new JPoint(this, x, y, z); + }; + + JPoint.prototype.toP = function toP() { + if (this.isInfinity()) + return this.curve.point(null, null); + + var zinv = this.z.redInvm(); + var zinv2 = zinv.redSqr(); + var ax = this.x.redMul(zinv2); + var ay = this.y.redMul(zinv2).redMul(zinv); + + return this.curve.point(ax, ay); + }; + + JPoint.prototype.neg = function neg() { + return this.curve.jpoint(this.x, this.y.redNeg(), this.z); + }; + + JPoint.prototype.add = function add(p) { + // O + P = P + if (this.isInfinity()) + return p; + + // P + O = P + if (p.isInfinity()) + return this; + + // 12M + 4S + 7A + var pz2 = p.z.redSqr(); + var z2 = this.z.redSqr(); + var u1 = this.x.redMul(pz2); + var u2 = p.x.redMul(z2); + var s1 = this.y.redMul(pz2.redMul(p.z)); + var s2 = p.y.redMul(z2.redMul(this.z)); + + var h = u1.redSub(u2); + var r = s1.redSub(s2); + if (h.cmpn(0) === 0) { + if (r.cmpn(0) !== 0) + return this.curve.jpoint(null, null, null); + else + return this.dbl(); + } + + var h2 = h.redSqr(); + var h3 = h2.redMul(h); + var v = u1.redMul(h2); + + var nx = r.redSqr().redIAdd(h3).redISub(v).redISub(v); + var ny = r.redMul(v.redISub(nx)).redISub(s1.redMul(h3)); + var nz = this.z.redMul(p.z).redMul(h); + + return this.curve.jpoint(nx, ny, nz); + }; + + JPoint.prototype.mixedAdd = function mixedAdd(p) { + // O + P = P + if (this.isInfinity()) + return p.toJ(); + + // P + O = P + if (p.isInfinity()) + return this; + + // 8M + 3S + 7A + var z2 = this.z.redSqr(); + var u1 = this.x; + var u2 = p.x.redMul(z2); + var s1 = this.y; + var s2 = p.y.redMul(z2).redMul(this.z); + + var h = u1.redSub(u2); + var r = s1.redSub(s2); + if (h.cmpn(0) === 0) { + if (r.cmpn(0) !== 0) + return this.curve.jpoint(null, null, null); + else + return this.dbl(); + } + + var h2 = h.redSqr(); + var h3 = h2.redMul(h); + var v = u1.redMul(h2); + + var nx = r.redSqr().redIAdd(h3).redISub(v).redISub(v); + var ny = r.redMul(v.redISub(nx)).redISub(s1.redMul(h3)); + var nz = this.z.redMul(h); + + return this.curve.jpoint(nx, ny, nz); + }; + + JPoint.prototype.dblp = function dblp(pow) { + if (pow === 0) + return this; + if (this.isInfinity()) + return this; + if (!pow) + return this.dbl(); + + if (this.curve.zeroA || this.curve.threeA) { + var r = this; + for (var i = 0; i < pow; i++) + r = r.dbl(); + return r; + } + + // 1M + 2S + 1A + N * (4S + 5M + 8A) + // N = 1 => 6M + 6S + 9A + var a = this.curve.a; + var tinv = this.curve.tinv; + + var jx = this.x; + var jy = this.y; + var jz = this.z; + var jz4 = jz.redSqr().redSqr(); + + // Reuse results + var jyd = jy.redAdd(jy); + for (var i = 0; i < pow; i++) { + var jx2 = jx.redSqr(); + var jyd2 = jyd.redSqr(); + var jyd4 = jyd2.redSqr(); + var c = jx2.redAdd(jx2).redIAdd(jx2).redIAdd(a.redMul(jz4)); + + var t1 = jx.redMul(jyd2); + var nx = c.redSqr().redISub(t1.redAdd(t1)); + var t2 = t1.redISub(nx); + var dny = c.redMul(t2); + dny = dny.redIAdd(dny).redISub(jyd4); + var nz = jyd.redMul(jz); + if (i + 1 < pow) + jz4 = jz4.redMul(jyd4); + + jx = nx; + jz = nz; + jyd = dny; + } + + return this.curve.jpoint(jx, jyd.redMul(tinv), jz); + }; + + JPoint.prototype.dbl = function dbl() { + if (this.isInfinity()) + return this; + + if (this.curve.zeroA) + return this._zeroDbl(); + else if (this.curve.threeA) + return this._threeDbl(); + else + return this._dbl(); + }; + + JPoint.prototype._zeroDbl = function _zeroDbl() { + var nx; + var ny; + var nz; + // Z = 1 + if (this.zOne) { + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html + // #doubling-mdbl-2007-bl + // 1M + 5S + 14A + + // XX = X1^2 + var xx = this.x.redSqr(); + // YY = Y1^2 + var yy = this.y.redSqr(); + // YYYY = YY^2 + var yyyy = yy.redSqr(); + // S = 2 * ((X1 + YY)^2 - XX - YYYY) + var s = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); + s = s.redIAdd(s); + // M = 3 * XX + a; a = 0 + var m = xx.redAdd(xx).redIAdd(xx); + // T = M ^ 2 - 2*S + var t = m.redSqr().redISub(s).redISub(s); + + // 8 * YYYY + var yyyy8 = yyyy.redIAdd(yyyy); + yyyy8 = yyyy8.redIAdd(yyyy8); + yyyy8 = yyyy8.redIAdd(yyyy8); + + // X3 = T + nx = t; + // Y3 = M * (S - T) - 8 * YYYY + ny = m.redMul(s.redISub(t)).redISub(yyyy8); + // Z3 = 2*Y1 + nz = this.y.redAdd(this.y); + } else { + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html + // #doubling-dbl-2009-l + // 2M + 5S + 13A + + // A = X1^2 + var a = this.x.redSqr(); + // B = Y1^2 + var b = this.y.redSqr(); + // C = B^2 + var c = b.redSqr(); + // D = 2 * ((X1 + B)^2 - A - C) + var d = this.x.redAdd(b).redSqr().redISub(a).redISub(c); + d = d.redIAdd(d); + // E = 3 * A + var e = a.redAdd(a).redIAdd(a); + // F = E^2 + var f = e.redSqr(); + + // 8 * C + var c8 = c.redIAdd(c); + c8 = c8.redIAdd(c8); + c8 = c8.redIAdd(c8); + + // X3 = F - 2 * D + nx = f.redISub(d).redISub(d); + // Y3 = E * (D - X3) - 8 * C + ny = e.redMul(d.redISub(nx)).redISub(c8); + // Z3 = 2 * Y1 * Z1 + nz = this.y.redMul(this.z); + nz = nz.redIAdd(nz); + } + + return this.curve.jpoint(nx, ny, nz); + }; + + JPoint.prototype._threeDbl = function _threeDbl() { + var nx; + var ny; + var nz; + // Z = 1 + if (this.zOne) { + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-3.html + // #doubling-mdbl-2007-bl + // 1M + 5S + 15A + + // XX = X1^2 + var xx = this.x.redSqr(); + // YY = Y1^2 + var yy = this.y.redSqr(); + // YYYY = YY^2 + var yyyy = yy.redSqr(); + // S = 2 * ((X1 + YY)^2 - XX - YYYY) + var s = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); + s = s.redIAdd(s); + // M = 3 * XX + a + var m = xx.redAdd(xx).redIAdd(xx).redIAdd(this.curve.a); + // T = M^2 - 2 * S + var t = m.redSqr().redISub(s).redISub(s); + // X3 = T + nx = t; + // Y3 = M * (S - T) - 8 * YYYY + var yyyy8 = yyyy.redIAdd(yyyy); + yyyy8 = yyyy8.redIAdd(yyyy8); + yyyy8 = yyyy8.redIAdd(yyyy8); + ny = m.redMul(s.redISub(t)).redISub(yyyy8); + // Z3 = 2 * Y1 + nz = this.y.redAdd(this.y); + } else { + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-3.html#doubling-dbl-2001-b + // 3M + 5S + + // delta = Z1^2 + var delta = this.z.redSqr(); + // gamma = Y1^2 + var gamma = this.y.redSqr(); + // beta = X1 * gamma + var beta = this.x.redMul(gamma); + // alpha = 3 * (X1 - delta) * (X1 + delta) + var alpha = this.x.redSub(delta).redMul(this.x.redAdd(delta)); + alpha = alpha.redAdd(alpha).redIAdd(alpha); + // X3 = alpha^2 - 8 * beta + var beta4 = beta.redIAdd(beta); + beta4 = beta4.redIAdd(beta4); + var beta8 = beta4.redAdd(beta4); + nx = alpha.redSqr().redISub(beta8); + // Z3 = (Y1 + Z1)^2 - gamma - delta + nz = this.y.redAdd(this.z).redSqr().redISub(gamma).redISub(delta); + // Y3 = alpha * (4 * beta - X3) - 8 * gamma^2 + var ggamma8 = gamma.redSqr(); + ggamma8 = ggamma8.redIAdd(ggamma8); + ggamma8 = ggamma8.redIAdd(ggamma8); + ggamma8 = ggamma8.redIAdd(ggamma8); + ny = alpha.redMul(beta4.redISub(nx)).redISub(ggamma8); + } + + return this.curve.jpoint(nx, ny, nz); + }; + + JPoint.prototype._dbl = function _dbl() { + var a = this.curve.a; + + // 4M + 6S + 10A + var jx = this.x; + var jy = this.y; + var jz = this.z; + var jz4 = jz.redSqr().redSqr(); + + var jx2 = jx.redSqr(); + var jy2 = jy.redSqr(); + + var c = jx2.redAdd(jx2).redIAdd(jx2).redIAdd(a.redMul(jz4)); + + var jxd4 = jx.redAdd(jx); + jxd4 = jxd4.redIAdd(jxd4); + var t1 = jxd4.redMul(jy2); + var nx = c.redSqr().redISub(t1.redAdd(t1)); + var t2 = t1.redISub(nx); + + var jyd8 = jy2.redSqr(); + jyd8 = jyd8.redIAdd(jyd8); + jyd8 = jyd8.redIAdd(jyd8); + jyd8 = jyd8.redIAdd(jyd8); + var ny = c.redMul(t2).redISub(jyd8); + var nz = jy.redAdd(jy).redMul(jz); + + return this.curve.jpoint(nx, ny, nz); + }; + + JPoint.prototype.trpl = function trpl() { + if (!this.curve.zeroA) + return this.dbl().add(this); + + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html#tripling-tpl-2007-bl + // 5M + 10S + ... + + // XX = X1^2 + var xx = this.x.redSqr(); + // YY = Y1^2 + var yy = this.y.redSqr(); + // ZZ = Z1^2 + var zz = this.z.redSqr(); + // YYYY = YY^2 + var yyyy = yy.redSqr(); + // M = 3 * XX + a * ZZ2; a = 0 + var m = xx.redAdd(xx).redIAdd(xx); + // MM = M^2 + var mm = m.redSqr(); + // E = 6 * ((X1 + YY)^2 - XX - YYYY) - MM + var e = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); + e = e.redIAdd(e); + e = e.redAdd(e).redIAdd(e); + e = e.redISub(mm); + // EE = E^2 + var ee = e.redSqr(); + // T = 16*YYYY + var t = yyyy.redIAdd(yyyy); + t = t.redIAdd(t); + t = t.redIAdd(t); + t = t.redIAdd(t); + // U = (M + E)^2 - MM - EE - T + var u = m.redIAdd(e).redSqr().redISub(mm).redISub(ee).redISub(t); + // X3 = 4 * (X1 * EE - 4 * YY * U) + var yyu4 = yy.redMul(u); + yyu4 = yyu4.redIAdd(yyu4); + yyu4 = yyu4.redIAdd(yyu4); + var nx = this.x.redMul(ee).redISub(yyu4); + nx = nx.redIAdd(nx); + nx = nx.redIAdd(nx); + // Y3 = 8 * Y1 * (U * (T - U) - E * EE) + var ny = this.y.redMul(u.redMul(t.redISub(u)).redISub(e.redMul(ee))); + ny = ny.redIAdd(ny); + ny = ny.redIAdd(ny); + ny = ny.redIAdd(ny); + // Z3 = (Z1 + E)^2 - ZZ - EE + var nz = this.z.redAdd(e).redSqr().redISub(zz).redISub(ee); + + return this.curve.jpoint(nx, ny, nz); + }; + + JPoint.prototype.mul = function mul(k, kbase) { + k = new bn(k, kbase); + + return this.curve._wnafMul(this, k); + }; + + JPoint.prototype.eq = function eq(p) { + if (p.type === 'affine') + return this.eq(p.toJ()); + + if (this === p) + return true; + + // x1 * z2^2 == x2 * z1^2 + var z2 = this.z.redSqr(); + var pz2 = p.z.redSqr(); + if (this.x.redMul(pz2).redISub(p.x.redMul(z2)).cmpn(0) !== 0) + return false; + + // y1 * z2^3 == y2 * z1^3 + var z3 = z2.redMul(this.z); + var pz3 = pz2.redMul(p.z); + return this.y.redMul(pz3).redISub(p.y.redMul(z3)).cmpn(0) === 0; + }; + + JPoint.prototype.eqXToP = function eqXToP(x) { + var zs = this.z.redSqr(); + var rx = x.toRed(this.curve.red).redMul(zs); + if (this.x.cmp(rx) === 0) + return true; + + var xc = x.clone(); + var t = this.curve.redN.redMul(zs); + for (;;) { + xc.iadd(this.curve.n); + if (xc.cmp(this.curve.p) >= 0) + return false; + + rx.redIAdd(t); + if (this.x.cmp(rx) === 0) + return true; + } + }; + + JPoint.prototype.inspect = function inspect() { + if (this.isInfinity()) + return ''; + return ''; + }; + + JPoint.prototype.isInfinity = function isInfinity() { + // XXX This code assumes that zero is always zero in red + return this.z.cmpn(0) === 0; + }; + + function MontCurve(conf) { + base.call(this, 'mont', conf); + + this.a = new bn(conf.a, 16).toRed(this.red); + this.b = new bn(conf.b, 16).toRed(this.red); + this.i4 = new bn(4).toRed(this.red).redInvm(); + this.two = new bn(2).toRed(this.red); + // Note: this implementation is according to the original paper + // by P. Montgomery, NOT the one by D. J. Bernstein. + this.a24 = this.i4.redMul(this.a.redAdd(this.two)); + } + inherits_browser(MontCurve, base); + var mont = MontCurve; + + MontCurve.prototype.validate = function validate(point) { + var x = point.normalize().x; + var x2 = x.redSqr(); + var rhs = x2.redMul(x).redAdd(x2.redMul(this.a)).redAdd(x); + var y = rhs.redSqrt(); + + return y.redSqr().cmp(rhs) === 0; + }; + + function Point$1(curve, x, z) { + base.BasePoint.call(this, curve, 'projective'); + if (x === null && z === null) { + this.x = this.curve.one; + this.z = this.curve.zero; + } else { + this.x = new bn(x, 16); + this.z = new bn(z, 16); + if (!this.x.red) + this.x = this.x.toRed(this.curve.red); + if (!this.z.red) + this.z = this.z.toRed(this.curve.red); + } + } + inherits_browser(Point$1, base.BasePoint); + + MontCurve.prototype.decodePoint = function decodePoint(bytes, enc) { + var bytes = utils_1$1.toArray(bytes, enc); + + // TODO Curve448 + // Montgomery curve points must be represented in the compressed format + // https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-02#appendix-B + if (bytes.length === 33 && bytes[0] === 0x40) + bytes = bytes.slice(1, 33).reverse(); // point must be little-endian + if (bytes.length !== 32) + throw new Error('Unknown point compression format'); + return this.point(bytes, 1); + }; + + MontCurve.prototype.point = function point(x, z) { + return new Point$1(this, x, z); + }; + + MontCurve.prototype.pointFromJSON = function pointFromJSON(obj) { + return Point$1.fromJSON(this, obj); + }; + + Point$1.prototype.precompute = function precompute() { + // No-op + }; + + Point$1.prototype._encode = function _encode(compact) { + var len = this.curve.p.byteLength(); + + // Note: the output should always be little-endian + // https://tools.ietf.org/html/draft-ietf-openpgp-rfc4880bis-02#appendix-B + if (compact) { + return [ 0x40 ].concat(this.getX().toArray('le', len)); + } else { + return this.getX().toArray('be', len); + } + }; + + Point$1.fromJSON = function fromJSON(curve, obj) { + return new Point$1(curve, obj[0], obj[1] || curve.one); + }; + + Point$1.prototype.inspect = function inspect() { + if (this.isInfinity()) + return ''; + return ''; + }; + + Point$1.prototype.isInfinity = function isInfinity() { + // XXX This code assumes that zero is always zero in red + return this.z.cmpn(0) === 0; + }; + + Point$1.prototype.dbl = function dbl() { + // http://hyperelliptic.org/EFD/g1p/auto-montgom-xz.html#doubling-dbl-1987-m-3 + // 2M + 2S + 4A + + // A = X1 + Z1 + var a = this.x.redAdd(this.z); + // AA = A^2 + var aa = a.redSqr(); + // B = X1 - Z1 + var b = this.x.redSub(this.z); + // BB = B^2 + var bb = b.redSqr(); + // C = AA - BB + var c = aa.redSub(bb); + // X3 = AA * BB + var nx = aa.redMul(bb); + // Z3 = C * (BB + A24 * C) + var nz = c.redMul(bb.redAdd(this.curve.a24.redMul(c))); + return this.curve.point(nx, nz); + }; + + Point$1.prototype.add = function add() { + throw new Error('Not supported on Montgomery curve'); + }; + + Point$1.prototype.diffAdd = function diffAdd(p, diff) { + // http://hyperelliptic.org/EFD/g1p/auto-montgom-xz.html#diffadd-dadd-1987-m-3 + // 4M + 2S + 6A + + // A = X2 + Z2 + var a = this.x.redAdd(this.z); + // B = X2 - Z2 + var b = this.x.redSub(this.z); + // C = X3 + Z3 + var c = p.x.redAdd(p.z); + // D = X3 - Z3 + var d = p.x.redSub(p.z); + // DA = D * A + var da = d.redMul(a); + // CB = C * B + var cb = c.redMul(b); + // X5 = Z1 * (DA + CB)^2 + var nx = diff.z.redMul(da.redAdd(cb).redSqr()); + // Z5 = X1 * (DA - CB)^2 + var nz = diff.x.redMul(da.redISub(cb).redSqr()); + return this.curve.point(nx, nz); + }; + + Point$1.prototype.mul = function mul(k) { + k = new bn(k, 16); + + var t = k.clone(); + var a = this; // (N / 2) * Q + Q + var b = this.curve.point(null, null); // (N / 2) * Q + var c = this; // Q + + for (var bits = []; t.cmpn(0) !== 0; t.iushrn(1)) + bits.push(t.andln(1)); + + for (var i = bits.length - 1; i >= 0; i--) { + if (bits[i] === 0) { + // N * Q + Q = ((N / 2) * Q + Q)) + (N / 2) * Q + a = a.diffAdd(b, c); + // N * Q = 2 * ((N / 2) * Q + Q)) + b = b.dbl(); + } else { + // N * Q = ((N / 2) * Q + Q) + ((N / 2) * Q) + b = a.diffAdd(b, c); + // N * Q + Q = 2 * ((N / 2) * Q + Q) + a = a.dbl(); + } + } + return b; + }; + + Point$1.prototype.mulAdd = function mulAdd() { + throw new Error('Not supported on Montgomery curve'); + }; + + Point$1.prototype.jumlAdd = function jumlAdd() { + throw new Error('Not supported on Montgomery curve'); + }; + + Point$1.prototype.eq = function eq(other) { + return this.getX().cmp(other.getX()) === 0; + }; + + Point$1.prototype.normalize = function normalize() { + this.x = this.x.redMul(this.z.redInvm()); + this.z = this.curve.one; + return this; + }; + + Point$1.prototype.getX = function getX() { + // Normalize coordinates + this.normalize(); + + return this.x.fromRed(); + }; + + var assert$4 = utils_1$1.assert; + + function EdwardsCurve(conf) { + // NOTE: Important as we are creating point in Base.call() + this.twisted = (conf.a | 0) !== 1; + this.mOneA = this.twisted && (conf.a | 0) === -1; + this.extended = this.mOneA; + + base.call(this, 'edwards', conf); + + this.a = new bn(conf.a, 16).umod(this.red.m); + this.a = this.a.toRed(this.red); + this.c = new bn(conf.c, 16).toRed(this.red); + this.c2 = this.c.redSqr(); + this.d = new bn(conf.d, 16).toRed(this.red); + this.dd = this.d.redAdd(this.d); + + assert$4(!this.twisted || this.c.fromRed().cmpn(1) === 0); + this.oneC = (conf.c | 0) === 1; + } + inherits_browser(EdwardsCurve, base); + var edwards = EdwardsCurve; + + EdwardsCurve.prototype._mulA = function _mulA(num) { + if (this.mOneA) + return num.redNeg(); + else + return this.a.redMul(num); + }; + + EdwardsCurve.prototype._mulC = function _mulC(num) { + if (this.oneC) + return num; + else + return this.c.redMul(num); + }; + + // Just for compatibility with Short curve + EdwardsCurve.prototype.jpoint = function jpoint(x, y, z, t) { + return this.point(x, y, z, t); + }; + + EdwardsCurve.prototype.pointFromX = function pointFromX(x, odd) { + x = new bn(x, 16); + if (!x.red) + x = x.toRed(this.red); + + var x2 = x.redSqr(); + var rhs = this.c2.redSub(this.a.redMul(x2)); + var lhs = this.one.redSub(this.c2.redMul(this.d).redMul(x2)); + + var y2 = rhs.redMul(lhs.redInvm()); + var y = y2.redSqrt(); + if (y.redSqr().redSub(y2).cmp(this.zero) !== 0) + throw new Error('invalid point'); + + var isOdd = y.fromRed().isOdd(); + if (odd && !isOdd || !odd && isOdd) + y = y.redNeg(); + + return this.point(x, y); + }; + + EdwardsCurve.prototype.pointFromY = function pointFromY(y, odd) { + y = new bn(y, 16); + if (!y.red) + y = y.toRed(this.red); + + // x^2 = (y^2 - c^2) / (c^2 d y^2 - a) + var y2 = y.redSqr(); + var lhs = y2.redSub(this.c2); + var rhs = y2.redMul(this.d).redMul(this.c2).redSub(this.a); + var x2 = lhs.redMul(rhs.redInvm()); + + if (x2.cmp(this.zero) === 0) { + if (odd) + throw new Error('invalid point'); + else + return this.point(this.zero, y); + } + + var x = x2.redSqrt(); + if (x.redSqr().redSub(x2).cmp(this.zero) !== 0) + throw new Error('invalid point'); + + if (x.fromRed().isOdd() !== odd) + x = x.redNeg(); + + return this.point(x, y); + }; + + EdwardsCurve.prototype.validate = function validate(point) { + if (point.isInfinity()) + return true; + + // Curve: A * X^2 + Y^2 = C^2 * (1 + D * X^2 * Y^2) + point.normalize(); + + var x2 = point.x.redSqr(); + var y2 = point.y.redSqr(); + var lhs = x2.redMul(this.a).redAdd(y2); + var rhs = this.c2.redMul(this.one.redAdd(this.d.redMul(x2).redMul(y2))); + + return lhs.cmp(rhs) === 0; + }; + + function Point$2(curve, x, y, z, t) { + base.BasePoint.call(this, curve, 'projective'); + if (x === null && y === null && z === null) { + this.x = this.curve.zero; + this.y = this.curve.one; + this.z = this.curve.one; + this.t = this.curve.zero; + this.zOne = true; + } else { + this.x = new bn(x, 16); + this.y = new bn(y, 16); + this.z = z ? new bn(z, 16) : this.curve.one; + this.t = t && new bn(t, 16); + if (!this.x.red) + this.x = this.x.toRed(this.curve.red); + if (!this.y.red) + this.y = this.y.toRed(this.curve.red); + if (!this.z.red) + this.z = this.z.toRed(this.curve.red); + if (this.t && !this.t.red) + this.t = this.t.toRed(this.curve.red); + this.zOne = this.z === this.curve.one; + + // Use extended coordinates + if (this.curve.extended && !this.t) { + this.t = this.x.redMul(this.y); + if (!this.zOne) + this.t = this.t.redMul(this.z.redInvm()); + } + } + } + inherits_browser(Point$2, base.BasePoint); + + EdwardsCurve.prototype.pointFromJSON = function pointFromJSON(obj) { + return Point$2.fromJSON(this, obj); + }; + + EdwardsCurve.prototype.point = function point(x, y, z, t) { + return new Point$2(this, x, y, z, t); + }; + + Point$2.fromJSON = function fromJSON(curve, obj) { + return new Point$2(curve, obj[0], obj[1], obj[2]); + }; + + Point$2.prototype.inspect = function inspect() { + if (this.isInfinity()) + return ''; + return ''; + }; + + Point$2.prototype.isInfinity = function isInfinity() { + // XXX This code assumes that zero is always zero in red + return this.x.cmpn(0) === 0 && + (this.y.cmp(this.z) === 0 || + (this.zOne && this.y.cmp(this.curve.c) === 0)); + }; + + Point$2.prototype._extDbl = function _extDbl() { + // hyperelliptic.org/EFD/g1p/auto-twisted-extended-1.html + // #doubling-dbl-2008-hwcd + // 4M + 4S + + // A = X1^2 + var a = this.x.redSqr(); + // B = Y1^2 + var b = this.y.redSqr(); + // C = 2 * Z1^2 + var c = this.z.redSqr(); + c = c.redIAdd(c); + // D = a * A + var d = this.curve._mulA(a); + // E = (X1 + Y1)^2 - A - B + var e = this.x.redAdd(this.y).redSqr().redISub(a).redISub(b); + // G = D + B + var g = d.redAdd(b); + // F = G - C + var f = g.redSub(c); + // H = D - B + var h = d.redSub(b); + // X3 = E * F + var nx = e.redMul(f); + // Y3 = G * H + var ny = g.redMul(h); + // T3 = E * H + var nt = e.redMul(h); + // Z3 = F * G + var nz = f.redMul(g); + return this.curve.point(nx, ny, nz, nt); + }; + + Point$2.prototype._projDbl = function _projDbl() { + // hyperelliptic.org/EFD/g1p/auto-twisted-projective.html + // #doubling-dbl-2008-bbjlp + // #doubling-dbl-2007-bl + // and others + // Generally 3M + 4S or 2M + 4S + + // B = (X1 + Y1)^2 + var b = this.x.redAdd(this.y).redSqr(); + // C = X1^2 + var c = this.x.redSqr(); + // D = Y1^2 + var d = this.y.redSqr(); + + var nx; + var ny; + var nz; + if (this.curve.twisted) { + // E = a * C + var e = this.curve._mulA(c); + // F = E + D + var f = e.redAdd(d); + if (this.zOne) { + // X3 = (B - C - D) * (F - 2) + nx = b.redSub(c).redSub(d).redMul(f.redSub(this.curve.two)); + // Y3 = F * (E - D) + ny = f.redMul(e.redSub(d)); + // Z3 = F^2 - 2 * F + nz = f.redSqr().redSub(f).redSub(f); + } else { + // H = Z1^2 + var h = this.z.redSqr(); + // J = F - 2 * H + var j = f.redSub(h).redISub(h); + // X3 = (B-C-D)*J + nx = b.redSub(c).redISub(d).redMul(j); + // Y3 = F * (E - D) + ny = f.redMul(e.redSub(d)); + // Z3 = F * J + nz = f.redMul(j); + } + } else { + // E = C + D + var e = c.redAdd(d); + // H = (c * Z1)^2 + var h = this.curve._mulC(this.z).redSqr(); + // J = E - 2 * H + var j = e.redSub(h).redSub(h); + // X3 = c * (B - E) * J + nx = this.curve._mulC(b.redISub(e)).redMul(j); + // Y3 = c * E * (C - D) + ny = this.curve._mulC(e).redMul(c.redISub(d)); + // Z3 = E * J + nz = e.redMul(j); + } + return this.curve.point(nx, ny, nz); + }; + + Point$2.prototype.dbl = function dbl() { + if (this.isInfinity()) + return this; + + // Double in extended coordinates + if (this.curve.extended) + return this._extDbl(); + else + return this._projDbl(); + }; + + Point$2.prototype._extAdd = function _extAdd(p) { + // hyperelliptic.org/EFD/g1p/auto-twisted-extended-1.html + // #addition-add-2008-hwcd-3 + // 8M + + // A = (Y1 - X1) * (Y2 - X2) + var a = this.y.redSub(this.x).redMul(p.y.redSub(p.x)); + // B = (Y1 + X1) * (Y2 + X2) + var b = this.y.redAdd(this.x).redMul(p.y.redAdd(p.x)); + // C = T1 * k * T2 + var c = this.t.redMul(this.curve.dd).redMul(p.t); + // D = Z1 * 2 * Z2 + var d = this.z.redMul(p.z.redAdd(p.z)); + // E = B - A + var e = b.redSub(a); + // F = D - C + var f = d.redSub(c); + // G = D + C + var g = d.redAdd(c); + // H = B + A + var h = b.redAdd(a); + // X3 = E * F + var nx = e.redMul(f); + // Y3 = G * H + var ny = g.redMul(h); + // T3 = E * H + var nt = e.redMul(h); + // Z3 = F * G + var nz = f.redMul(g); + return this.curve.point(nx, ny, nz, nt); + }; + + Point$2.prototype._projAdd = function _projAdd(p) { + // hyperelliptic.org/EFD/g1p/auto-twisted-projective.html + // #addition-add-2008-bbjlp + // #addition-add-2007-bl + // 10M + 1S + + // A = Z1 * Z2 + var a = this.z.redMul(p.z); + // B = A^2 + var b = a.redSqr(); + // C = X1 * X2 + var c = this.x.redMul(p.x); + // D = Y1 * Y2 + var d = this.y.redMul(p.y); + // E = d * C * D + var e = this.curve.d.redMul(c).redMul(d); + // F = B - E + var f = b.redSub(e); + // G = B + E + var g = b.redAdd(e); + // X3 = A * F * ((X1 + Y1) * (X2 + Y2) - C - D) + var tmp = this.x.redAdd(this.y).redMul(p.x.redAdd(p.y)).redISub(c).redISub(d); + var nx = a.redMul(f).redMul(tmp); + var ny; + var nz; + if (this.curve.twisted) { + // Y3 = A * G * (D - a * C) + ny = a.redMul(g).redMul(d.redSub(this.curve._mulA(c))); + // Z3 = F * G + nz = f.redMul(g); + } else { + // Y3 = A * G * (D - C) + ny = a.redMul(g).redMul(d.redSub(c)); + // Z3 = c * F * G + nz = this.curve._mulC(f).redMul(g); + } + return this.curve.point(nx, ny, nz); + }; + + Point$2.prototype.add = function add(p) { + if (this.isInfinity()) + return p; + if (p.isInfinity()) + return this; + + if (this.curve.extended) + return this._extAdd(p); + else + return this._projAdd(p); + }; + + Point$2.prototype.mul = function mul(k) { + if (this._hasDoubles(k)) + return this.curve._fixedNafMul(this, k); + else + return this.curve._wnafMul(this, k); + }; + + Point$2.prototype.mulAdd = function mulAdd(k1, p, k2) { + return this.curve._wnafMulAdd(1, [ this, p ], [ k1, k2 ], 2, false); + }; + + Point$2.prototype.jmulAdd = function jmulAdd(k1, p, k2) { + return this.curve._wnafMulAdd(1, [ this, p ], [ k1, k2 ], 2, true); + }; + + Point$2.prototype.normalize = function normalize() { + if (this.zOne) + return this; + + // Normalize coordinates + var zi = this.z.redInvm(); + this.x = this.x.redMul(zi); + this.y = this.y.redMul(zi); + if (this.t) + this.t = this.t.redMul(zi); + this.z = this.curve.one; + this.zOne = true; + return this; + }; + + Point$2.prototype.neg = function neg() { + return this.curve.point(this.x.redNeg(), + this.y, + this.z, + this.t && this.t.redNeg()); + }; + + Point$2.prototype.getX = function getX() { + this.normalize(); + return this.x.fromRed(); + }; + + Point$2.prototype.getY = function getY() { + this.normalize(); + return this.y.fromRed(); + }; + + Point$2.prototype.eq = function eq(other) { + return this === other || + this.getX().cmp(other.getX()) === 0 && + this.getY().cmp(other.getY()) === 0; + }; + + Point$2.prototype.eqXToP = function eqXToP(x) { + var rx = x.toRed(this.curve.red).redMul(this.z); + if (this.x.cmp(rx) === 0) + return true; + + var xc = x.clone(); + var t = this.curve.redN.redMul(this.z); + for (;;) { + xc.iadd(this.curve.n); + if (xc.cmp(this.curve.p) >= 0) + return false; + + rx.redIAdd(t); + if (this.x.cmp(rx) === 0) + return true; + } + }; + + // Compatibility with BaseCurve + Point$2.prototype.toP = Point$2.prototype.normalize; + Point$2.prototype.mixedAdd = Point$2.prototype.add; + + var curve_1 = createCommonjsModule(function (module, exports) { + + var curve = exports; + + curve.base = base; + curve.short = short_1; + curve.mont = mont; + curve.edwards = edwards; + }); + + var rotl32$2 = utils.rotl32; + var sum32$3 = utils.sum32; + var sum32_5$2 = utils.sum32_5; + var ft_1$1 = common$1.ft_1; + var BlockHash$4 = common.BlockHash; + + var sha1_K = [ + 0x5A827999, 0x6ED9EBA1, + 0x8F1BBCDC, 0xCA62C1D6 + ]; + + function SHA1() { + if (!(this instanceof SHA1)) + return new SHA1(); + + BlockHash$4.call(this); + this.h = [ + 0x67452301, 0xefcdab89, 0x98badcfe, + 0x10325476, 0xc3d2e1f0 ]; + this.W = new Array(80); + } + + utils.inherits(SHA1, BlockHash$4); + var _1 = SHA1; + + SHA1.blockSize = 512; + SHA1.outSize = 160; + SHA1.hmacStrength = 80; + SHA1.padLength = 64; + + SHA1.prototype._update = function _update(msg, start) { + var W = this.W; + + for (var i = 0; i < 16; i++) + W[i] = msg[start + i]; + + for(; i < W.length; i++) + W[i] = rotl32$2(W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16], 1); + + var a = this.h[0]; + var b = this.h[1]; + var c = this.h[2]; + var d = this.h[3]; + var e = this.h[4]; + + for (i = 0; i < W.length; i++) { + var s = ~~(i / 20); + var t = sum32_5$2(rotl32$2(a, 5), ft_1$1(s, b, c, d), e, W[i], sha1_K[s]); + e = d; + d = c; + c = rotl32$2(b, 30); + b = a; + a = t; + } + + this.h[0] = sum32$3(this.h[0], a); + this.h[1] = sum32$3(this.h[1], b); + this.h[2] = sum32$3(this.h[2], c); + this.h[3] = sum32$3(this.h[3], d); + this.h[4] = sum32$3(this.h[4], e); + }; + + SHA1.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h, 'big'); + else + return utils.split32(this.h, 'big'); + }; + + var sha1 = _1; + var sha224 = _224; + var sha256 = _256; + var sha384 = _384; + var sha512 = _512; + + var sha = { + sha1: sha1, + sha224: sha224, + sha256: sha256, + sha384: sha384, + sha512: sha512 + }; + + function Hmac(hash, key, enc) { + if (!(this instanceof Hmac)) + return new Hmac(hash, key, enc); + this.Hash = hash; + this.blockSize = hash.blockSize / 8; + this.outSize = hash.outSize / 8; + this.inner = null; + this.outer = null; + + this._init(utils.toArray(key, enc)); + } + var hmac = Hmac; + + Hmac.prototype._init = function init(key) { + // Shorten key, if needed + if (key.length > this.blockSize) + key = new this.Hash().update(key).digest(); + minimalisticAssert(key.length <= this.blockSize); + + // Add padding to key + for (var i = key.length; i < this.blockSize; i++) + key.push(0); + + for (i = 0; i < key.length; i++) + key[i] ^= 0x36; + this.inner = new this.Hash().update(key); + + // 0x36 ^ 0x5c = 0x6a + for (i = 0; i < key.length; i++) + key[i] ^= 0x6a; + this.outer = new this.Hash().update(key); + }; + + Hmac.prototype.update = function update(msg, enc) { + this.inner.update(msg, enc); + return this; + }; + + Hmac.prototype.digest = function digest(enc) { + this.outer.update(this.inner.digest()); + return this.outer.digest(enc); + }; + + var hash_1 = createCommonjsModule(function (module, exports) { + var hash = exports; + + hash.utils = utils; + hash.common = common; + hash.sha = sha; + hash.ripemd = ripemd; + hash.hmac = hmac; + + // Proxy hash functions to the main object + hash.sha1 = hash.sha.sha1; + hash.sha256 = hash.sha.sha256; + hash.sha224 = hash.sha.sha224; + hash.sha384 = hash.sha.sha384; + hash.sha512 = hash.sha.sha512; + hash.ripemd160 = hash.ripemd.ripemd160; + }); + + var secp256k1 = { + doubles: { + step: 4, + points: [ + [ + 'e60fce93b59e9ec53011aabc21c23e97b2a31369b87a5ae9c44ee89e2a6dec0a', + 'f7e3507399e595929db99f34f57937101296891e44d23f0be1f32cce69616821' + ], + [ + '8282263212c609d9ea2a6e3e172de238d8c39cabd5ac1ca10646e23fd5f51508', + '11f8a8098557dfe45e8256e830b60ace62d613ac2f7b17bed31b6eaff6e26caf' + ], + [ + '175e159f728b865a72f99cc6c6fc846de0b93833fd2222ed73fce5b551e5b739', + 'd3506e0d9e3c79eba4ef97a51ff71f5eacb5955add24345c6efa6ffee9fed695' + ], + [ + '363d90d447b00c9c99ceac05b6262ee053441c7e55552ffe526bad8f83ff4640', + '4e273adfc732221953b445397f3363145b9a89008199ecb62003c7f3bee9de9' + ], + [ + '8b4b5f165df3c2be8c6244b5b745638843e4a781a15bcd1b69f79a55dffdf80c', + '4aad0a6f68d308b4b3fbd7813ab0da04f9e336546162ee56b3eff0c65fd4fd36' + ], + [ + '723cbaa6e5db996d6bf771c00bd548c7b700dbffa6c0e77bcb6115925232fcda', + '96e867b5595cc498a921137488824d6e2660a0653779494801dc069d9eb39f5f' + ], + [ + 'eebfa4d493bebf98ba5feec812c2d3b50947961237a919839a533eca0e7dd7fa', + '5d9a8ca3970ef0f269ee7edaf178089d9ae4cdc3a711f712ddfd4fdae1de8999' + ], + [ + '100f44da696e71672791d0a09b7bde459f1215a29b3c03bfefd7835b39a48db0', + 'cdd9e13192a00b772ec8f3300c090666b7ff4a18ff5195ac0fbd5cd62bc65a09' + ], + [ + 'e1031be262c7ed1b1dc9227a4a04c017a77f8d4464f3b3852c8acde6e534fd2d', + '9d7061928940405e6bb6a4176597535af292dd419e1ced79a44f18f29456a00d' + ], + [ + 'feea6cae46d55b530ac2839f143bd7ec5cf8b266a41d6af52d5e688d9094696d', + 'e57c6b6c97dce1bab06e4e12bf3ecd5c981c8957cc41442d3155debf18090088' + ], + [ + 'da67a91d91049cdcb367be4be6ffca3cfeed657d808583de33fa978bc1ec6cb1', + '9bacaa35481642bc41f463f7ec9780e5dec7adc508f740a17e9ea8e27a68be1d' + ], + [ + '53904faa0b334cdda6e000935ef22151ec08d0f7bb11069f57545ccc1a37b7c0', + '5bc087d0bc80106d88c9eccac20d3c1c13999981e14434699dcb096b022771c8' + ], + [ + '8e7bcd0bd35983a7719cca7764ca906779b53a043a9b8bcaeff959f43ad86047', + '10b7770b2a3da4b3940310420ca9514579e88e2e47fd68b3ea10047e8460372a' + ], + [ + '385eed34c1cdff21e6d0818689b81bde71a7f4f18397e6690a841e1599c43862', + '283bebc3e8ea23f56701de19e9ebf4576b304eec2086dc8cc0458fe5542e5453' + ], + [ + '6f9d9b803ecf191637c73a4413dfa180fddf84a5947fbc9c606ed86c3fac3a7', + '7c80c68e603059ba69b8e2a30e45c4d47ea4dd2f5c281002d86890603a842160' + ], + [ + '3322d401243c4e2582a2147c104d6ecbf774d163db0f5e5313b7e0e742d0e6bd', + '56e70797e9664ef5bfb019bc4ddaf9b72805f63ea2873af624f3a2e96c28b2a0' + ], + [ + '85672c7d2de0b7da2bd1770d89665868741b3f9af7643397721d74d28134ab83', + '7c481b9b5b43b2eb6374049bfa62c2e5e77f17fcc5298f44c8e3094f790313a6' + ], + [ + '948bf809b1988a46b06c9f1919413b10f9226c60f668832ffd959af60c82a0a', + '53a562856dcb6646dc6b74c5d1c3418c6d4dff08c97cd2bed4cb7f88d8c8e589' + ], + [ + '6260ce7f461801c34f067ce0f02873a8f1b0e44dfc69752accecd819f38fd8e8', + 'bc2da82b6fa5b571a7f09049776a1ef7ecd292238051c198c1a84e95b2b4ae17' + ], + [ + 'e5037de0afc1d8d43d8348414bbf4103043ec8f575bfdc432953cc8d2037fa2d', + '4571534baa94d3b5f9f98d09fb990bddbd5f5b03ec481f10e0e5dc841d755bda' + ], + [ + 'e06372b0f4a207adf5ea905e8f1771b4e7e8dbd1c6a6c5b725866a0ae4fce725', + '7a908974bce18cfe12a27bb2ad5a488cd7484a7787104870b27034f94eee31dd' + ], + [ + '213c7a715cd5d45358d0bbf9dc0ce02204b10bdde2a3f58540ad6908d0559754', + '4b6dad0b5ae462507013ad06245ba190bb4850f5f36a7eeddff2c27534b458f2' + ], + [ + '4e7c272a7af4b34e8dbb9352a5419a87e2838c70adc62cddf0cc3a3b08fbd53c', + '17749c766c9d0b18e16fd09f6def681b530b9614bff7dd33e0b3941817dcaae6' + ], + [ + 'fea74e3dbe778b1b10f238ad61686aa5c76e3db2be43057632427e2840fb27b6', + '6e0568db9b0b13297cf674deccb6af93126b596b973f7b77701d3db7f23cb96f' + ], + [ + '76e64113f677cf0e10a2570d599968d31544e179b760432952c02a4417bdde39', + 'c90ddf8dee4e95cf577066d70681f0d35e2a33d2b56d2032b4b1752d1901ac01' + ], + [ + 'c738c56b03b2abe1e8281baa743f8f9a8f7cc643df26cbee3ab150242bcbb891', + '893fb578951ad2537f718f2eacbfbbbb82314eef7880cfe917e735d9699a84c3' + ], + [ + 'd895626548b65b81e264c7637c972877d1d72e5f3a925014372e9f6588f6c14b', + 'febfaa38f2bc7eae728ec60818c340eb03428d632bb067e179363ed75d7d991f' + ], + [ + 'b8da94032a957518eb0f6433571e8761ceffc73693e84edd49150a564f676e03', + '2804dfa44805a1e4d7c99cc9762808b092cc584d95ff3b511488e4e74efdf6e7' + ], + [ + 'e80fea14441fb33a7d8adab9475d7fab2019effb5156a792f1a11778e3c0df5d', + 'eed1de7f638e00771e89768ca3ca94472d155e80af322ea9fcb4291b6ac9ec78' + ], + [ + 'a301697bdfcd704313ba48e51d567543f2a182031efd6915ddc07bbcc4e16070', + '7370f91cfb67e4f5081809fa25d40f9b1735dbf7c0a11a130c0d1a041e177ea1' + ], + [ + '90ad85b389d6b936463f9d0512678de208cc330b11307fffab7ac63e3fb04ed4', + 'e507a3620a38261affdcbd9427222b839aefabe1582894d991d4d48cb6ef150' + ], + [ + '8f68b9d2f63b5f339239c1ad981f162ee88c5678723ea3351b7b444c9ec4c0da', + '662a9f2dba063986de1d90c2b6be215dbbea2cfe95510bfdf23cbf79501fff82' + ], + [ + 'e4f3fb0176af85d65ff99ff9198c36091f48e86503681e3e6686fd5053231e11', + '1e63633ad0ef4f1c1661a6d0ea02b7286cc7e74ec951d1c9822c38576feb73bc' + ], + [ + '8c00fa9b18ebf331eb961537a45a4266c7034f2f0d4e1d0716fb6eae20eae29e', + 'efa47267fea521a1a9dc343a3736c974c2fadafa81e36c54e7d2a4c66702414b' + ], + [ + 'e7a26ce69dd4829f3e10cec0a9e98ed3143d084f308b92c0997fddfc60cb3e41', + '2a758e300fa7984b471b006a1aafbb18d0a6b2c0420e83e20e8a9421cf2cfd51' + ], + [ + 'b6459e0ee3662ec8d23540c223bcbdc571cbcb967d79424f3cf29eb3de6b80ef', + '67c876d06f3e06de1dadf16e5661db3c4b3ae6d48e35b2ff30bf0b61a71ba45' + ], + [ + 'd68a80c8280bb840793234aa118f06231d6f1fc67e73c5a5deda0f5b496943e8', + 'db8ba9fff4b586d00c4b1f9177b0e28b5b0e7b8f7845295a294c84266b133120' + ], + [ + '324aed7df65c804252dc0270907a30b09612aeb973449cea4095980fc28d3d5d', + '648a365774b61f2ff130c0c35aec1f4f19213b0c7e332843967224af96ab7c84' + ], + [ + '4df9c14919cde61f6d51dfdbe5fee5dceec4143ba8d1ca888e8bd373fd054c96', + '35ec51092d8728050974c23a1d85d4b5d506cdc288490192ebac06cad10d5d' + ], + [ + '9c3919a84a474870faed8a9c1cc66021523489054d7f0308cbfc99c8ac1f98cd', + 'ddb84f0f4a4ddd57584f044bf260e641905326f76c64c8e6be7e5e03d4fc599d' + ], + [ + '6057170b1dd12fdf8de05f281d8e06bb91e1493a8b91d4cc5a21382120a959e5', + '9a1af0b26a6a4807add9a2daf71df262465152bc3ee24c65e899be932385a2a8' + ], + [ + 'a576df8e23a08411421439a4518da31880cef0fba7d4df12b1a6973eecb94266', + '40a6bf20e76640b2c92b97afe58cd82c432e10a7f514d9f3ee8be11ae1b28ec8' + ], + [ + '7778a78c28dec3e30a05fe9629de8c38bb30d1f5cf9a3a208f763889be58ad71', + '34626d9ab5a5b22ff7098e12f2ff580087b38411ff24ac563b513fc1fd9f43ac' + ], + [ + '928955ee637a84463729fd30e7afd2ed5f96274e5ad7e5cb09eda9c06d903ac', + 'c25621003d3f42a827b78a13093a95eeac3d26efa8a8d83fc5180e935bcd091f' + ], + [ + '85d0fef3ec6db109399064f3a0e3b2855645b4a907ad354527aae75163d82751', + '1f03648413a38c0be29d496e582cf5663e8751e96877331582c237a24eb1f962' + ], + [ + 'ff2b0dce97eece97c1c9b6041798b85dfdfb6d8882da20308f5404824526087e', + '493d13fef524ba188af4c4dc54d07936c7b7ed6fb90e2ceb2c951e01f0c29907' + ], + [ + '827fbbe4b1e880ea9ed2b2e6301b212b57f1ee148cd6dd28780e5e2cf856e241', + 'c60f9c923c727b0b71bef2c67d1d12687ff7a63186903166d605b68baec293ec' + ], + [ + 'eaa649f21f51bdbae7be4ae34ce6e5217a58fdce7f47f9aa7f3b58fa2120e2b3', + 'be3279ed5bbbb03ac69a80f89879aa5a01a6b965f13f7e59d47a5305ba5ad93d' + ], + [ + 'e4a42d43c5cf169d9391df6decf42ee541b6d8f0c9a137401e23632dda34d24f', + '4d9f92e716d1c73526fc99ccfb8ad34ce886eedfa8d8e4f13a7f7131deba9414' + ], + [ + '1ec80fef360cbdd954160fadab352b6b92b53576a88fea4947173b9d4300bf19', + 'aeefe93756b5340d2f3a4958a7abbf5e0146e77f6295a07b671cdc1cc107cefd' + ], + [ + '146a778c04670c2f91b00af4680dfa8bce3490717d58ba889ddb5928366642be', + 'b318e0ec3354028add669827f9d4b2870aaa971d2f7e5ed1d0b297483d83efd0' + ], + [ + 'fa50c0f61d22e5f07e3acebb1aa07b128d0012209a28b9776d76a8793180eef9', + '6b84c6922397eba9b72cd2872281a68a5e683293a57a213b38cd8d7d3f4f2811' + ], + [ + 'da1d61d0ca721a11b1a5bf6b7d88e8421a288ab5d5bba5220e53d32b5f067ec2', + '8157f55a7c99306c79c0766161c91e2966a73899d279b48a655fba0f1ad836f1' + ], + [ + 'a8e282ff0c9706907215ff98e8fd416615311de0446f1e062a73b0610d064e13', + '7f97355b8db81c09abfb7f3c5b2515888b679a3e50dd6bd6cef7c73111f4cc0c' + ], + [ + '174a53b9c9a285872d39e56e6913cab15d59b1fa512508c022f382de8319497c', + 'ccc9dc37abfc9c1657b4155f2c47f9e6646b3a1d8cb9854383da13ac079afa73' + ], + [ + '959396981943785c3d3e57edf5018cdbe039e730e4918b3d884fdff09475b7ba', + '2e7e552888c331dd8ba0386a4b9cd6849c653f64c8709385e9b8abf87524f2fd' + ], + [ + 'd2a63a50ae401e56d645a1153b109a8fcca0a43d561fba2dbb51340c9d82b151', + 'e82d86fb6443fcb7565aee58b2948220a70f750af484ca52d4142174dcf89405' + ], + [ + '64587e2335471eb890ee7896d7cfdc866bacbdbd3839317b3436f9b45617e073', + 'd99fcdd5bf6902e2ae96dd6447c299a185b90a39133aeab358299e5e9faf6589' + ], + [ + '8481bde0e4e4d885b3a546d3e549de042f0aa6cea250e7fd358d6c86dd45e458', + '38ee7b8cba5404dd84a25bf39cecb2ca900a79c42b262e556d64b1b59779057e' + ], + [ + '13464a57a78102aa62b6979ae817f4637ffcfed3c4b1ce30bcd6303f6caf666b', + '69be159004614580ef7e433453ccb0ca48f300a81d0942e13f495a907f6ecc27' + ], + [ + 'bc4a9df5b713fe2e9aef430bcc1dc97a0cd9ccede2f28588cada3a0d2d83f366', + 'd3a81ca6e785c06383937adf4b798caa6e8a9fbfa547b16d758d666581f33c1' + ], + [ + '8c28a97bf8298bc0d23d8c749452a32e694b65e30a9472a3954ab30fe5324caa', + '40a30463a3305193378fedf31f7cc0eb7ae784f0451cb9459e71dc73cbef9482' + ], + [ + '8ea9666139527a8c1dd94ce4f071fd23c8b350c5a4bb33748c4ba111faccae0', + '620efabbc8ee2782e24e7c0cfb95c5d735b783be9cf0f8e955af34a30e62b945' + ], + [ + 'dd3625faef5ba06074669716bbd3788d89bdde815959968092f76cc4eb9a9787', + '7a188fa3520e30d461da2501045731ca941461982883395937f68d00c644a573' + ], + [ + 'f710d79d9eb962297e4f6232b40e8f7feb2bc63814614d692c12de752408221e', + 'ea98e67232d3b3295d3b535532115ccac8612c721851617526ae47a9c77bfc82' + ] + ] + }, + naf: { + wnd: 7, + points: [ + [ + 'f9308a019258c31049344f85f89d5229b531c845836f99b08601f113bce036f9', + '388f7b0f632de8140fe337e62a37f3566500a99934c2231b6cb9fd7584b8e672' + ], + [ + '2f8bde4d1a07209355b4a7250a5c5128e88b84bddc619ab7cba8d569b240efe4', + 'd8ac222636e5e3d6d4dba9dda6c9c426f788271bab0d6840dca87d3aa6ac62d6' + ], + [ + '5cbdf0646e5db4eaa398f365f2ea7a0e3d419b7e0330e39ce92bddedcac4f9bc', + '6aebca40ba255960a3178d6d861a54dba813d0b813fde7b5a5082628087264da' + ], + [ + 'acd484e2f0c7f65309ad178a9f559abde09796974c57e714c35f110dfc27ccbe', + 'cc338921b0a7d9fd64380971763b61e9add888a4375f8e0f05cc262ac64f9c37' + ], + [ + '774ae7f858a9411e5ef4246b70c65aac5649980be5c17891bbec17895da008cb', + 'd984a032eb6b5e190243dd56d7b7b365372db1e2dff9d6a8301d74c9c953c61b' + ], + [ + 'f28773c2d975288bc7d1d205c3748651b075fbc6610e58cddeeddf8f19405aa8', + 'ab0902e8d880a89758212eb65cdaf473a1a06da521fa91f29b5cb52db03ed81' + ], + [ + 'd7924d4f7d43ea965a465ae3095ff41131e5946f3c85f79e44adbcf8e27e080e', + '581e2872a86c72a683842ec228cc6defea40af2bd896d3a5c504dc9ff6a26b58' + ], + [ + 'defdea4cdb677750a420fee807eacf21eb9898ae79b9768766e4faa04a2d4a34', + '4211ab0694635168e997b0ead2a93daeced1f4a04a95c0f6cfb199f69e56eb77' + ], + [ + '2b4ea0a797a443d293ef5cff444f4979f06acfebd7e86d277475656138385b6c', + '85e89bc037945d93b343083b5a1c86131a01f60c50269763b570c854e5c09b7a' + ], + [ + '352bbf4a4cdd12564f93fa332ce333301d9ad40271f8107181340aef25be59d5', + '321eb4075348f534d59c18259dda3e1f4a1b3b2e71b1039c67bd3d8bcf81998c' + ], + [ + '2fa2104d6b38d11b0230010559879124e42ab8dfeff5ff29dc9cdadd4ecacc3f', + '2de1068295dd865b64569335bd5dd80181d70ecfc882648423ba76b532b7d67' + ], + [ + '9248279b09b4d68dab21a9b066edda83263c3d84e09572e269ca0cd7f5453714', + '73016f7bf234aade5d1aa71bdea2b1ff3fc0de2a887912ffe54a32ce97cb3402' + ], + [ + 'daed4f2be3a8bf278e70132fb0beb7522f570e144bf615c07e996d443dee8729', + 'a69dce4a7d6c98e8d4a1aca87ef8d7003f83c230f3afa726ab40e52290be1c55' + ], + [ + 'c44d12c7065d812e8acf28d7cbb19f9011ecd9e9fdf281b0e6a3b5e87d22e7db', + '2119a460ce326cdc76c45926c982fdac0e106e861edf61c5a039063f0e0e6482' + ], + [ + '6a245bf6dc698504c89a20cfded60853152b695336c28063b61c65cbd269e6b4', + 'e022cf42c2bd4a708b3f5126f16a24ad8b33ba48d0423b6efd5e6348100d8a82' + ], + [ + '1697ffa6fd9de627c077e3d2fe541084ce13300b0bec1146f95ae57f0d0bd6a5', + 'b9c398f186806f5d27561506e4557433a2cf15009e498ae7adee9d63d01b2396' + ], + [ + '605bdb019981718b986d0f07e834cb0d9deb8360ffb7f61df982345ef27a7479', + '2972d2de4f8d20681a78d93ec96fe23c26bfae84fb14db43b01e1e9056b8c49' + ], + [ + '62d14dab4150bf497402fdc45a215e10dcb01c354959b10cfe31c7e9d87ff33d', + '80fc06bd8cc5b01098088a1950eed0db01aa132967ab472235f5642483b25eaf' + ], + [ + '80c60ad0040f27dade5b4b06c408e56b2c50e9f56b9b8b425e555c2f86308b6f', + '1c38303f1cc5c30f26e66bad7fe72f70a65eed4cbe7024eb1aa01f56430bd57a' + ], + [ + '7a9375ad6167ad54aa74c6348cc54d344cc5dc9487d847049d5eabb0fa03c8fb', + 'd0e3fa9eca8726909559e0d79269046bdc59ea10c70ce2b02d499ec224dc7f7' + ], + [ + 'd528ecd9b696b54c907a9ed045447a79bb408ec39b68df504bb51f459bc3ffc9', + 'eecf41253136e5f99966f21881fd656ebc4345405c520dbc063465b521409933' + ], + [ + '49370a4b5f43412ea25f514e8ecdad05266115e4a7ecb1387231808f8b45963', + '758f3f41afd6ed428b3081b0512fd62a54c3f3afbb5b6764b653052a12949c9a' + ], + [ + '77f230936ee88cbbd73df930d64702ef881d811e0e1498e2f1c13eb1fc345d74', + '958ef42a7886b6400a08266e9ba1b37896c95330d97077cbbe8eb3c7671c60d6' + ], + [ + 'f2dac991cc4ce4b9ea44887e5c7c0bce58c80074ab9d4dbaeb28531b7739f530', + 'e0dedc9b3b2f8dad4da1f32dec2531df9eb5fbeb0598e4fd1a117dba703a3c37' + ], + [ + '463b3d9f662621fb1b4be8fbbe2520125a216cdfc9dae3debcba4850c690d45b', + '5ed430d78c296c3543114306dd8622d7c622e27c970a1de31cb377b01af7307e' + ], + [ + 'f16f804244e46e2a09232d4aff3b59976b98fac14328a2d1a32496b49998f247', + 'cedabd9b82203f7e13d206fcdf4e33d92a6c53c26e5cce26d6579962c4e31df6' + ], + [ + 'caf754272dc84563b0352b7a14311af55d245315ace27c65369e15f7151d41d1', + 'cb474660ef35f5f2a41b643fa5e460575f4fa9b7962232a5c32f908318a04476' + ], + [ + '2600ca4b282cb986f85d0f1709979d8b44a09c07cb86d7c124497bc86f082120', + '4119b88753c15bd6a693b03fcddbb45d5ac6be74ab5f0ef44b0be9475a7e4b40' + ], + [ + '7635ca72d7e8432c338ec53cd12220bc01c48685e24f7dc8c602a7746998e435', + '91b649609489d613d1d5e590f78e6d74ecfc061d57048bad9e76f302c5b9c61' + ], + [ + '754e3239f325570cdbbf4a87deee8a66b7f2b33479d468fbc1a50743bf56cc18', + '673fb86e5bda30fb3cd0ed304ea49a023ee33d0197a695d0c5d98093c536683' + ], + [ + 'e3e6bd1071a1e96aff57859c82d570f0330800661d1c952f9fe2694691d9b9e8', + '59c9e0bba394e76f40c0aa58379a3cb6a5a2283993e90c4167002af4920e37f5' + ], + [ + '186b483d056a033826ae73d88f732985c4ccb1f32ba35f4b4cc47fdcf04aa6eb', + '3b952d32c67cf77e2e17446e204180ab21fb8090895138b4a4a797f86e80888b' + ], + [ + 'df9d70a6b9876ce544c98561f4be4f725442e6d2b737d9c91a8321724ce0963f', + '55eb2dafd84d6ccd5f862b785dc39d4ab157222720ef9da217b8c45cf2ba2417' + ], + [ + '5edd5cc23c51e87a497ca815d5dce0f8ab52554f849ed8995de64c5f34ce7143', + 'efae9c8dbc14130661e8cec030c89ad0c13c66c0d17a2905cdc706ab7399a868' + ], + [ + '290798c2b6476830da12fe02287e9e777aa3fba1c355b17a722d362f84614fba', + 'e38da76dcd440621988d00bcf79af25d5b29c094db2a23146d003afd41943e7a' + ], + [ + 'af3c423a95d9f5b3054754efa150ac39cd29552fe360257362dfdecef4053b45', + 'f98a3fd831eb2b749a93b0e6f35cfb40c8cd5aa667a15581bc2feded498fd9c6' + ], + [ + '766dbb24d134e745cccaa28c99bf274906bb66b26dcf98df8d2fed50d884249a', + '744b1152eacbe5e38dcc887980da38b897584a65fa06cedd2c924f97cbac5996' + ], + [ + '59dbf46f8c94759ba21277c33784f41645f7b44f6c596a58ce92e666191abe3e', + 'c534ad44175fbc300f4ea6ce648309a042ce739a7919798cd85e216c4a307f6e' + ], + [ + 'f13ada95103c4537305e691e74e9a4a8dd647e711a95e73cb62dc6018cfd87b8', + 'e13817b44ee14de663bf4bc808341f326949e21a6a75c2570778419bdaf5733d' + ], + [ + '7754b4fa0e8aced06d4167a2c59cca4cda1869c06ebadfb6488550015a88522c', + '30e93e864e669d82224b967c3020b8fa8d1e4e350b6cbcc537a48b57841163a2' + ], + [ + '948dcadf5990e048aa3874d46abef9d701858f95de8041d2a6828c99e2262519', + 'e491a42537f6e597d5d28a3224b1bc25df9154efbd2ef1d2cbba2cae5347d57e' + ], + [ + '7962414450c76c1689c7b48f8202ec37fb224cf5ac0bfa1570328a8a3d7c77ab', + '100b610ec4ffb4760d5c1fc133ef6f6b12507a051f04ac5760afa5b29db83437' + ], + [ + '3514087834964b54b15b160644d915485a16977225b8847bb0dd085137ec47ca', + 'ef0afbb2056205448e1652c48e8127fc6039e77c15c2378b7e7d15a0de293311' + ], + [ + 'd3cc30ad6b483e4bc79ce2c9dd8bc54993e947eb8df787b442943d3f7b527eaf', + '8b378a22d827278d89c5e9be8f9508ae3c2ad46290358630afb34db04eede0a4' + ], + [ + '1624d84780732860ce1c78fcbfefe08b2b29823db913f6493975ba0ff4847610', + '68651cf9b6da903e0914448c6cd9d4ca896878f5282be4c8cc06e2a404078575' + ], + [ + '733ce80da955a8a26902c95633e62a985192474b5af207da6df7b4fd5fc61cd4', + 'f5435a2bd2badf7d485a4d8b8db9fcce3e1ef8e0201e4578c54673bc1dc5ea1d' + ], + [ + '15d9441254945064cf1a1c33bbd3b49f8966c5092171e699ef258dfab81c045c', + 'd56eb30b69463e7234f5137b73b84177434800bacebfc685fc37bbe9efe4070d' + ], + [ + 'a1d0fcf2ec9de675b612136e5ce70d271c21417c9d2b8aaaac138599d0717940', + 'edd77f50bcb5a3cab2e90737309667f2641462a54070f3d519212d39c197a629' + ], + [ + 'e22fbe15c0af8ccc5780c0735f84dbe9a790badee8245c06c7ca37331cb36980', + 'a855babad5cd60c88b430a69f53a1a7a38289154964799be43d06d77d31da06' + ], + [ + '311091dd9860e8e20ee13473c1155f5f69635e394704eaa74009452246cfa9b3', + '66db656f87d1f04fffd1f04788c06830871ec5a64feee685bd80f0b1286d8374' + ], + [ + '34c1fd04d301be89b31c0442d3e6ac24883928b45a9340781867d4232ec2dbdf', + '9414685e97b1b5954bd46f730174136d57f1ceeb487443dc5321857ba73abee' + ], + [ + 'f219ea5d6b54701c1c14de5b557eb42a8d13f3abbcd08affcc2a5e6b049b8d63', + '4cb95957e83d40b0f73af4544cccf6b1f4b08d3c07b27fb8d8c2962a400766d1' + ], + [ + 'd7b8740f74a8fbaab1f683db8f45de26543a5490bca627087236912469a0b448', + 'fa77968128d9c92ee1010f337ad4717eff15db5ed3c049b3411e0315eaa4593b' + ], + [ + '32d31c222f8f6f0ef86f7c98d3a3335ead5bcd32abdd94289fe4d3091aa824bf', + '5f3032f5892156e39ccd3d7915b9e1da2e6dac9e6f26e961118d14b8462e1661' + ], + [ + '7461f371914ab32671045a155d9831ea8793d77cd59592c4340f86cbc18347b5', + '8ec0ba238b96bec0cbdddcae0aa442542eee1ff50c986ea6b39847b3cc092ff6' + ], + [ + 'ee079adb1df1860074356a25aa38206a6d716b2c3e67453d287698bad7b2b2d6', + '8dc2412aafe3be5c4c5f37e0ecc5f9f6a446989af04c4e25ebaac479ec1c8c1e' + ], + [ + '16ec93e447ec83f0467b18302ee620f7e65de331874c9dc72bfd8616ba9da6b5', + '5e4631150e62fb40d0e8c2a7ca5804a39d58186a50e497139626778e25b0674d' + ], + [ + 'eaa5f980c245f6f038978290afa70b6bd8855897f98b6aa485b96065d537bd99', + 'f65f5d3e292c2e0819a528391c994624d784869d7e6ea67fb18041024edc07dc' + ], + [ + '78c9407544ac132692ee1910a02439958ae04877151342ea96c4b6b35a49f51', + 'f3e0319169eb9b85d5404795539a5e68fa1fbd583c064d2462b675f194a3ddb4' + ], + [ + '494f4be219a1a77016dcd838431aea0001cdc8ae7a6fc688726578d9702857a5', + '42242a969283a5f339ba7f075e36ba2af925ce30d767ed6e55f4b031880d562c' + ], + [ + 'a598a8030da6d86c6bc7f2f5144ea549d28211ea58faa70ebf4c1e665c1fe9b5', + '204b5d6f84822c307e4b4a7140737aec23fc63b65b35f86a10026dbd2d864e6b' + ], + [ + 'c41916365abb2b5d09192f5f2dbeafec208f020f12570a184dbadc3e58595997', + '4f14351d0087efa49d245b328984989d5caf9450f34bfc0ed16e96b58fa9913' + ], + [ + '841d6063a586fa475a724604da03bc5b92a2e0d2e0a36acfe4c73a5514742881', + '73867f59c0659e81904f9a1c7543698e62562d6744c169ce7a36de01a8d6154' + ], + [ + '5e95bb399a6971d376026947f89bde2f282b33810928be4ded112ac4d70e20d5', + '39f23f366809085beebfc71181313775a99c9aed7d8ba38b161384c746012865' + ], + [ + '36e4641a53948fd476c39f8a99fd974e5ec07564b5315d8bf99471bca0ef2f66', + 'd2424b1b1abe4eb8164227b085c9aa9456ea13493fd563e06fd51cf5694c78fc' + ], + [ + '336581ea7bfbbb290c191a2f507a41cf5643842170e914faeab27c2c579f726', + 'ead12168595fe1be99252129b6e56b3391f7ab1410cd1e0ef3dcdcabd2fda224' + ], + [ + '8ab89816dadfd6b6a1f2634fcf00ec8403781025ed6890c4849742706bd43ede', + '6fdcef09f2f6d0a044e654aef624136f503d459c3e89845858a47a9129cdd24e' + ], + [ + '1e33f1a746c9c5778133344d9299fcaa20b0938e8acff2544bb40284b8c5fb94', + '60660257dd11b3aa9c8ed618d24edff2306d320f1d03010e33a7d2057f3b3b6' + ], + [ + '85b7c1dcb3cec1b7ee7f30ded79dd20a0ed1f4cc18cbcfcfa410361fd8f08f31', + '3d98a9cdd026dd43f39048f25a8847f4fcafad1895d7a633c6fed3c35e999511' + ], + [ + '29df9fbd8d9e46509275f4b125d6d45d7fbe9a3b878a7af872a2800661ac5f51', + 'b4c4fe99c775a606e2d8862179139ffda61dc861c019e55cd2876eb2a27d84b' + ], + [ + 'a0b1cae06b0a847a3fea6e671aaf8adfdfe58ca2f768105c8082b2e449fce252', + 'ae434102edde0958ec4b19d917a6a28e6b72da1834aff0e650f049503a296cf2' + ], + [ + '4e8ceafb9b3e9a136dc7ff67e840295b499dfb3b2133e4ba113f2e4c0e121e5', + 'cf2174118c8b6d7a4b48f6d534ce5c79422c086a63460502b827ce62a326683c' + ], + [ + 'd24a44e047e19b6f5afb81c7ca2f69080a5076689a010919f42725c2b789a33b', + '6fb8d5591b466f8fc63db50f1c0f1c69013f996887b8244d2cdec417afea8fa3' + ], + [ + 'ea01606a7a6c9cdd249fdfcfacb99584001edd28abbab77b5104e98e8e3b35d4', + '322af4908c7312b0cfbfe369f7a7b3cdb7d4494bc2823700cfd652188a3ea98d' + ], + [ + 'af8addbf2b661c8a6c6328655eb96651252007d8c5ea31be4ad196de8ce2131f', + '6749e67c029b85f52a034eafd096836b2520818680e26ac8f3dfbcdb71749700' + ], + [ + 'e3ae1974566ca06cc516d47e0fb165a674a3dabcfca15e722f0e3450f45889', + '2aeabe7e4531510116217f07bf4d07300de97e4874f81f533420a72eeb0bd6a4' + ], + [ + '591ee355313d99721cf6993ffed1e3e301993ff3ed258802075ea8ced397e246', + 'b0ea558a113c30bea60fc4775460c7901ff0b053d25ca2bdeee98f1a4be5d196' + ], + [ + '11396d55fda54c49f19aa97318d8da61fa8584e47b084945077cf03255b52984', + '998c74a8cd45ac01289d5833a7beb4744ff536b01b257be4c5767bea93ea57a4' + ], + [ + '3c5d2a1ba39c5a1790000738c9e0c40b8dcdfd5468754b6405540157e017aa7a', + 'b2284279995a34e2f9d4de7396fc18b80f9b8b9fdd270f6661f79ca4c81bd257' + ], + [ + 'cc8704b8a60a0defa3a99a7299f2e9c3fbc395afb04ac078425ef8a1793cc030', + 'bdd46039feed17881d1e0862db347f8cf395b74fc4bcdc4e940b74e3ac1f1b13' + ], + [ + 'c533e4f7ea8555aacd9777ac5cad29b97dd4defccc53ee7ea204119b2889b197', + '6f0a256bc5efdf429a2fb6242f1a43a2d9b925bb4a4b3a26bb8e0f45eb596096' + ], + [ + 'c14f8f2ccb27d6f109f6d08d03cc96a69ba8c34eec07bbcf566d48e33da6593', + 'c359d6923bb398f7fd4473e16fe1c28475b740dd098075e6c0e8649113dc3a38' + ], + [ + 'a6cbc3046bc6a450bac24789fa17115a4c9739ed75f8f21ce441f72e0b90e6ef', + '21ae7f4680e889bb130619e2c0f95a360ceb573c70603139862afd617fa9b9f' + ], + [ + '347d6d9a02c48927ebfb86c1359b1caf130a3c0267d11ce6344b39f99d43cc38', + '60ea7f61a353524d1c987f6ecec92f086d565ab687870cb12689ff1e31c74448' + ], + [ + 'da6545d2181db8d983f7dcb375ef5866d47c67b1bf31c8cf855ef7437b72656a', + '49b96715ab6878a79e78f07ce5680c5d6673051b4935bd897fea824b77dc208a' + ], + [ + 'c40747cc9d012cb1a13b8148309c6de7ec25d6945d657146b9d5994b8feb1111', + '5ca560753be2a12fc6de6caf2cb489565db936156b9514e1bb5e83037e0fa2d4' + ], + [ + '4e42c8ec82c99798ccf3a610be870e78338c7f713348bd34c8203ef4037f3502', + '7571d74ee5e0fb92a7a8b33a07783341a5492144cc54bcc40a94473693606437' + ], + [ + '3775ab7089bc6af823aba2e1af70b236d251cadb0c86743287522a1b3b0dedea', + 'be52d107bcfa09d8bcb9736a828cfa7fac8db17bf7a76a2c42ad961409018cf7' + ], + [ + 'cee31cbf7e34ec379d94fb814d3d775ad954595d1314ba8846959e3e82f74e26', + '8fd64a14c06b589c26b947ae2bcf6bfa0149ef0be14ed4d80f448a01c43b1c6d' + ], + [ + 'b4f9eaea09b6917619f6ea6a4eb5464efddb58fd45b1ebefcdc1a01d08b47986', + '39e5c9925b5a54b07433a4f18c61726f8bb131c012ca542eb24a8ac07200682a' + ], + [ + 'd4263dfc3d2df923a0179a48966d30ce84e2515afc3dccc1b77907792ebcc60e', + '62dfaf07a0f78feb30e30d6295853ce189e127760ad6cf7fae164e122a208d54' + ], + [ + '48457524820fa65a4f8d35eb6930857c0032acc0a4a2de422233eeda897612c4', + '25a748ab367979d98733c38a1fa1c2e7dc6cc07db2d60a9ae7a76aaa49bd0f77' + ], + [ + 'dfeeef1881101f2cb11644f3a2afdfc2045e19919152923f367a1767c11cceda', + 'ecfb7056cf1de042f9420bab396793c0c390bde74b4bbdff16a83ae09a9a7517' + ], + [ + '6d7ef6b17543f8373c573f44e1f389835d89bcbc6062ced36c82df83b8fae859', + 'cd450ec335438986dfefa10c57fea9bcc521a0959b2d80bbf74b190dca712d10' + ], + [ + 'e75605d59102a5a2684500d3b991f2e3f3c88b93225547035af25af66e04541f', + 'f5c54754a8f71ee540b9b48728473e314f729ac5308b06938360990e2bfad125' + ], + [ + 'eb98660f4c4dfaa06a2be453d5020bc99a0c2e60abe388457dd43fefb1ed620c', + '6cb9a8876d9cb8520609af3add26cd20a0a7cd8a9411131ce85f44100099223e' + ], + [ + '13e87b027d8514d35939f2e6892b19922154596941888336dc3563e3b8dba942', + 'fef5a3c68059a6dec5d624114bf1e91aac2b9da568d6abeb2570d55646b8adf1' + ], + [ + 'ee163026e9fd6fe017c38f06a5be6fc125424b371ce2708e7bf4491691e5764a', + '1acb250f255dd61c43d94ccc670d0f58f49ae3fa15b96623e5430da0ad6c62b2' + ], + [ + 'b268f5ef9ad51e4d78de3a750c2dc89b1e626d43505867999932e5db33af3d80', + '5f310d4b3c99b9ebb19f77d41c1dee018cf0d34fd4191614003e945a1216e423' + ], + [ + 'ff07f3118a9df035e9fad85eb6c7bfe42b02f01ca99ceea3bf7ffdba93c4750d', + '438136d603e858a3a5c440c38eccbaddc1d2942114e2eddd4740d098ced1f0d8' + ], + [ + '8d8b9855c7c052a34146fd20ffb658bea4b9f69e0d825ebec16e8c3ce2b526a1', + 'cdb559eedc2d79f926baf44fb84ea4d44bcf50fee51d7ceb30e2e7f463036758' + ], + [ + '52db0b5384dfbf05bfa9d472d7ae26dfe4b851ceca91b1eba54263180da32b63', + 'c3b997d050ee5d423ebaf66a6db9f57b3180c902875679de924b69d84a7b375' + ], + [ + 'e62f9490d3d51da6395efd24e80919cc7d0f29c3f3fa48c6fff543becbd43352', + '6d89ad7ba4876b0b22c2ca280c682862f342c8591f1daf5170e07bfd9ccafa7d' + ], + [ + '7f30ea2476b399b4957509c88f77d0191afa2ff5cb7b14fd6d8e7d65aaab1193', + 'ca5ef7d4b231c94c3b15389a5f6311e9daff7bb67b103e9880ef4bff637acaec' + ], + [ + '5098ff1e1d9f14fb46a210fada6c903fef0fb7b4a1dd1d9ac60a0361800b7a00', + '9731141d81fc8f8084d37c6e7542006b3ee1b40d60dfe5362a5b132fd17ddc0' + ], + [ + '32b78c7de9ee512a72895be6b9cbefa6e2f3c4ccce445c96b9f2c81e2778ad58', + 'ee1849f513df71e32efc3896ee28260c73bb80547ae2275ba497237794c8753c' + ], + [ + 'e2cb74fddc8e9fbcd076eef2a7c72b0ce37d50f08269dfc074b581550547a4f7', + 'd3aa2ed71c9dd2247a62df062736eb0baddea9e36122d2be8641abcb005cc4a4' + ], + [ + '8438447566d4d7bedadc299496ab357426009a35f235cb141be0d99cd10ae3a8', + 'c4e1020916980a4da5d01ac5e6ad330734ef0d7906631c4f2390426b2edd791f' + ], + [ + '4162d488b89402039b584c6fc6c308870587d9c46f660b878ab65c82c711d67e', + '67163e903236289f776f22c25fb8a3afc1732f2b84b4e95dbda47ae5a0852649' + ], + [ + '3fad3fa84caf0f34f0f89bfd2dcf54fc175d767aec3e50684f3ba4a4bf5f683d', + 'cd1bc7cb6cc407bb2f0ca647c718a730cf71872e7d0d2a53fa20efcdfe61826' + ], + [ + '674f2600a3007a00568c1a7ce05d0816c1fb84bf1370798f1c69532faeb1a86b', + '299d21f9413f33b3edf43b257004580b70db57da0b182259e09eecc69e0d38a5' + ], + [ + 'd32f4da54ade74abb81b815ad1fb3b263d82d6c692714bcff87d29bd5ee9f08f', + 'f9429e738b8e53b968e99016c059707782e14f4535359d582fc416910b3eea87' + ], + [ + '30e4e670435385556e593657135845d36fbb6931f72b08cb1ed954f1e3ce3ff6', + '462f9bce619898638499350113bbc9b10a878d35da70740dc695a559eb88db7b' + ], + [ + 'be2062003c51cc3004682904330e4dee7f3dcd10b01e580bf1971b04d4cad297', + '62188bc49d61e5428573d48a74e1c655b1c61090905682a0d5558ed72dccb9bc' + ], + [ + '93144423ace3451ed29e0fb9ac2af211cb6e84a601df5993c419859fff5df04a', + '7c10dfb164c3425f5c71a3f9d7992038f1065224f72bb9d1d902a6d13037b47c' + ], + [ + 'b015f8044f5fcbdcf21ca26d6c34fb8197829205c7b7d2a7cb66418c157b112c', + 'ab8c1e086d04e813744a655b2df8d5f83b3cdc6faa3088c1d3aea1454e3a1d5f' + ], + [ + 'd5e9e1da649d97d89e4868117a465a3a4f8a18de57a140d36b3f2af341a21b52', + '4cb04437f391ed73111a13cc1d4dd0db1693465c2240480d8955e8592f27447a' + ], + [ + 'd3ae41047dd7ca065dbf8ed77b992439983005cd72e16d6f996a5316d36966bb', + 'bd1aeb21ad22ebb22a10f0303417c6d964f8cdd7df0aca614b10dc14d125ac46' + ], + [ + '463e2763d885f958fc66cdd22800f0a487197d0a82e377b49f80af87c897b065', + 'bfefacdb0e5d0fd7df3a311a94de062b26b80c61fbc97508b79992671ef7ca7f' + ], + [ + '7985fdfd127c0567c6f53ec1bb63ec3158e597c40bfe747c83cddfc910641917', + '603c12daf3d9862ef2b25fe1de289aed24ed291e0ec6708703a5bd567f32ed03' + ], + [ + '74a1ad6b5f76e39db2dd249410eac7f99e74c59cb83d2d0ed5ff1543da7703e9', + 'cc6157ef18c9c63cd6193d83631bbea0093e0968942e8c33d5737fd790e0db08' + ], + [ + '30682a50703375f602d416664ba19b7fc9bab42c72747463a71d0896b22f6da3', + '553e04f6b018b4fa6c8f39e7f311d3176290d0e0f19ca73f17714d9977a22ff8' + ], + [ + '9e2158f0d7c0d5f26c3791efefa79597654e7a2b2464f52b1ee6c1347769ef57', + '712fcdd1b9053f09003a3481fa7762e9ffd7c8ef35a38509e2fbf2629008373' + ], + [ + '176e26989a43c9cfeba4029c202538c28172e566e3c4fce7322857f3be327d66', + 'ed8cc9d04b29eb877d270b4878dc43c19aefd31f4eee09ee7b47834c1fa4b1c3' + ], + [ + '75d46efea3771e6e68abb89a13ad747ecf1892393dfc4f1b7004788c50374da8', + '9852390a99507679fd0b86fd2b39a868d7efc22151346e1a3ca4726586a6bed8' + ], + [ + '809a20c67d64900ffb698c4c825f6d5f2310fb0451c869345b7319f645605721', + '9e994980d9917e22b76b061927fa04143d096ccc54963e6a5ebfa5f3f8e286c1' + ], + [ + '1b38903a43f7f114ed4500b4eac7083fdefece1cf29c63528d563446f972c180', + '4036edc931a60ae889353f77fd53de4a2708b26b6f5da72ad3394119daf408f9' + ] + ] + } + }; + + var curves_1 = createCommonjsModule(function (module, exports) { + + var curves = exports; + + + + + + var assert = utils_1$1.assert; + + function PresetCurve(options) { + if (options.type === 'short') + this.curve = new curve_1.short(options); + else if (options.type === 'edwards') + this.curve = new curve_1.edwards(options); + else if (options.type === 'mont') + this.curve = new curve_1.mont(options); + else throw new Error('Unknown curve type.'); + this.g = this.curve.g; + this.n = this.curve.n; + this.hash = options.hash; + + assert(this.g.validate(), 'Invalid curve'); + assert(this.g.mul(this.n).isInfinity(), 'Invalid curve, n*G != O'); + } + curves.PresetCurve = PresetCurve; + + function defineCurve(name, options) { + Object.defineProperty(curves, name, { + configurable: true, + enumerable: true, + get: function() { + var curve = new PresetCurve(options); + Object.defineProperty(curves, name, { + configurable: true, + enumerable: true, + value: curve + }); + return curve; + } + }); + } + + defineCurve('p192', { + type: 'short', + prime: 'p192', + p: 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff', + a: 'ffffffff ffffffff ffffffff fffffffe ffffffff fffffffc', + b: '64210519 e59c80e7 0fa7e9ab 72243049 feb8deec c146b9b1', + n: 'ffffffff ffffffff ffffffff 99def836 146bc9b1 b4d22831', + hash: hash_1.sha256, + gRed: false, + g: [ + '188da80e b03090f6 7cbf20eb 43a18800 f4ff0afd 82ff1012', + '07192b95 ffc8da78 631011ed 6b24cdd5 73f977a1 1e794811' + ] + }); + + defineCurve('p224', { + type: 'short', + prime: 'p224', + p: 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001', + a: 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff fffffffe', + b: 'b4050a85 0c04b3ab f5413256 5044b0b7 d7bfd8ba 270b3943 2355ffb4', + n: 'ffffffff ffffffff ffffffff ffff16a2 e0b8f03e 13dd2945 5c5c2a3d', + hash: hash_1.sha256, + gRed: false, + g: [ + 'b70e0cbd 6bb4bf7f 321390b9 4a03c1d3 56c21122 343280d6 115c1d21', + 'bd376388 b5f723fb 4c22dfe6 cd4375a0 5a074764 44d58199 85007e34' + ] + }); + + defineCurve('p256', { + type: 'short', + prime: null, + p: 'ffffffff 00000001 00000000 00000000 00000000 ffffffff ffffffff ffffffff', + a: 'ffffffff 00000001 00000000 00000000 00000000 ffffffff ffffffff fffffffc', + b: '5ac635d8 aa3a93e7 b3ebbd55 769886bc 651d06b0 cc53b0f6 3bce3c3e 27d2604b', + n: 'ffffffff 00000000 ffffffff ffffffff bce6faad a7179e84 f3b9cac2 fc632551', + hash: hash_1.sha256, + gRed: false, + g: [ + '6b17d1f2 e12c4247 f8bce6e5 63a440f2 77037d81 2deb33a0 f4a13945 d898c296', + '4fe342e2 fe1a7f9b 8ee7eb4a 7c0f9e16 2bce3357 6b315ece cbb64068 37bf51f5' + ] + }); + + defineCurve('p384', { + type: 'short', + prime: null, + p: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'fffffffe ffffffff 00000000 00000000 ffffffff', + a: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'fffffffe ffffffff 00000000 00000000 fffffffc', + b: 'b3312fa7 e23ee7e4 988e056b e3f82d19 181d9c6e fe814112 0314088f ' + + '5013875a c656398d 8a2ed19d 2a85c8ed d3ec2aef', + n: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff c7634d81 ' + + 'f4372ddf 581a0db2 48b0a77a ecec196a ccc52973', + hash: hash_1.sha384, + gRed: false, + g: [ + 'aa87ca22 be8b0537 8eb1c71e f320ad74 6e1d3b62 8ba79b98 59f741e0 82542a38 ' + + '5502f25d bf55296c 3a545e38 72760ab7', + '3617de4a 96262c6f 5d9e98bf 9292dc29 f8f41dbd 289a147c e9da3113 b5f0b8c0 ' + + '0a60b1ce 1d7e819d 7a431d7c 90ea0e5f' + ] + }); + + defineCurve('p521', { + type: 'short', + prime: null, + p: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff ffffffff ffffffff ffffffff', + a: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff ffffffff ffffffff fffffffc', + b: '00000051 953eb961 8e1c9a1f 929a21a0 b68540ee a2da725b ' + + '99b315f3 b8b48991 8ef109e1 56193951 ec7e937b 1652c0bd ' + + '3bb1bf07 3573df88 3d2c34f1 ef451fd4 6b503f00', + n: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff fffffffa 51868783 bf2f966b 7fcc0148 ' + + 'f709a5d0 3bb5c9b8 899c47ae bb6fb71e 91386409', + hash: hash_1.sha512, + gRed: false, + g: [ + '000000c6 858e06b7 0404e9cd 9e3ecb66 2395b442 9c648139 ' + + '053fb521 f828af60 6b4d3dba a14b5e77 efe75928 fe1dc127 ' + + 'a2ffa8de 3348b3c1 856a429b f97e7e31 c2e5bd66', + '00000118 39296a78 9a3bc004 5c8a5fb4 2c7d1bd9 98f54449 ' + + '579b4468 17afbd17 273e662c 97ee7299 5ef42640 c550b901 ' + + '3fad0761 353c7086 a272c240 88be9476 9fd16650' + ] + }); + + // https://tools.ietf.org/html/rfc7748#section-4.1 + defineCurve('curve25519', { + type: 'mont', + prime: 'p25519', + p: '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed', + a: '76d06', + b: '1', + n: '1000000000000000 0000000000000000 14def9dea2f79cd6 5812631a5cf5d3ed', + cofactor: '8', + hash: hash_1.sha256, + gRed: false, + g: [ + '9' + ] + }); + + defineCurve('ed25519', { + type: 'edwards', + prime: 'p25519', + p: '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed', + a: '-1', + c: '1', + // -121665 * (121666^(-1)) (mod P) + d: '52036cee2b6ffe73 8cc740797779e898 00700a4d4141d8ab 75eb4dca135978a3', + n: '1000000000000000 0000000000000000 14def9dea2f79cd6 5812631a5cf5d3ed', + cofactor: '8', + hash: hash_1.sha256, + gRed: false, + g: [ + '216936d3cd6e53fec0a4e231fdd6dc5c692cc7609525a7b2c9562d608f25d51a', + // 4/5 + '6666666666666666666666666666666666666666666666666666666666666658' + ] + }); + + // https://tools.ietf.org/html/rfc5639#section-3.4 + defineCurve('brainpoolP256r1', { + type: 'short', + prime: null, + p: 'A9FB57DB A1EEA9BC 3E660A90 9D838D72 6E3BF623 D5262028 2013481D 1F6E5377', + a: '7D5A0975 FC2C3057 EEF67530 417AFFE7 FB8055C1 26DC5C6C E94A4B44 F330B5D9', + b: '26DC5C6C E94A4B44 F330B5D9 BBD77CBF 95841629 5CF7E1CE 6BCCDC18 FF8C07B6', + n: 'A9FB57DB A1EEA9BC 3E660A90 9D838D71 8C397AA3 B561A6F7 901E0E82 974856A7', + hash: hash_1.sha256, // or 384, or 512 + gRed: false, + g: [ + '8BD2AEB9CB7E57CB2C4B482FFC81B7AFB9DE27E1E3BD23C23A4453BD9ACE3262', + '547EF835C3DAC4FD97F8461A14611DC9C27745132DED8E545C1D54C72F046997' + ] + }); + + // https://tools.ietf.org/html/rfc5639#section-3.6 + defineCurve('brainpoolP384r1', { + type: 'short', + prime: null, + p: '8CB91E82 A3386D28 0F5D6F7E 50E641DF 152F7109 ED5456B4 12B1DA19 7FB71123' + + 'ACD3A729 901D1A71 87470013 3107EC53', + a: '7BC382C6 3D8C150C 3C72080A CE05AFA0 C2BEA28E 4FB22787 139165EF BA91F90F' + + '8AA5814A 503AD4EB 04A8C7DD 22CE2826', + b: '04A8C7DD 22CE2826 8B39B554 16F0447C 2FB77DE1 07DCD2A6 2E880EA5 3EEB62D5' + + '7CB43902 95DBC994 3AB78696 FA504C11', + n: '8CB91E82 A3386D28 0F5D6F7E 50E641DF 152F7109 ED5456B3 1F166E6C AC0425A7' + + 'CF3AB6AF 6B7FC310 3B883202 E9046565', + hash: hash_1.sha384, // or 512 + gRed: false, + g: [ + '1D1C64F068CF45FFA2A63A81B7C13F6B8847A3E77EF14FE3DB7FCAFE0CBD10' + + 'E8E826E03436D646AAEF87B2E247D4AF1E', + '8ABE1D7520F9C2A45CB1EB8E95CFD55262B70B29FEEC5864E19C054FF99129' + + '280E4646217791811142820341263C5315' + ] + }); + + // https://tools.ietf.org/html/rfc5639#section-3.7 + defineCurve('brainpoolP512r1', { + type: 'short', + prime: null, + p: 'AADD9DB8 DBE9C48B 3FD4E6AE 33C9FC07 CB308DB3 B3C9D20E D6639CCA 70330871' + + '7D4D9B00 9BC66842 AECDA12A E6A380E6 2881FF2F 2D82C685 28AA6056 583A48F3', + a: '7830A331 8B603B89 E2327145 AC234CC5 94CBDD8D 3DF91610 A83441CA EA9863BC' + + '2DED5D5A A8253AA1 0A2EF1C9 8B9AC8B5 7F1117A7 2BF2C7B9 E7C1AC4D 77FC94CA', + b: '3DF91610 A83441CA EA9863BC 2DED5D5A A8253AA1 0A2EF1C9 8B9AC8B5 7F1117A7' + + '2BF2C7B9 E7C1AC4D 77FC94CA DC083E67 984050B7 5EBAE5DD 2809BD63 8016F723', + n: 'AADD9DB8 DBE9C48B 3FD4E6AE 33C9FC07 CB308DB3 B3C9D20E D6639CCA 70330870' + + '553E5C41 4CA92619 41866119 7FAC1047 1DB1D381 085DDADD B5879682 9CA90069', + hash: hash_1.sha512, + gRed: false, + g: [ + '81AEE4BDD82ED9645A21322E9C4C6A9385ED9F70B5D916C1B43B62EEF4D009' + + '8EFF3B1F78E2D0D48D50D1687B93B97D5F7C6D5047406A5E688B352209BCB9F822', + '7DDE385D566332ECC0EABFA9CF7822FDF209F70024A57B1AA000C55B881F81' + + '11B2DCDE494A5F485E5BCA4BD88A2763AED1CA2B2FA8F0540678CD1E0F3AD80892' + ] + }); + + // https://en.bitcoin.it/wiki/Secp256k1 + var pre; + try { + pre = secp256k1; + } catch (e) { + pre = undefined; + } + + defineCurve('secp256k1', { + type: 'short', + prime: 'k256', + p: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f', + a: '0', + b: '7', + n: 'ffffffff ffffffff ffffffff fffffffe baaedce6 af48a03b bfd25e8c d0364141', + h: '1', + hash: hash_1.sha256, + + // Precomputed endomorphism + beta: '7ae96a2b657c07106e64479eac3434e99cf0497512f58995c1396c28719501ee', + lambda: '5363ad4cc05c30e0a5261c028812645a122e22ea20816678df02967c1b23bd72', + basis: [ + { + a: '3086d221a7d46bcde86c90e49284eb15', + b: '-e4437ed6010e88286f547fa90abfe4c3' + }, + { + a: '114ca50f7a8e2f3f657c1108d9d44cfd8', + b: '3086d221a7d46bcde86c90e49284eb15' + } + ], + + gRed: false, + g: [ + '79be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798', + '483ada7726a3c4655da4fbfc0e1108a8fd17b448a68554199c47d08ffb10d4b8', + pre + ] + }); + }); + + function HmacDRBG(options) { + if (!(this instanceof HmacDRBG)) + return new HmacDRBG(options); + this.hash = options.hash; + this.predResist = !!options.predResist; + + this.outLen = this.hash.outSize; + this.minEntropy = options.minEntropy || this.hash.hmacStrength; + + this._reseed = null; + this.reseedInterval = null; + this.K = null; + this.V = null; + + var entropy = utils_1.toArray(options.entropy, options.entropyEnc || 'hex'); + var nonce = utils_1.toArray(options.nonce, options.nonceEnc || 'hex'); + var pers = utils_1.toArray(options.pers, options.persEnc || 'hex'); + minimalisticAssert(entropy.length >= (this.minEntropy / 8), + 'Not enough entropy. Minimum is: ' + this.minEntropy + ' bits'); + this._init(entropy, nonce, pers); + } + var hmacDrbg = HmacDRBG; + + HmacDRBG.prototype._init = function init(entropy, nonce, pers) { + var seed = entropy.concat(nonce).concat(pers); + + this.K = new Array(this.outLen / 8); + this.V = new Array(this.outLen / 8); + for (var i = 0; i < this.V.length; i++) { + this.K[i] = 0x00; + this.V[i] = 0x01; + } + + this._update(seed); + this._reseed = 1; + this.reseedInterval = 0x1000000000000; // 2^48 + }; + + HmacDRBG.prototype._hmac = function hmac() { + return new hash_1.hmac(this.hash, this.K); + }; + + HmacDRBG.prototype._update = function update(seed) { + var kmac = this._hmac() + .update(this.V) + .update([ 0x00 ]); + if (seed) + kmac = kmac.update(seed); + this.K = kmac.digest(); + this.V = this._hmac().update(this.V).digest(); + if (!seed) + return; + + this.K = this._hmac() + .update(this.V) + .update([ 0x01 ]) + .update(seed) + .digest(); + this.V = this._hmac().update(this.V).digest(); + }; + + HmacDRBG.prototype.reseed = function reseed(entropy, entropyEnc, add, addEnc) { + // Optional entropy enc + if (typeof entropyEnc !== 'string') { + addEnc = add; + add = entropyEnc; + entropyEnc = null; + } + + entropy = utils_1.toArray(entropy, entropyEnc); + add = utils_1.toArray(add, addEnc); + + minimalisticAssert(entropy.length >= (this.minEntropy / 8), + 'Not enough entropy. Minimum is: ' + this.minEntropy + ' bits'); + + this._update(entropy.concat(add || [])); + this._reseed = 1; + }; + + HmacDRBG.prototype.generate = function generate(len, enc, add, addEnc) { + if (this._reseed > this.reseedInterval) + throw new Error('Reseed is required'); + + // Optional encoding + if (typeof enc !== 'string') { + addEnc = add; + add = enc; + enc = null; + } + + // Optional additional data + if (add) { + add = utils_1.toArray(add, addEnc || 'hex'); + this._update(add); + } + + var temp = []; + while (temp.length < len) { + this.V = this._hmac().update(this.V).digest(); + temp = temp.concat(this.V); + } + + var res = temp.slice(0, len); + this._update(add); + this._reseed++; + return utils_1.encode(res, enc); + }; + + var assert$5 = utils_1$1.assert; + + function KeyPair(ec, options) { + this.ec = ec; + this.priv = null; + this.pub = null; + + // KeyPair(ec, { priv: ..., pub: ... }) + if (options.priv) + this._importPrivate(options.priv, options.privEnc); + if (options.pub) + this._importPublic(options.pub, options.pubEnc); + } + var key = KeyPair; + + KeyPair.fromPublic = function fromPublic(ec, pub, enc) { + if (pub instanceof KeyPair) + return pub; + + return new KeyPair(ec, { + pub: pub, + pubEnc: enc + }); + }; + + KeyPair.fromPrivate = function fromPrivate(ec, priv, enc) { + if (priv instanceof KeyPair) + return priv; + + return new KeyPair(ec, { + priv: priv, + privEnc: enc + }); + }; + + // TODO: should not validate for X25519 + KeyPair.prototype.validate = function validate() { + var pub = this.getPublic(); + + if (pub.isInfinity()) + return { result: false, reason: 'Invalid public key' }; + if (!pub.validate()) + return { result: false, reason: 'Public key is not a point' }; + if (!pub.mul(this.ec.curve.n).isInfinity()) + return { result: false, reason: 'Public key * N != O' }; + + return { result: true, reason: null }; + }; + + KeyPair.prototype.getPublic = function getPublic(enc, compact) { + if (!this.pub) + this.pub = this.ec.g.mul(this.priv); + + if (!enc) + return this.pub; + + return this.pub.encode(enc, compact); + }; + + KeyPair.prototype.getPrivate = function getPrivate(enc) { + if (enc === 'hex') + return this.priv.toString(16, 2); + else + return this.priv; + }; + + KeyPair.prototype._importPrivate = function _importPrivate(key, enc) { + this.priv = new bn(key, enc || 16); + + // For Curve25519/Curve448 we have a specific procedure. + // TODO Curve448 + if (this.ec.curve.type === 'mont') { + var one = this.ec.curve.one; + var mask = one.ushln(255 - 3).sub(one).ushln(3); + this.priv = this.priv.or(one.ushln(255 - 1)); + this.priv = this.priv.and(mask); + } else + // Ensure that the priv won't be bigger than n, otherwise we may fail + // in fixed multiplication method + this.priv = this.priv.umod(this.ec.curve.n); + }; + + KeyPair.prototype._importPublic = function _importPublic(key, enc) { + if (key.x || key.y) { + // Montgomery points only have an `x` coordinate. + // Weierstrass/Edwards points on the other hand have both `x` and + // `y` coordinates. + if (this.ec.curve.type === 'mont') { + assert$5(key.x, 'Need x coordinate'); + } else if (this.ec.curve.type === 'short' || + this.ec.curve.type === 'edwards') { + assert$5(key.x && key.y, 'Need both x and y coordinate'); + } + this.pub = this.ec.curve.point(key.x, key.y); + return; + } + this.pub = this.ec.curve.decodePoint(key, enc); + }; + + // ECDH + KeyPair.prototype.derive = function derive(pub) { + return pub.mul(this.priv).getX(); + }; + + // ECDSA + KeyPair.prototype.sign = function sign(msg, enc, options) { + return this.ec.sign(msg, this, enc, options); + }; + + KeyPair.prototype.verify = function verify(msg, signature) { + return this.ec.verify(msg, signature, this); + }; + + KeyPair.prototype.inspect = function inspect() { + return ''; + }; + + var assert$6 = utils_1$1.assert; + + function Signature$1(options, enc) { + if (options instanceof Signature$1) + return options; + + if (this._importDER(options, enc)) + return; + + assert$6(options.r && options.s, 'Signature without r or s'); + this.r = new bn(options.r, 16); + this.s = new bn(options.s, 16); + if (options.recoveryParam === undefined) + this.recoveryParam = null; + else + this.recoveryParam = options.recoveryParam; + } + var signature$1 = Signature$1; + + function Position() { + this.place = 0; + } + + function getLength(buf, p) { + var initial = buf[p.place++]; + if (!(initial & 0x80)) { + return initial; + } + var octetLen = initial & 0xf; + var val = 0; + for (var i = 0, off = p.place; i < octetLen; i++, off++) { + val <<= 8; + val |= buf[off]; + } + p.place = off; + return val; + } + + function rmPadding(buf) { + var i = 0; + var len = buf.length - 1; + while (!buf[i] && !(buf[i + 1] & 0x80) && i < len) { + i++; + } + if (i === 0) { + return buf; + } + return buf.slice(i); + } + + Signature$1.prototype._importDER = function _importDER(data, enc) { + data = utils_1$1.toArray(data, enc); + var p = new Position(); + if (data[p.place++] !== 0x30) { + return false; + } + var len = getLength(data, p); + if ((len + p.place) !== data.length) { + return false; + } + if (data[p.place++] !== 0x02) { + return false; + } + var rlen = getLength(data, p); + var r = data.slice(p.place, rlen + p.place); + p.place += rlen; + if (data[p.place++] !== 0x02) { + return false; + } + var slen = getLength(data, p); + if (data.length !== slen + p.place) { + return false; + } + var s = data.slice(p.place, slen + p.place); + if (r[0] === 0 && (r[1] & 0x80)) { + r = r.slice(1); + } + if (s[0] === 0 && (s[1] & 0x80)) { + s = s.slice(1); + } + + this.r = new bn(r); + this.s = new bn(s); + this.recoveryParam = null; + + return true; + }; + + function constructLength(arr, len) { + if (len < 0x80) { + arr.push(len); + return; + } + var octets = 1 + (Math.log(len) / Math.LN2 >>> 3); + arr.push(octets | 0x80); + while (--octets) { + arr.push((len >>> (octets << 3)) & 0xff); + } + arr.push(len); + } + + Signature$1.prototype.toDER = function toDER(enc) { + var r = this.r.toArray(); + var s = this.s.toArray(); + + // Pad values + if (r[0] & 0x80) + r = [ 0 ].concat(r); + // Pad values + if (s[0] & 0x80) + s = [ 0 ].concat(s); + + r = rmPadding(r); + s = rmPadding(s); + + while (!s[0] && !(s[1] & 0x80)) { + s = s.slice(1); + } + var arr = [ 0x02 ]; + constructLength(arr, r.length); + arr = arr.concat(r); + arr.push(0x02); + constructLength(arr, s.length); + var backHalf = arr.concat(s); + var res = [ 0x30 ]; + constructLength(res, backHalf.length); + res = res.concat(backHalf); + return utils_1$1.encode(res, enc); + }; + + var assert$7 = utils_1$1.assert; + + + + + function EC(options) { + if (!(this instanceof EC)) + return new EC(options); + + // Shortcut `elliptic.ec(curve-name)` + if (typeof options === 'string') { + assert$7(curves_1.hasOwnProperty(options), 'Unknown curve ' + options); + + options = curves_1[options]; + } + + // Shortcut for `elliptic.ec(elliptic.curves.curveName)` + if (options instanceof curves_1.PresetCurve) + options = { curve: options }; + + this.curve = options.curve.curve; + this.n = this.curve.n; + this.nh = this.n.ushrn(1); + this.g = this.curve.g; + + // Point on curve + this.g = options.curve.g; + this.g.precompute(options.curve.n.bitLength() + 1); + + // Hash function for DRBG + this.hash = options.hash || options.curve.hash; + } + var ec = EC; + + EC.prototype.keyPair = function keyPair(options) { + return new key(this, options); + }; + + EC.prototype.keyFromPrivate = function keyFromPrivate(priv, enc) { + return key.fromPrivate(this, priv, enc); + }; + + EC.prototype.keyFromPublic = function keyFromPublic(pub, enc) { + return key.fromPublic(this, pub, enc); + }; + + EC.prototype.genKeyPair = function genKeyPair(options) { + if (!options) + options = {}; + + // Instantiate Hmac_DRBG + var drbg = new hmacDrbg({ + hash: this.hash, + pers: options.pers, + persEnc: options.persEnc || 'utf8', + entropy: options.entropy || brorand(this.hash.hmacStrength), + entropyEnc: options.entropy && options.entropyEnc || 'utf8', + nonce: this.n.toArray() + }); + + // Key generation for curve25519 is simpler + if (this.curve.type === 'mont') { + var priv = new bn(drbg.generate(32)); + return this.keyFromPrivate(priv); + } + + var bytes = this.n.byteLength(); + var ns2 = this.n.sub(new bn(2)); + do { + var priv = new bn(drbg.generate(bytes)); + if (priv.cmp(ns2) > 0) + continue; + + priv.iaddn(1); + return this.keyFromPrivate(priv); + } while (true); + }; + + EC.prototype._truncateToN = function truncateToN(msg, truncOnly, bitSize) { + bitSize = bitSize || msg.byteLength() * 8; + var delta = bitSize - this.n.bitLength(); + if (delta > 0) + msg = msg.ushrn(delta); + if (!truncOnly && msg.cmp(this.n) >= 0) + return msg.sub(this.n); + else + return msg; + }; + + EC.prototype.truncateMsg = function truncateMSG(msg) { + // Bit size is only determined correctly for Uint8Arrays and hex strings + var bitSize; + if (msg instanceof Uint8Array) { + bitSize = msg.byteLength * 8; + msg = this._truncateToN(new bn(msg, 16), false, bitSize); + } else if (typeof msg === 'string') { + bitSize = msg.length * 4; + msg = this._truncateToN(new bn(msg, 16), false, bitSize); + } else { + msg = this._truncateToN(new bn(msg, 16)); + } + return msg; + }; + + EC.prototype.sign = function sign(msg, key, enc, options) { + if (typeof enc === 'object') { + options = enc; + enc = null; + } + if (!options) + options = {}; + + key = this.keyFromPrivate(key, enc); + msg = this.truncateMsg(msg); + + // Zero-extend key to provide enough entropy + var bytes = this.n.byteLength(); + var bkey = key.getPrivate().toArray('be', bytes); + + // Zero-extend nonce to have the same byte size as N + var nonce = msg.toArray('be', bytes); + + // Instantiate Hmac_DRBG + var drbg = new hmacDrbg({ + hash: this.hash, + entropy: bkey, + nonce: nonce, + pers: options.pers, + persEnc: options.persEnc || 'utf8' + }); + + // Number of bytes to generate + var ns1 = this.n.sub(new bn(1)); + + for (var iter = 0; true; iter++) { + var k = options.k ? + options.k(iter) : + new bn(drbg.generate(this.n.byteLength())); + k = this._truncateToN(k, true); + if (k.cmpn(1) <= 0 || k.cmp(ns1) >= 0) + continue; + + var kp = this.g.mul(k); + if (kp.isInfinity()) + continue; + + var kpX = kp.getX(); + var r = kpX.umod(this.n); + if (r.cmpn(0) === 0) + continue; + + var s = k.invm(this.n).mul(r.mul(key.getPrivate()).iadd(msg)); + s = s.umod(this.n); + if (s.cmpn(0) === 0) + continue; + + var recoveryParam = (kp.getY().isOdd() ? 1 : 0) | + (kpX.cmp(r) !== 0 ? 2 : 0); + + // Use complement of `s`, if it is > `n / 2` + if (options.canonical && s.cmp(this.nh) > 0) { + s = this.n.sub(s); + recoveryParam ^= 1; + } + + return new signature$1({ r: r, s: s, recoveryParam: recoveryParam }); + } + }; + + EC.prototype.verify = function verify(msg, signature, key, enc) { + key = this.keyFromPublic(key, enc); + signature = new signature$1(signature, 'hex'); + // Fallback to the old code + var ret = this._verify(this.truncateMsg(msg), signature, key) || + this._verify(this._truncateToN(new bn(msg, 16)), signature, key); + return ret; + }; + + EC.prototype._verify = function _verify(msg, signature, key) { + // Perform primitive values validation + var r = signature.r; + var s = signature.s; + if (r.cmpn(1) < 0 || r.cmp(this.n) >= 0) + return false; + if (s.cmpn(1) < 0 || s.cmp(this.n) >= 0) + return false; + + // Validate signature + var sinv = s.invm(this.n); + var u1 = sinv.mul(msg).umod(this.n); + var u2 = sinv.mul(r).umod(this.n); + + if (!this.curve._maxwellTrick) { + var p = this.g.mulAdd(u1, key.getPublic(), u2); + if (p.isInfinity()) + return false; + + return p.getX().umod(this.n).cmp(r) === 0; + } + + // NOTE: Greg Maxwell's trick, inspired by: + // https://git.io/vad3K + + var p = this.g.jmulAdd(u1, key.getPublic(), u2); + if (p.isInfinity()) + return false; + + // Compare `p.x` of Jacobian point with `r`, + // this will do `p.x == r * p.z^2` instead of multiplying `p.x` by the + // inverse of `p.z^2` + return p.eqXToP(r); + }; + + EC.prototype.recoverPubKey = function(msg, signature, j, enc) { + assert$7((3 & j) === j, 'The recovery param is more than two bits'); + signature = new signature$1(signature, enc); + + var n = this.n; + var e = new bn(msg); + var r = signature.r; + var s = signature.s; + + // A set LSB signifies that the y-coordinate is odd + var isYOdd = j & 1; + var isSecondKey = j >> 1; + if (r.cmp(this.curve.p.umod(this.curve.n)) >= 0 && isSecondKey) + throw new Error('Unable to find sencond key candinate'); + + // 1.1. Let x = r + jn. + if (isSecondKey) + r = this.curve.pointFromX(r.add(this.curve.n), isYOdd); + else + r = this.curve.pointFromX(r, isYOdd); + + var rInv = signature.r.invm(n); + var s1 = n.sub(e).mul(rInv).umod(n); + var s2 = s.mul(rInv).umod(n); + + // 1.6.1 Compute Q = r^-1 (sR - eG) + // Q = r^-1 (sR + -eG) + return this.g.mulAdd(s1, r, s2); + }; + + EC.prototype.getKeyRecoveryParam = function(e, signature, Q, enc) { + signature = new signature$1(signature, enc); + if (signature.recoveryParam !== null) + return signature.recoveryParam; + + for (var i = 0; i < 4; i++) { + var Qprime; + try { + Qprime = this.recoverPubKey(e, signature, i); + } catch (e) { + continue; + } + + if (Qprime.eq(Q)) + return i; + } + throw new Error('Unable to find valid recovery factor'); + }; + + var assert$8 = utils_1$1.assert; + var parseBytes = utils_1$1.parseBytes; + var cachedProperty = utils_1$1.cachedProperty; + + /** + * @param {EDDSA} eddsa - instance + * @param {Object} params - public/private key parameters + * + * @param {Array} [params.secret] - secret seed bytes + * @param {Point} [params.pub] - public key point (aka `A` in eddsa terms) + * @param {Array} [params.pub] - public key point encoded as bytes + * + */ + function KeyPair$1(eddsa, params) { + this.eddsa = eddsa; + if (params.hasOwnProperty('secret')) + this._secret = parseBytes(params.secret); + if (eddsa.isPoint(params.pub)) + this._pub = params.pub; + else { + this._pubBytes = parseBytes(params.pub); + if (this._pubBytes && this._pubBytes.length === 33 && + this._pubBytes[0] === 0x40) + this._pubBytes = this._pubBytes.slice(1, 33); + if (this._pubBytes && this._pubBytes.length !== 32) + throw new Error('Unknown point compression format'); + } + } + + KeyPair$1.fromPublic = function fromPublic(eddsa, pub) { + if (pub instanceof KeyPair$1) + return pub; + return new KeyPair$1(eddsa, { pub: pub }); + }; + + KeyPair$1.fromSecret = function fromSecret(eddsa, secret) { + if (secret instanceof KeyPair$1) + return secret; + return new KeyPair$1(eddsa, { secret: secret }); + }; + + KeyPair$1.prototype.secret = function secret() { + return this._secret; + }; + + cachedProperty(KeyPair$1, 'pubBytes', function pubBytes() { + return this.eddsa.encodePoint(this.pub()); + }); + + cachedProperty(KeyPair$1, 'pub', function pub() { + if (this._pubBytes) + return this.eddsa.decodePoint(this._pubBytes); + return this.eddsa.g.mul(this.priv()); + }); + + cachedProperty(KeyPair$1, 'privBytes', function privBytes() { + var eddsa = this.eddsa; + var hash = this.hash(); + var lastIx = eddsa.encodingLength - 1; + + // https://tools.ietf.org/html/rfc8032#section-5.1.5 + var a = hash.slice(0, eddsa.encodingLength); + a[0] &= 248; + a[lastIx] &= 127; + a[lastIx] |= 64; + + return a; + }); + + cachedProperty(KeyPair$1, 'priv', function priv() { + return this.eddsa.decodeInt(this.privBytes()); + }); + + cachedProperty(KeyPair$1, 'hash', function hash() { + return this.eddsa.hash().update(this.secret()).digest(); + }); + + cachedProperty(KeyPair$1, 'messagePrefix', function messagePrefix() { + return this.hash().slice(this.eddsa.encodingLength); + }); + + KeyPair$1.prototype.sign = function sign(message) { + assert$8(this._secret, 'KeyPair can only verify'); + return this.eddsa.sign(message, this); + }; + + KeyPair$1.prototype.verify = function verify(message, sig) { + return this.eddsa.verify(message, sig, this); + }; + + KeyPair$1.prototype.getSecret = function getSecret(enc) { + assert$8(this._secret, 'KeyPair is public only'); + return utils_1$1.encode(this.secret(), enc); + }; + + KeyPair$1.prototype.getPublic = function getPublic(enc, compact) { + return utils_1$1.encode((compact ? [ 0x40 ] : []).concat(this.pubBytes()), enc); + }; + + var key$1 = KeyPair$1; + + var assert$9 = utils_1$1.assert; + var cachedProperty$1 = utils_1$1.cachedProperty; + var parseBytes$1 = utils_1$1.parseBytes; + + /** + * @param {EDDSA} eddsa - eddsa instance + * @param {Array|Object} sig - + * @param {Array|Point} [sig.R] - R point as Point or bytes + * @param {Array|bn} [sig.S] - S scalar as bn or bytes + * @param {Array} [sig.Rencoded] - R point encoded + * @param {Array} [sig.Sencoded] - S scalar encoded + */ + function Signature$2(eddsa, sig) { + this.eddsa = eddsa; + + if (typeof sig !== 'object') + sig = parseBytes$1(sig); + + if (Array.isArray(sig)) { + sig = { + R: sig.slice(0, eddsa.encodingLength), + S: sig.slice(eddsa.encodingLength) + }; + } + + assert$9(sig.R && sig.S, 'Signature without R or S'); + + if (eddsa.isPoint(sig.R)) + this._R = sig.R; + if (sig.S instanceof bn) + this._S = sig.S; + + this._Rencoded = Array.isArray(sig.R) ? sig.R : sig.Rencoded; + this._Sencoded = Array.isArray(sig.S) ? sig.S : sig.Sencoded; + } + + cachedProperty$1(Signature$2, 'S', function S() { + return this.eddsa.decodeInt(this.Sencoded()); + }); + + cachedProperty$1(Signature$2, 'R', function R() { + return this.eddsa.decodePoint(this.Rencoded()); + }); + + cachedProperty$1(Signature$2, 'Rencoded', function Rencoded() { + return this.eddsa.encodePoint(this.R()); + }); + + cachedProperty$1(Signature$2, 'Sencoded', function Sencoded() { + return this.eddsa.encodeInt(this.S()); + }); + + Signature$2.prototype.toBytes = function toBytes() { + return this.Rencoded().concat(this.Sencoded()); + }; + + Signature$2.prototype.toHex = function toHex() { + return utils_1$1.encode(this.toBytes(), 'hex').toUpperCase(); + }; + + var signature$2 = Signature$2; + + var assert$a = utils_1$1.assert; + var parseBytes$2 = utils_1$1.parseBytes; + + + + function EDDSA(curve) { + assert$a(curve === 'ed25519', 'only tested with ed25519 so far'); + + if (!(this instanceof EDDSA)) + return new EDDSA(curve); + + var curve = curves_1[curve].curve; + this.curve = curve; + this.g = curve.g; + this.g.precompute(curve.n.bitLength() + 1); + + this.pointClass = curve.point().constructor; + this.encodingLength = Math.ceil(curve.n.bitLength() / 8); + this.hash = hash_1.sha512; + } + + var eddsa$1 = EDDSA; + + /** + * @param {Array|String} message - message bytes + * @param {Array|String|KeyPair} secret - secret bytes or a keypair + * @returns {Signature} - signature + */ + EDDSA.prototype.sign = function sign(message, secret) { + message = parseBytes$2(message); + var key = this.keyFromSecret(secret); + var r = this.hashInt(key.messagePrefix(), message); + var R = this.g.mul(r); + var Rencoded = this.encodePoint(R); + var s_ = this.hashInt(Rencoded, key.pubBytes(), message) + .mul(key.priv()); + var S = r.add(s_).umod(this.curve.n); + return this.makeSignature({ R: R, S: S, Rencoded: Rencoded }); + }; + + /** + * @param {Array} message - message bytes + * @param {Array|String|Signature} sig - sig bytes + * @param {Array|String|Point|KeyPair} pub - public key + * @returns {Boolean} - true if public key matches sig of message + */ + EDDSA.prototype.verify = function verify(message, sig, pub) { + message = parseBytes$2(message); + sig = this.makeSignature(sig); + var key = this.keyFromPublic(pub); + var h = this.hashInt(sig.Rencoded(), key.pubBytes(), message); + var SG = this.g.mul(sig.S()); + var RplusAh = sig.R().add(key.pub().mul(h)); + return RplusAh.eq(SG); + }; + + EDDSA.prototype.hashInt = function hashInt() { + var hash = this.hash(); + for (var i = 0; i < arguments.length; i++) + hash.update(arguments[i]); + return utils_1$1.intFromLE(hash.digest()).umod(this.curve.n); + }; + + EDDSA.prototype.keyPair = function keyPair(options) { + return new key$1(this, options); + }; + + EDDSA.prototype.keyFromPublic = function keyFromPublic(pub) { + return key$1.fromPublic(this, pub); + }; + + EDDSA.prototype.keyFromSecret = function keyFromSecret(secret) { + return key$1.fromSecret(this, secret); + }; + + EDDSA.prototype.genKeyPair = function genKeyPair(options) { + if (!options) + options = {}; + + // Instantiate Hmac_DRBG + var drbg = new hmacDrbg({ + hash: this.hash, + pers: options.pers, + persEnc: options.persEnc || 'utf8', + entropy: options.entropy || brorand(this.hash.hmacStrength), + entropyEnc: options.entropy && options.entropyEnc || 'utf8', + nonce: this.curve.n.toArray() + }); + + return this.keyFromSecret(drbg.generate(32)); + }; + + EDDSA.prototype.makeSignature = function makeSignature(sig) { + if (sig instanceof signature$2) + return sig; + return new signature$2(this, sig); + }; + + /** + * * https://tools.ietf.org/html/draft-josefsson-eddsa-ed25519-03#section-5.2 + * + * EDDSA defines methods for encoding and decoding points and integers. These are + * helper convenience methods, that pass along to utility functions implied + * parameters. + * + */ + EDDSA.prototype.encodePoint = function encodePoint(point) { + var enc = point.getY().toArray('le', this.encodingLength); + enc[this.encodingLength - 1] |= point.getX().isOdd() ? 0x80 : 0; + return enc; + }; + + EDDSA.prototype.decodePoint = function decodePoint(bytes) { + bytes = utils_1$1.parseBytes(bytes); + + var lastIx = bytes.length - 1; + var normed = bytes.slice(0, lastIx).concat(bytes[lastIx] & ~0x80); + var xIsOdd = (bytes[lastIx] & 0x80) !== 0; + + var y = utils_1$1.intFromLE(normed); + return this.curve.pointFromY(y, xIsOdd); + }; + + EDDSA.prototype.encodeInt = function encodeInt(num) { + return num.toArray('le', this.encodingLength); + }; + + EDDSA.prototype.decodeInt = function decodeInt(bytes) { + return utils_1$1.intFromLE(bytes); + }; + + EDDSA.prototype.isPoint = function isPoint(val) { + return val instanceof this.pointClass; + }; + + var elliptic_1 = createCommonjsModule(function (module, exports) { + + var elliptic = exports; + + elliptic.utils = utils_1$1; + elliptic.rand = brorand; + elliptic.curve = curve_1; + elliptic.curves = curves_1; + + // Protocols + elliptic.ec = ec; + elliptic.eddsa = eddsa$1; + }); + + var elliptic$1 = /*#__PURE__*/Object.freeze({ + __proto__: null, + 'default': elliptic_1, + __moduleExports: elliptic_1 + }); + + exports.AEADEncryptedDataPacket = AEADEncryptedDataPacket; + exports.CleartextMessage = CleartextMessage; + exports.CompressedDataPacket = CompressedDataPacket; + exports.LiteralDataPacket = LiteralDataPacket; + exports.MarkerPacket = MarkerPacket; + exports.Message = Message; + exports.OnePassSignaturePacket = OnePassSignaturePacket; + exports.PacketList = PacketList; + exports.PrivateKey = PrivateKey; + exports.PublicKey = PublicKey; + exports.PublicKeyEncryptedSessionKeyPacket = PublicKeyEncryptedSessionKeyPacket; + exports.PublicKeyPacket = PublicKeyPacket; + exports.PublicSubkeyPacket = PublicSubkeyPacket; + exports.SecretKeyPacket = SecretKeyPacket; + exports.SecretSubkeyPacket = SecretSubkeyPacket; + exports.Signature = Signature; + exports.SignaturePacket = SignaturePacket; + exports.Subkey = Subkey; + exports.SymEncryptedIntegrityProtectedDataPacket = SymEncryptedIntegrityProtectedDataPacket; + exports.SymEncryptedSessionKeyPacket = SymEncryptedSessionKeyPacket; + exports.SymmetricallyEncryptedDataPacket = SymmetricallyEncryptedDataPacket; + exports.TrustPacket = TrustPacket; + exports.UnparseablePacket = UnparseablePacket; + exports.UserAttributePacket = UserAttributePacket; + exports.UserIDPacket = UserIDPacket; + exports.armor = armor; + exports.config = config; + exports.createCleartextMessage = createCleartextMessage; + exports.createMessage = createMessage; + exports.decrypt = decrypt$4; + exports.decryptKey = decryptKey; + exports.decryptSessionKeys = decryptSessionKeys; + exports.encrypt = encrypt$4; + exports.encryptKey = encryptKey; + exports.encryptSessionKey = encryptSessionKey; + exports.enums = enums; + exports.generateKey = generateKey; + exports.generateSessionKey = generateSessionKey$1; + exports.readCleartextMessage = readCleartextMessage; + exports.readKey = readKey; + exports.readKeys = readKeys; + exports.readMessage = readMessage; + exports.readPrivateKey = readPrivateKey; + exports.readPrivateKeys = readPrivateKeys; + exports.readSignature = readSignature; + exports.reformatKey = reformatKey; + exports.revokeKey = revokeKey; + exports.sign = sign$5; + exports.unarmor = unarmor; + exports.verify = verify$5; + + Object.defineProperty(exports, '__esModule', { value: true }); + + return exports; + +}({})); diff --git a/nationchains/www/adminapx/static/js/openpgp.min.js b/nationchains/www/cdn/share/js/openpgp.min.js similarity index 100% rename from nationchains/www/adminapx/static/js/openpgp.min.js rename to nationchains/www/cdn/share/js/openpgp.min.js diff --git a/nationchains/www/adminapx/static/js/openpgp.min.js.map b/nationchains/www/cdn/share/js/openpgp.min.js.map similarity index 100% rename from nationchains/www/adminapx/static/js/openpgp.min.js.map rename to nationchains/www/cdn/share/js/openpgp.min.js.map diff --git a/nationchains/www/cdn/share/logo/favicon.png b/nationchains/www/cdn/share/logo/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..599a05ccc9c65c325d9e030e79eb0061d1d5dbb7 GIT binary patch literal 3785 zcmaJ^c{CK>+n`0001JYGP>3 z9E+H(@XTpupWn5)%^cVvCQcCm0FU-R?F1+@a)Jp8J}`28U=!x|0Okd~2Y|s~ih-fQ z5k6j!dx~LD|H3sbApn5e-qcXvHo9oF_#wpAPPAXQW3nnn%}T9ggFwz(TA%f8Sz^V@D^ew%Xgt5XSy`%AcymgMG2IfFwSbwN zriuBhzZRcdnS+f?leCeqhE=~z(+{uicRhTxL)zLGMb2q(^QVX(gtB?!6l%%x#_y7v zFZKzIc=H8oq}=^4;HuK(iG1yu-rz#`U!S~jC%e(nr{#Kxr4|TlY~mH}GKj4UAaBK6 zqkKp6v8c6z@hb*zsOlpz`MT>97gG9I%hP>S@0sBs@mjiPi#tplAD;h1{J#KG>X<#sP!!E@phcGSW{Hp8NCC}+(8#kMmB%GU*Y z-K^1tCL`apvMRceJHzuy^(rW&xLGDi-$tYh2PQ3NKG++!y@5f0C<3-xsi1Q;I(4 zhvdG{|AKeIRh8x8G5cYiQj74(dsbh$RCH>N?(@)v>gSxk$BI3@#eHy01}_E2Pb1>L zg31C}`H1bj6;UGJk61!t3=d~~HU#vxSsT}5*C~R}1`_gNd{1DYAqz3}DaYSEY;^gA zwYK4>epv&6+7AAkafsOFc%c(N#BBSGK*qhBC1mG`59vRp5V5`h9rSs)3!~4sNIC@V z1gyY~?p^;5kt^*f>uTeg*pOd2h+A$*VSH?_xxA9u>RU*6hzXWmiMbSY05;dC*hnWA zNw;XfIHDy-33Uvq-UTH)tuw!FWh=?1kBpiUlW}IQU8B1ao>oLuxv2QXYk; z#M^vM{U>J^O?iUd0&++0DU9nHZd5St_REzWipRKtZC0pvEJ(THTd+{8<%n&R79+UcK*)91)2c zs$nU`_~Oj98o9nou>r6n{uV0-{{S^Dp{?W%(F1tg#lsV_rQ2fu1+UPJtZmmJ6wjCZ zt*80TBX@HsJy?FRdMGb7*269fxxOp^hm*r9ip3Ury`h7yJDH0Iwt#46`ubw3)7%8V z*9U{M2U^R17t&*M`6S(0ekov2d~-c{0hu*%AHmZ;9JyceBEZCA%J2_GRiK1T1^j|w z{&f|8d#r$j}*-Kg`GH(JAxk>h_!^N+Jme4A3tYH$sDEZA8v_>c02DZUY~j%S!Zo0cjqlcc zdsb{jp2_%j`~H|d=H+tz(<$~cGP=Dy-*5ECCuZse2}5fjxN~JI5}DpM?MFR(E33w}3XXN8 zxuIaie0EhINDVkDyAO~}sZKc$&%R?Kj9ZkLJC5zAm(Ddn7v*XbKNnIMs-8fBssod^ z7)STE{nd>vOm`Z}+5LH^fyVr$0l6%~x5sLv02DrXaBe=bO$)^lKD#t9B^3I&{IsUF zg(Fr}ulADnA1$xxw?fXzQK&E%2h@d5>tXFveh5>r*+d zZyS1XcmdVLbu@Nw0{4n|w(v7}Ye{ zRp2(f$faI9mZ;_`o1>VNps0dy7bqB6>J?J7Ki<7z+>~GCmsNuxe_mVlDZr zxD|g**mx1%PCiix_OGp*+1bPd#fd7Uu8KX&TwU9MOHW!Le=JCW^zGw8glfwhh*&WC zucctS4l|l3twIUymeHExh<#)3s~5>T@jeoUz7uH)P1;`Sc&g^w&f+j+kIT42-w%TM zpY;Ofb3Z(inwpkAH-)H(iurwp2&q;^v)(j|;b>w9MmNt}f0T=- z4hjIhf%zPT^xSDysS9x3$F0>!yc2W7i{>cn8H-k3ahDCsNg@QN)WUg_WNaT|s^~@t zx)_YcK_r5@6-cX5k{qIT`={R}6{h}XD_eAHtI&ASoz}Rg(^g@^8H_f3(8b|`AV=}d zHP4xT-|C#Y`Rz0`TXPR^0sFdPqcVqeSQ6cIA_ zj&gSi?gkDc3fPjNhvU$KaJY-_er~{Lp-}n#pEmpcrgyD?Y(LA7w0Hh`8COsgRaD%Q zJdGcq1$__-RlQ>Vq*ocf5jq>wCwAz8g-UCg7$a0a+BJ1OQDfejC#QtNlFLM)S!`|_ z{}>qsahf9HNim*-!%RhelSeEoar)q}&;C*~td;1ymO9tEekPo*B?I6S`ctQ%aNNp_ z`VQC2xOS4}oC!VPN~)UcYY&Tf6AOjiO)h!EyRK$4hH|c;*OH6Vz9I>2Y(6jNAPo@Qhkuk($VFR^CXzv-y}8HW2wnB zh|1)B_=v-HOV>}C&gK>XCJ-GUo!4dx)Z9o?5#&=;CjpGM4K`9G(9UkIP6nYup@iwt zHl}=X%q#UGoSbd;r`bIc^GQnF-o%hpNF_bk5#2#*u40mNnRYvqlMENE8)LAEqmA9w zfzTmH!c^%tw`P}KV69xGXT=9ik9bXwJVHt{&C)3oPhrOK#=)uHE@(mfpQD@d$+J>1 z)4zhY0~aMQKIm39&*S5H#zHZxie(-_I1K?`DeGCc&&fRD2&*foig-p+B6a6`P0c~! zeQ8#(#G;vKKBB>##b*1dRn+Ft)y^>V=-PMd@UrtgnN_C<_Za8zMxq9(uc^`Yln07R z+f&}xBvHdRNms8`2Wz{OZ*AA-7k@snZPMU-_cO+ONAKnDE-?#?tpm8KDSFnvW#;H< z@bQfYUE)>hi$iexXi0D1b5#%>0bunWhHDEmp}qrI!_qTi4?EPt59?vT3FaRkX9#X+ zJiXkIO)#gD-j7B%cdqH(ZYfx0p;%ynbQ_z?=~mB>x*lKJOc8^x4aQ}Zc4{`;Me)@} zTuJMtWD&at!Q~g(wH-oc1r)~DH%D;oaFz)@qzv31e5ztuWkTihi)(D%@DCxps*zpB;DeM~6Gj4=N+5C=!l7*kI88 zS|fv8S){i;brAfj)8O9oJt1X{#_uy7dtAX0(a0jAMzp_)=o%l-28d9M((ytm(NdaV zV_)Uc`v-&V)SrFAGL|_@XI~CS{&hvuQlrbv*#D%IsCB;R?R-2!qDyWjTx2b_{r>zu zhSGaDW^7}d?-Um}dCn>#(^X6A*wBZXu_Z1oDJ(irYbY>^@$NQYpAKu>jv-*+rTP}fT4E+k|wKa zN_wY*$9t1Lt}YXL8jataBgm#D&s$*oWKa`?1|KtW!seB62mW7O;^SH4j##HuZ=nB; z=VKy@B9TkN&tGc2F;Xdm1euSd3a#}4-9*}!Etn;UuV9qP`+uvkIuHuKIM;&pTh>av zI^^j1YS*b1FX$5O`7YoScQt>-e=n(4)F(}Z{0nR%&b%V-)a`iNGs{qbsS(()-oW$8 Fe*kCWg7N?W literal 0 HcmV?d00001 diff --git a/nationchains/www/cdn/share/logo/logobgdark.png b/nationchains/www/cdn/share/logo/logobgdark.png new file mode 100644 index 0000000000000000000000000000000000000000..89d274ce92173c621f499b2b956e56067f8859d3 GIT binary patch literal 7188 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 + + + + + + + + + + + + + + + + + + + + + + apXtrib + BLOCKCHAIN OF DEMOCRACY + + diff --git a/nationchains/www/cdn/share/logo/logobglight.png b/nationchains/www/cdn/share/logo/logobglight.png new file mode 100644 index 0000000000000000000000000000000000000000..882c9fc089c9cc0a7b5637fd31faa6271e163b75 GIT binary patch literal 10193 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 Kp+?Rk;y(cY8-5%B literal 0 HcmV?d00001 diff --git a/nationchains/www/cdn/share/logo/logobglight.svg b/nationchains/www/cdn/share/logo/logobglight.svg new file mode 100644 index 0000000..1e9a259 --- /dev/null +++ b/nationchains/www/cdn/share/logo/logobglight.svg @@ -0,0 +1,156 @@ + + + + + + + + + + + + + + + + + + + + + apXtrib + BLOCKCHAIN OF DEMOCRACY + + + diff --git a/nationchains/www/cdn/share/logo/logocarredark.png b/nationchains/www/cdn/share/logo/logocarredark.png new file mode 100644 index 0000000000000000000000000000000000000000..e27c2179eeabf3a9f73a1d251646b04e5d846969 GIT binary patch literal 4606 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 literal 0 HcmV?d00001 diff --git a/nationchains/www/cdn/share/logo/logocarrelight.png b/nationchains/www/cdn/share/logo/logocarrelight.png new file mode 100644 index 0000000000000000000000000000000000000000..c49780384e77270b7ef2abfe81b0969ccf5dc579 GIT binary patch literal 4255 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