From 1d1df6f1bc3adb950c75df9ebee4c51cad947970 Mon Sep 17 00:00:00 2001 From: aybanda Date: Mon, 23 Sep 2024 16:22:13 +0530 Subject: [PATCH] Migrate to Manifest V3 --- Gruntfile.js | 88 +- package-lock.json | 4435 ++++++++++++++++++++++++++++++++++++++++++ src/js/background.js | 1789 +---------------- src/js/gsStorage.js | 472 +++-- src/js/gsUtils.js | 66 +- src/manifest.json | 56 +- 6 files changed, 4857 insertions(+), 2049 deletions(-) create mode 100644 package-lock.json diff --git a/Gruntfile.js b/Gruntfile.js index 6f2c194f..032f1301 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -14,58 +14,31 @@ module.exports = function(grunt) { }, copy: { main: { - expand: true, - src: ['src/**', '!src/tests.html', '!src/js/tests/**'], - dest: '<%= config.tempDir %>', + files: [ + // Copy other files and directories + { expand: true, cwd: 'src/', src: ['**'], dest: 'build/' }, + // Copy manifest.json to the root of the build directory + { expand: true, cwd: 'src/', src: ['manifest.json'], dest: 'build/' }, + // Copy background.js to the root of the build directory + { + expand: true, + cwd: 'src/js/', + src: ['background.js'], + dest: 'build/', + }, + ], }, }, 'string-replace': { debugoff: { files: { - '<%= config.tempDir %>src/js/': - '<%= config.tempDir %>src/js/gsUtils.js', - }, - options: { - replacements: [ - { - pattern: /debugInfo\s*=\s*true/, - replacement: 'debugInfo = false', - }, - { - pattern: /debugError\s*=\s*true/, - replacement: 'debugError = false', - }, - ], - }, - }, - debugon: { - files: { - '<%= config.tempDir %>src/js/': - '<%= config.tempDir %>src/js/gsUtils.js', - }, - options: { - replacements: [ - { - pattern: /debugInfo\s*=\s*false/, - replacement: 'debugInfo = true', - }, - { - pattern: /debugError\s*=\s*false/, - replacement: 'debugError = true', - }, - ], - }, - }, - localesTgut: { - files: { - '<%= config.tempDir %>src/_locales/': - '<%= config.tempDir %>src/_locales/**', + 'build/': 'build/**/*', }, options: { replacements: [ { - pattern: /The Great Suspender/gi, - replacement: 'The Great Tester', + pattern: /console\.log\(.+?\);/g, + replacement: '', }, ], }, @@ -73,39 +46,28 @@ module.exports = function(grunt) { }, crx: { public: { - src: [ - '<%= config.tempDir %>src/**/*', - '!**/html2canvas.js', - '!**/Thumbs.db', - ], - dest: 'build/zip/<%= config.buildName %>.zip', + src: 'build/**/*', + dest: 'build/zip/tgs-<%= pkg.version %>.zip', }, private: { - src: [ - '<%= config.tempDir %>src/**/*', - '!**/html2canvas.js', - '!**/Thumbs.db', - ], - dest: 'build/crx/<%= config.buildName %>.crx', + src: 'build/**/*', + dest: 'build/crx/tgs-<%= pkg.version %>.crx', options: { privateKey: 'key.pem', }, }, }, - clean: ['<%= config.tempDir %>'], + clean: { + build: ['build/'], + }, }); grunt.loadNpmTasks('grunt-contrib-copy'); grunt.loadNpmTasks('grunt-string-replace'); grunt.loadNpmTasks('grunt-crx'); grunt.loadNpmTasks('grunt-contrib-clean'); - grunt.registerTask('default', [ - 'copy', - 'string-replace:debugoff', - 'crx:public', - 'crx:private', - 'clean', - ]); + + grunt.registerTask('default', ['clean', 'copy', 'string-replace', 'crx']); grunt.registerTask('tgut', [ 'copy', 'string-replace:debugon', diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 00000000..3d1cc16d --- /dev/null +++ b/package-lock.json @@ -0,0 +1,4435 @@ +{ + "name": "themarvelloussuspender", + "version": "0.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "themarvelloussuspender", + "version": "0.0.0", + "license": "GPLv2", + "dependencies": { + "db.js": "^0.15.0" + }, + "devDependencies": { + "eslint": "^4.19.1", + "eslint-config-prettier": "^2.9.0", + "eslint-config-standard": "^10.2.1", + "eslint-plugin-import": "^2.22.1", + "eslint-plugin-node": "^5.1.1", + "eslint-plugin-promise": "^3.5.0", + "eslint-plugin-standard": "^3.0.1", + "grunt": "^1.3.0", + "grunt-cli": "^1.2.0", + "grunt-contrib-clean": "^1.1.0", + "grunt-contrib-copy": "^1.0.0", + "grunt-crx": "~2.0.0", + "grunt-string-replace": "^1.3.1", + "prettier": "1.13.7", + "time-grunt": "~1.2.1" + } + }, + "node_modules/@rtsao/scc": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@rtsao/scc/-/scc-1.1.0.tgz", + "integrity": "sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==", + "dev": true + }, + "node_modules/@types/json5": { + "version": "0.0.29", + "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", + "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==", + "dev": true + }, + "node_modules/abbrev": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", + "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", + "dev": true + }, + "node_modules/acorn": { + "version": "5.7.4", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.7.4.tgz", + "integrity": "sha512-1D++VG7BhrtvQpNbBzovKNc1FLGGEE/oGe7b9xJm/RFHMBeUaUGpluV9RLjZa47YFdPcDAenEYuq9pQPcMdLJg==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-3.0.1.tgz", + "integrity": "sha512-AU7pnZkguthwBjKgCg6998ByQNIMjbuDQZ8bb78QAFZwPfmKia8AIzgY/gWgqCjnht8JLdXmB4YxA0KaV60ncQ==", + "dev": true, + "dependencies": { + "acorn": "^3.0.4" + } + }, + "node_modules/acorn-jsx/node_modules/acorn": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz", + "integrity": "sha512-OLUyIIZ7mF5oaAUT1w0TFqQS81q3saT46x8t7ukpPjMNk+nbs4ZHhs7ToV8EWnLYLepjETXd4XaCE4uxkMeqUw==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/ajv": { + "version": "5.5.2", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz", + "integrity": "sha512-Ajr4IcMXq/2QmMkEmSvxqfLN5zGmJ92gHXAeOXq1OekoH2rfDNsgdDoL2f7QaRCy7G/E6TpxBVdRuNraMztGHw==", + "dev": true, + "dependencies": { + "co": "^4.6.0", + "fast-deep-equal": "^1.0.0", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.3.0" + } + }, + "node_modules/ajv-keywords": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-2.1.1.tgz", + "integrity": "sha512-ZFztHzVRdGLAzJmpUT9LNFLe1YiVOEylcaNpEutM26PVTCtOD919IMfD01CgbRouB42Dd9atjx1HseC15DgOZA==", + "dev": true, + "peerDependencies": { + "ajv": "^5.0.0" + } + }, + "node_modules/ansi-escapes": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz", + "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/archiver": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/archiver/-/archiver-3.1.1.tgz", + "integrity": "sha512-5Hxxcig7gw5Jod/8Gq0OneVgLYET+oNHcxgWItq4TbhOzRLKNAFUb9edAftiMKXvXfCB0vbGrJdZDNq0dWMsxg==", + "dev": true, + "dependencies": { + "archiver-utils": "^2.1.0", + "async": "^2.6.3", + "buffer-crc32": "^0.2.1", + "glob": "^7.1.4", + "readable-stream": "^3.4.0", + "tar-stream": "^2.1.0", + "zip-stream": "^2.1.2" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/archiver-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/archiver-utils/-/archiver-utils-2.1.0.tgz", + "integrity": "sha512-bEL/yUb/fNNiNTuUz979Z0Yg5L+LzLxGJz8x79lYmR54fmTIb6ob/hNQgkQnIUDWIFjZVQwl9Xs356I6BAMHfw==", + "dev": true, + "dependencies": { + "glob": "^7.1.4", + "graceful-fs": "^4.2.0", + "lazystream": "^1.0.0", + "lodash.defaults": "^4.2.0", + "lodash.difference": "^4.5.0", + "lodash.flatten": "^4.4.0", + "lodash.isplainobject": "^4.0.6", + "lodash.union": "^4.6.0", + "normalize-path": "^3.0.0", + "readable-stream": "^2.0.0" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/archiver/node_modules/async": { + "version": "2.6.4", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz", + "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==", + "dev": true, + "dependencies": { + "lodash": "^4.17.14" + } + }, + "node_modules/archiver/node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/array-buffer-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.1.tgz", + "integrity": "sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "is-array-buffer": "^3.0.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array-each": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/array-each/-/array-each-1.0.1.tgz", + "integrity": "sha512-zHjL5SZa68hkKHBFBK6DJCTtr9sfTCPCaph/L7tMSLcTFgy+zX7E+6q5UArbtOtMBCtxdICpfTCspRse+ywyXA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/array-includes": { + "version": "3.1.8", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.8.tgz", + "integrity": "sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.4", + "is-string": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array-slice": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/array-slice/-/array-slice-1.1.0.tgz", + "integrity": "sha512-B1qMD3RBP7O8o0H2KbrXDyB0IccejMF15+87Lvlor12ONPRHP6gTjXMNkt/d3ZuOGbAe66hFmaCfECI24Ufp6w==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/array.prototype.findlastindex": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.5.tgz", + "integrity": "sha512-zfETvRFA8o7EiNn++N5f/kaCw221hrpGsDmcpndVupkPzEc1Wuf3VgC0qby1BbHs7f5DVYjgtEU2LLh5bqeGfQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "es-shim-unscopables": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.flat": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.2.tgz", + "integrity": "sha512-djYB+Zx2vLewY8RWlNCUdHjDXs2XOgm602S9E7P/UpHgfeHL00cRiIF+IN/G/aUJ7kGPb6yO/ErDI5V2s8iycA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "es-shim-unscopables": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.flatmap": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.2.tgz", + "integrity": "sha512-Ewyx0c9PmpcsByhSW4r+9zDU7sGjFc86qf/kKtuSCRdhfbk0SNLLkaT5qvcHnRGgc5NP/ly/y+qkXkqONX54CQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "es-shim-unscopables": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/arraybuffer.prototype.slice": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.3.tgz", + "integrity": "sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==", + "dev": true, + "dependencies": { + "array-buffer-byte-length": "^1.0.1", + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "es-abstract": "^1.22.3", + "es-errors": "^1.2.1", + "get-intrinsic": "^1.2.3", + "is-array-buffer": "^3.0.4", + "is-shared-array-buffer": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/asn1": { + "version": "0.2.6", + "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.6.tgz", + "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==", + "dev": true, + "dependencies": { + "safer-buffer": "~2.1.0" + } + }, + "node_modules/async": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz", + "integrity": "sha512-nSVgobk4rv61R9PUSDtYt7mPVB2olxNR5RWJcAsH676/ef11bUZwvu7+RGYrYauVdDPcO519v68wRhXQtxsV9w==", + "dev": true + }, + "node_modules/available-typed-arrays": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", + "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", + "dev": true, + "dependencies": { + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha512-XqYMR2dfdGMW+hd0IUZ2PwK+fGeFkOxZJ0wY+JaQAHzt1Zx8LcvpiZD2NiGkEG8qx0CfkAOr5xt76d1e8vG90g==", + "dev": true, + "dependencies": { + "chalk": "^1.1.3", + "esutils": "^2.0.2", + "js-tokens": "^3.0.2" + } + }, + "node_modules/babel-code-frame/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "node_modules/base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/bl": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "dev": true, + "dependencies": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, + "node_modules/bl/node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, + "node_modules/buffer-crc32": { + "version": "0.2.13", + "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", + "integrity": "sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true + }, + "node_modules/call-bind": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", + "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", + "dev": true, + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "set-function-length": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/caller-path": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz", + "integrity": "sha512-UJiE1otjXPF5/x+T3zTnSFiTOEmJoGTD9HmBoxnCUwho61a2eSNn/VwtwuIBDAo2SEOv1AJ7ARI5gCmohFLu/g==", + "dev": true, + "dependencies": { + "callsites": "^0.2.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/callsites": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz", + "integrity": "sha512-Zv4Dns9IbXXmPkgRRUjAaJQgfN4xX5p6+RQFhWUqscdvvK2xK/ZL8b3IXIJsj+4sD+f24NwnWy2BY8AJ82JB0A==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "dependencies": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/chardet": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.4.2.tgz", + "integrity": "sha512-j/Toj7f1z98Hh2cYo2BVr85EpIRWqUi7rtRSGxh/cqUjqrnJe9l9UE7IUGd2vQ2p+kSHLkSzObQPZPLUC6TQwg==", + "dev": true + }, + "node_modules/circular-json": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.3.tgz", + "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==", + "deprecated": "CircularJSON is in maintenance only, flatted is its successor.", + "dev": true + }, + "node_modules/cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha512-8lgKz8LmCRYZZQDpRyT2m5rKJ08TnU4tR9FFFW2rxpxR1FzWi4PQ/NfyODchAatHaUgnSPVcx/R5w6NuTBzFiw==", + "dev": true, + "dependencies": { + "restore-cursor": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/cli-width": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz", + "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", + "dev": true + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "dev": true, + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true + }, + "node_modules/colors": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.1.2.tgz", + "integrity": "sha512-ENwblkFQpqqia6b++zLD/KUWafYlVY/UNnAp7oz7LY7E924wmpye416wBOmvv/HMWzl8gL1kJlfvId/1Dg176w==", + "dev": true, + "engines": { + "node": ">=0.1.90" + } + }, + "node_modules/commander": { + "version": "2.20.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", + "dev": true + }, + "node_modules/compress-commons": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/compress-commons/-/compress-commons-2.1.1.tgz", + "integrity": "sha512-eVw6n7CnEMFzc3duyFVrQEuY1BlHR3rYsSztyG32ibGMW722i3C6IizEGMFmfMU+A+fALvBIwxN3czffTcdA+Q==", + "dev": true, + "dependencies": { + "buffer-crc32": "^0.2.13", + "crc32-stream": "^3.0.1", + "normalize-path": "^3.0.0", + "readable-stream": "^2.3.6" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "node_modules/concat-stream": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", + "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", + "dev": true, + "engines": [ + "node >= 0.8" + ], + "dependencies": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^2.2.2", + "typedarray": "^0.0.6" + } + }, + "node_modules/core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==", + "dev": true + }, + "node_modules/crc": { + "version": "3.8.0", + "resolved": "https://registry.npmjs.org/crc/-/crc-3.8.0.tgz", + "integrity": "sha512-iX3mfgcTMIq3ZKLIsVFAbv7+Mc10kxabAGQb8HvjA1o3T1PIYprbakQ65d3I+2HGHt6nSKkM9PYjgoJO2KcFBQ==", + "dev": true, + "dependencies": { + "buffer": "^5.1.0" + } + }, + "node_modules/crc32-stream": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/crc32-stream/-/crc32-stream-3.0.1.tgz", + "integrity": "sha512-mctvpXlbzsvK+6z8kJwSJ5crm7yBwrQMTybJzMw1O4lLGJqjlDCXY2Zw7KheiA6XBEcBmfLx1D88mjRGVJtY9w==", + "dev": true, + "dependencies": { + "crc": "^3.4.4", + "readable-stream": "^3.4.0" + }, + "engines": { + "node": ">= 6.9.0" + } + }, + "node_modules/crc32-stream/node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/cross-spawn": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz", + "integrity": "sha512-pTgQJ5KC0d2hcY8eyL1IzlBPYjTkyH72XRZPnLyKus2mBfNjQs3klqbJU2VILqZryAZUt9JOb3h/mWMy23/f5A==", + "dev": true, + "dependencies": { + "lru-cache": "^4.0.1", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, + "node_modules/crx": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/crx/-/crx-5.0.1.tgz", + "integrity": "sha512-n/PzBx/fR1+xZCiJBats9y5zw/a+YBcoJ0ABnUaY56xb1RpXuFhsiCMpNY6WjVtylLzhUUXSWsbitesVg7v2vg==", + "dev": true, + "dependencies": { + "archiver": "^3.0.3", + "commander": "^2.20.0", + "node-rsa": "^1.0.5", + "pbf": "^3.2.0" + }, + "bin": { + "crx": "src/cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/data-view-buffer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.1.tgz", + "integrity": "sha512-0lht7OugA5x3iJLOWFhWK/5ehONdprk0ISXqVFn/NFrDu+cuc8iADFrGQz5BnRK7LLU3JmkbXSxaqX+/mXYtUA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/data-view-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.1.tgz", + "integrity": "sha512-4J7wRJD3ABAzr8wP+OcIcqq2dlUKp4DVflx++hs5h5ZKydWMI6/D/fAot+yh6g2tHh8fLFTvNOaVN357NvSrOQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/data-view-byte-offset": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.0.tgz", + "integrity": "sha512-t/Ygsytq+R995EJ5PZlD4Cu56sWa8InXySaViRzw9apusqsOO2bQP+SbYzAhR0pFKoB+43lYy8rWban9JSuXnA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/date-time": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/date-time/-/date-time-1.1.0.tgz", + "integrity": "sha512-RrxZQ06cdKe7YQ5oqIxs3GMc7W3vXscy7Ds+aZIqmxA59QnVtTiCseA4jbzVUub9xCbo9GuYVZo0OrZLYXnnmw==", + "dev": true, + "dependencies": { + "time-zone": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/dateformat": { + "version": "4.6.3", + "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-4.6.3.tgz", + "integrity": "sha512-2P0p0pFGzHS5EMnhdxQi7aJN+iMheud0UhG4dlE1DLAlvL8JHjJJTX/CSm4JXwV0Ka5nGk3zC5mcb5bUQUxxMA==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/db.js": { + "version": "0.15.0", + "resolved": "https://registry.npmjs.org/db.js/-/db.js-0.15.0.tgz", + "integrity": "sha512-pTMjJG21rfh//qE2EKPgUlJ8JgXctgN4XNjN5hzsL2hfirx4Sw5MFEXRSLtqvFSXs2tMeDF2rOVh8aKQMuvwGw==", + "engines": { + "node": ">=0.8" + } + }, + "node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "node_modules/define-data-property": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", + "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", + "dev": true, + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/define-properties": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", + "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", + "dev": true, + "dependencies": { + "define-data-property": "^1.0.1", + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/detect-file": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/detect-file/-/detect-file-1.0.0.tgz", + "integrity": "sha512-DtCOLG98P007x7wiiOmfI0fi3eIKyWiLTGJ2MDnVi/E04lWGbf+JzrRHMm0rgIIZJGtHpKpbVgLWHrv8xXpc3Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/end-of-stream": { + "version": "1.4.4", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", + "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", + "dev": true, + "dependencies": { + "once": "^1.4.0" + } + }, + "node_modules/es-abstract": { + "version": "1.23.3", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.23.3.tgz", + "integrity": "sha512-e+HfNH61Bj1X9/jLc5v1owaLYuHdeHHSQlkhCBiTK8rBvKaULl/beGMxwrMXjpYrv4pz22BlY570vVePA2ho4A==", + "dev": true, + "dependencies": { + "array-buffer-byte-length": "^1.0.1", + "arraybuffer.prototype.slice": "^1.0.3", + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "data-view-buffer": "^1.0.1", + "data-view-byte-length": "^1.0.1", + "data-view-byte-offset": "^1.0.0", + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "es-set-tostringtag": "^2.0.3", + "es-to-primitive": "^1.2.1", + "function.prototype.name": "^1.1.6", + "get-intrinsic": "^1.2.4", + "get-symbol-description": "^1.0.2", + "globalthis": "^1.0.3", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2", + "has-proto": "^1.0.3", + "has-symbols": "^1.0.3", + "hasown": "^2.0.2", + "internal-slot": "^1.0.7", + "is-array-buffer": "^3.0.4", + "is-callable": "^1.2.7", + "is-data-view": "^1.0.1", + "is-negative-zero": "^2.0.3", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.3", + "is-string": "^1.0.7", + "is-typed-array": "^1.1.13", + "is-weakref": "^1.0.2", + "object-inspect": "^1.13.1", + "object-keys": "^1.1.1", + "object.assign": "^4.1.5", + "regexp.prototype.flags": "^1.5.2", + "safe-array-concat": "^1.1.2", + "safe-regex-test": "^1.0.3", + "string.prototype.trim": "^1.2.9", + "string.prototype.trimend": "^1.0.8", + "string.prototype.trimstart": "^1.0.8", + "typed-array-buffer": "^1.0.2", + "typed-array-byte-length": "^1.0.1", + "typed-array-byte-offset": "^1.0.2", + "typed-array-length": "^1.0.6", + "unbox-primitive": "^1.0.2", + "which-typed-array": "^1.1.15" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/es-define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", + "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==", + "dev": true, + "dependencies": { + "get-intrinsic": "^1.2.4" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "dev": true, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-object-atoms": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.0.0.tgz", + "integrity": "sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==", + "dev": true, + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-set-tostringtag": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.3.tgz", + "integrity": "sha512-3T8uNMC3OQTHkFUsFq8r/BwAXLHvU/9O9mE0fBc/MY5iq/8H7ncvO947LmYA6ldWw9Uh8Yhf25zu6n7nML5QWQ==", + "dev": true, + "dependencies": { + "get-intrinsic": "^1.2.4", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-shim-unscopables": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.0.2.tgz", + "integrity": "sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw==", + "dev": true, + "dependencies": { + "hasown": "^2.0.0" + } + }, + "node_modules/es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, + "dependencies": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/eslint": { + "version": "4.19.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-4.19.1.tgz", + "integrity": "sha512-bT3/1x1EbZB7phzYu7vCr1v3ONuzDtX8WjuM9c0iYxe+cq+pwcKEoQjl7zd3RpC6YOLgnSy3cTN58M2jcoPDIQ==", + "dev": true, + "dependencies": { + "ajv": "^5.3.0", + "babel-code-frame": "^6.22.0", + "chalk": "^2.1.0", + "concat-stream": "^1.6.0", + "cross-spawn": "^5.1.0", + "debug": "^3.1.0", + "doctrine": "^2.1.0", + "eslint-scope": "^3.7.1", + "eslint-visitor-keys": "^1.0.0", + "espree": "^3.5.4", + "esquery": "^1.0.0", + "esutils": "^2.0.2", + "file-entry-cache": "^2.0.0", + "functional-red-black-tree": "^1.0.1", + "glob": "^7.1.2", + "globals": "^11.0.1", + "ignore": "^3.3.3", + "imurmurhash": "^0.1.4", + "inquirer": "^3.0.6", + "is-resolvable": "^1.0.0", + "js-yaml": "^3.9.1", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.3.0", + "lodash": "^4.17.4", + "minimatch": "^3.0.2", + "mkdirp": "^0.5.1", + "natural-compare": "^1.4.0", + "optionator": "^0.8.2", + "path-is-inside": "^1.0.2", + "pluralize": "^7.0.0", + "progress": "^2.0.0", + "regexpp": "^1.0.1", + "require-uncached": "^1.0.3", + "semver": "^5.3.0", + "strip-ansi": "^4.0.0", + "strip-json-comments": "~2.0.1", + "table": "4.0.2", + "text-table": "~0.2.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-config-prettier": { + "version": "2.10.0", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-2.10.0.tgz", + "integrity": "sha512-Mhl90VLucfBuhmcWBgbUNtgBiK955iCDK1+aHAz7QfDQF6wuzWZ6JjihZ3ejJoGlJWIuko7xLqNm8BA5uenKhA==", + "dev": true, + "dependencies": { + "get-stdin": "^5.0.1" + }, + "bin": { + "eslint-config-prettier-check": "bin/cli.js" + }, + "peerDependencies": { + "eslint": ">=3.14.1" + } + }, + "node_modules/eslint-config-standard": { + "version": "10.2.1", + "resolved": "https://registry.npmjs.org/eslint-config-standard/-/eslint-config-standard-10.2.1.tgz", + "integrity": "sha512-UkFojTV1o0GOe1edOEiuI5ccYLJSuNngtqSeClNzhsmG8KPJ+7mRxgtp2oYhqZAK/brlXMoCd+VgXViE0AfyKw==", + "dev": true, + "peerDependencies": { + "eslint": ">=3.19.0", + "eslint-plugin-import": ">=2.2.0", + "eslint-plugin-node": ">=4.2.2", + "eslint-plugin-promise": ">=3.5.0", + "eslint-plugin-standard": ">=3.0.0" + } + }, + "node_modules/eslint-import-resolver-node": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz", + "integrity": "sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==", + "dev": true, + "dependencies": { + "debug": "^3.2.7", + "is-core-module": "^2.13.0", + "resolve": "^1.22.4" + } + }, + "node_modules/eslint-module-utils": { + "version": "2.11.0", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.11.0.tgz", + "integrity": "sha512-gbBE5Hitek/oG6MUVj6sFuzEjA/ClzNflVrLovHi/JgLdC7fiN5gLAY1WIPW1a0V5I999MnsrvVrCOGmmVqDBQ==", + "dev": true, + "dependencies": { + "debug": "^3.2.7" + }, + "engines": { + "node": ">=4" + }, + "peerDependenciesMeta": { + "eslint": { + "optional": true + } + } + }, + "node_modules/eslint-plugin-import": { + "version": "2.30.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.30.0.tgz", + "integrity": "sha512-/mHNE9jINJfiD2EKkg1BKyPyUk4zdnT54YgbOgfjSakWT5oyX/qQLVNTkehyfpcMxZXMy1zyonZ2v7hZTX43Yw==", + "dev": true, + "dependencies": { + "@rtsao/scc": "^1.1.0", + "array-includes": "^3.1.8", + "array.prototype.findlastindex": "^1.2.5", + "array.prototype.flat": "^1.3.2", + "array.prototype.flatmap": "^1.3.2", + "debug": "^3.2.7", + "doctrine": "^2.1.0", + "eslint-import-resolver-node": "^0.3.9", + "eslint-module-utils": "^2.9.0", + "hasown": "^2.0.2", + "is-core-module": "^2.15.1", + "is-glob": "^4.0.3", + "minimatch": "^3.1.2", + "object.fromentries": "^2.0.8", + "object.groupby": "^1.0.3", + "object.values": "^1.2.0", + "semver": "^6.3.1", + "tsconfig-paths": "^3.15.0" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8" + } + }, + "node_modules/eslint-plugin-import/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/eslint-plugin-node": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-node/-/eslint-plugin-node-5.2.1.tgz", + "integrity": "sha512-xhPXrh0Vl/b7870uEbaumb2Q+LxaEcOQ3kS1jtIXanBAwpMre1l5q/l2l/hESYJGEFKuI78bp6Uw50hlpr7B+g==", + "dev": true, + "dependencies": { + "ignore": "^3.3.6", + "minimatch": "^3.0.4", + "resolve": "^1.3.3", + "semver": "5.3.0" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": ">=3.1.0" + } + }, + "node_modules/eslint-plugin-node/node_modules/semver": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.3.0.tgz", + "integrity": "sha512-mfmm3/H9+67MCVix1h+IXTpDwL6710LyHuk7+cWC9T1mE0qz4iHhh6r4hU2wrIT9iTsAAC2XQRvfblL028cpLw==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/eslint-plugin-promise": { + "version": "3.8.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-3.8.0.tgz", + "integrity": "sha512-JiFL9UFR15NKpHyGii1ZcvmtIqa3UTwiDAGb8atSffe43qJ3+1czVGN6UtkklpcJ2DVnqvTMzEKRaJdBkAL2aQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-plugin-standard": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-standard/-/eslint-plugin-standard-3.1.0.tgz", + "integrity": "sha512-fVcdyuKRr0EZ4fjWl3c+gp1BANFJD1+RaWa2UPYfMZ6jCtp5RG00kSaXnK/dE5sYzt4kaWJ9qdxqUfc0d9kX0w==", + "dev": true, + "peerDependencies": { + "eslint": ">=3.19.0" + } + }, + "node_modules/eslint-scope": { + "version": "3.7.3", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-3.7.3.tgz", + "integrity": "sha512-W+B0SvF4gamyCTmUc+uITPY0989iXVfKvhwtmJocTaYoc/3khEHmEmvfY/Gn9HA9VV75jrQECsHizkNw1b68FA==", + "dev": true, + "dependencies": { + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/espree": { + "version": "3.5.4", + "resolved": "https://registry.npmjs.org/espree/-/espree-3.5.4.tgz", + "integrity": "sha512-yAcIQxtmMiB/jL32dzEp2enBeidsB7xWPLNiw3IIkpVds1P+h7qF9YwJq1yUNzp2OKXgAprs4F61ih66UsoD1A==", + "dev": true, + "dependencies": { + "acorn": "^5.5.0", + "acorn-jsx": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "dev": true, + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esquery/node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esrecurse/node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/eventemitter2": { + "version": "0.4.14", + "resolved": "https://registry.npmjs.org/eventemitter2/-/eventemitter2-0.4.14.tgz", + "integrity": "sha512-K7J4xq5xAD5jHsGM5ReWXRTFa3JRGofHiMcVgQ8PRwgWxzjHpMWCIzsmyf60+mh8KLsqYPcjUMa0AC4hd6lPyQ==", + "dev": true + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expand-tilde": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/expand-tilde/-/expand-tilde-2.0.2.tgz", + "integrity": "sha512-A5EmesHW6rfnZ9ysHQjPdJRni0SRar0tjtG5MNtm9n5TUvsYU8oozprtRD4AqHxcZWWlVuAmQo2nWKfN9oyjTw==", + "dev": true, + "dependencies": { + "homedir-polyfill": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", + "dev": true + }, + "node_modules/external-editor": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-2.2.0.tgz", + "integrity": "sha512-bSn6gvGxKt+b7+6TKEv1ZycHleA7aHhRHyAqJyp5pbUFuYYNIzpZnQDk7AsYckyWdEnTeAnay0aCy2aV6iTk9A==", + "dev": true, + "dependencies": { + "chardet": "^0.4.0", + "iconv-lite": "^0.4.17", + "tmp": "^0.0.33" + }, + "engines": { + "node": ">=0.12" + } + }, + "node_modules/external-editor/node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-deep-equal": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz", + "integrity": "sha512-fueX787WZKCV0Is4/T2cyAdM4+x1S3MXXOAhavE1ys/W42SHAPacLTQhucja22QBYrfGw50M2sRiXPtTGv9Ymw==", + "dev": true + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "node_modules/figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha512-Oa2M9atig69ZkfwiApY8F2Yy+tzMbazyvqv21R0NsSC8floSOC09BbT1ITWAdoMGQvJ/aZnR1KMwdx9tvHnTNA==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/file-entry-cache": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz", + "integrity": "sha512-uXP/zGzxxFvFfcZGgBIwotm+Tdc55ddPAzF7iHshP4YGaXMww7rSF9peD9D1sui5ebONg5UobsZv+FfgEpGv/w==", + "dev": true, + "dependencies": { + "flat-cache": "^1.2.1", + "object-assign": "^4.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/file-sync-cmp": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/file-sync-cmp/-/file-sync-cmp-0.1.1.tgz", + "integrity": "sha512-0k45oWBokCqh2MOexeYKpyqmGKG+8mQ2Wd8iawx+uWd/weWJQAZ6SoPybagdCI4xFisag8iAR77WPm4h3pTfxA==", + "dev": true + }, + "node_modules/fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dev": true, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/findup-sync": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-5.0.0.tgz", + "integrity": "sha512-MzwXju70AuyflbgeOhzvQWAvvQdo1XL0A9bVvlXsYcFEBM87WR4OakL4OfZq+QRmr+duJubio+UtNQCPsVESzQ==", + "dev": true, + "dependencies": { + "detect-file": "^1.0.0", + "is-glob": "^4.0.3", + "micromatch": "^4.0.4", + "resolve-dir": "^1.0.1" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/fined": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/fined/-/fined-1.2.0.tgz", + "integrity": "sha512-ZYDqPLGxDkDhDZBjZBb+oD1+j0rA4E0pXY50eplAAOPg2N/gUBSSk5IM1/QhPfyVo19lJ+CvXpqfvk+b2p/8Ng==", + "dev": true, + "dependencies": { + "expand-tilde": "^2.0.2", + "is-plain-object": "^2.0.3", + "object.defaults": "^1.1.0", + "object.pick": "^1.2.0", + "parse-filepath": "^1.0.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/flagged-respawn": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/flagged-respawn/-/flagged-respawn-1.0.1.tgz", + "integrity": "sha512-lNaHNVymajmk0OJMBn8fVUAU1BtDeKIqKoVhk4xAALB57aALg6b4W0MfJ/cUE0g9YBXy5XhSlPIpYIJ7HaY/3Q==", + "dev": true, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/flat-cache": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.3.4.tgz", + "integrity": "sha512-VwyB3Lkgacfik2vhqR4uv2rvebqmDvFu4jlN/C1RzWoJEo8I7z4Q404oiqYCkq41mni8EzQnm95emU9seckwtg==", + "dev": true, + "dependencies": { + "circular-json": "^0.3.1", + "graceful-fs": "^4.1.2", + "rimraf": "~2.6.2", + "write": "^0.2.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/for-each": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", + "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==", + "dev": true, + "dependencies": { + "is-callable": "^1.1.3" + } + }, + "node_modules/for-in": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", + "integrity": "sha512-7EwmXrOjyL+ChxMhmG5lnW9MPt1aIeZEwKhQzoBUdTV0N3zuwWDZYVJatDvZ2OyzPUvdIAZDsCetk3coyMfcnQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/for-own": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/for-own/-/for-own-1.0.0.tgz", + "integrity": "sha512-0OABksIGrxKK8K4kynWkQ7y1zounQxP+CWnyclVwj81KW3vlLlGUx57DKGcP/LH216GzqnstnPocF16Nxs0Ycg==", + "dev": true, + "dependencies": { + "for-in": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fs-constants": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", + "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", + "dev": true + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/function.prototype.name": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.6.tgz", + "integrity": "sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "functions-have-names": "^1.2.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/functional-red-black-tree": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", + "integrity": "sha512-dsKNQNdj6xA3T+QlADDA7mOSlX0qiMINjn0cgr+eGHGsbSHzTabcIogz2+p/iqP1Xs6EP/sS2SbqH+brGTbq0g==", + "dev": true + }, + "node_modules/functions-have-names": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", + "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-intrinsic": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", + "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", + "dev": true, + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3", + "hasown": "^2.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-stdin": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-5.0.1.tgz", + "integrity": "sha512-jZV7n6jGE3Gt7fgSTJoz91Ak5MuTLwMwkoYdjxuJ/AmjIsE1UC03y/IWkZCQGEvVNS9qoRNwy5BCqxImv0FVeA==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/get-symbol-description": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.2.tgz", + "integrity": "sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/getobject": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/getobject/-/getobject-1.0.2.tgz", + "integrity": "sha512-2zblDBaFcb3rB4rF77XVnuINOE2h2k/OnqXAiy0IrTxUfV1iFp3la33oAQVY9pCpWU268WFYVt2t71hlMuLsOg==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/global-modules": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-1.0.0.tgz", + "integrity": "sha512-sKzpEkf11GpOFuw0Zzjzmt4B4UZwjOcG757PPvrfhxcLFbq0wpsgpOqxpxtxFiCG4DtG93M6XRVbF2oGdev7bg==", + "dev": true, + "dependencies": { + "global-prefix": "^1.0.1", + "is-windows": "^1.0.1", + "resolve-dir": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/global-prefix": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-1.0.2.tgz", + "integrity": "sha512-5lsx1NUDHtSjfg0eHlmYvZKv8/nVqX4ckFbM+FrGcQ+04KWcWFo9P5MxPZYSzUvyzmdTbI7Eix8Q4IbELDqzKg==", + "dev": true, + "dependencies": { + "expand-tilde": "^2.0.2", + "homedir-polyfill": "^1.0.1", + "ini": "^1.3.4", + "is-windows": "^1.0.1", + "which": "^1.2.14" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/globalthis": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.4.tgz", + "integrity": "sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==", + "dev": true, + "dependencies": { + "define-properties": "^1.2.1", + "gopd": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gopd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz", + "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==", + "dev": true, + "dependencies": { + "get-intrinsic": "^1.1.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true + }, + "node_modules/grunt": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/grunt/-/grunt-1.6.1.tgz", + "integrity": "sha512-/ABUy3gYWu5iBmrUSRBP97JLpQUm0GgVveDCp6t3yRNIoltIYw7rEj3g5y1o2PGPR2vfTRGa7WC/LZHLTXnEzA==", + "dev": true, + "dependencies": { + "dateformat": "~4.6.2", + "eventemitter2": "~0.4.13", + "exit": "~0.1.2", + "findup-sync": "~5.0.0", + "glob": "~7.1.6", + "grunt-cli": "~1.4.3", + "grunt-known-options": "~2.0.0", + "grunt-legacy-log": "~3.0.0", + "grunt-legacy-util": "~2.0.1", + "iconv-lite": "~0.6.3", + "js-yaml": "~3.14.0", + "minimatch": "~3.0.4", + "nopt": "~3.0.6" + }, + "bin": { + "grunt": "bin/grunt" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/grunt-cli": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/grunt-cli/-/grunt-cli-1.5.0.tgz", + "integrity": "sha512-rILKAFoU0dzlf22SUfDtq2R1fosChXXlJM5j7wI6uoW8gwmXDXzbUvirlKZSYCdXl3LXFbR+8xyS+WFo+b6vlA==", + "dev": true, + "dependencies": { + "grunt-known-options": "~2.0.0", + "interpret": "~1.1.0", + "liftup": "~3.0.1", + "nopt": "~5.0.0", + "v8flags": "^4.0.1" + }, + "bin": { + "grunt": "bin/grunt" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/grunt-cli/node_modules/nopt": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-5.0.0.tgz", + "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", + "dev": true, + "dependencies": { + "abbrev": "1" + }, + "bin": { + "nopt": "bin/nopt.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/grunt-contrib-clean": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/grunt-contrib-clean/-/grunt-contrib-clean-1.1.0.tgz", + "integrity": "sha512-tET+TYTd8vCtKeGwbLjoH8+SdI8ngVzGbPr7vlWkewG7mYYHIccd2Ldxq+PK3DyBp5Www3ugdkfsjoNKUl5MTg==", + "dev": true, + "dependencies": { + "async": "^1.5.2", + "rimraf": "^2.5.1" + }, + "engines": { + "node": ">= 0.10.0" + }, + "peerDependencies": { + "grunt": ">=0.4.5" + } + }, + "node_modules/grunt-contrib-copy": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/grunt-contrib-copy/-/grunt-contrib-copy-1.0.0.tgz", + "integrity": "sha512-gFRFUB0ZbLcjKb67Magz1yOHGBkyU6uL29hiEW1tdQ9gQt72NuMKIy/kS6dsCbV0cZ0maNCb0s6y+uT1FKU7jA==", + "dev": true, + "dependencies": { + "chalk": "^1.1.1", + "file-sync-cmp": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/grunt-contrib-copy/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/grunt-contrib-copy/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/grunt-contrib-copy/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/grunt-contrib-copy/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/grunt-contrib-copy/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/grunt-crx": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/grunt-crx/-/grunt-crx-2.0.0.tgz", + "integrity": "sha512-sy1otIiHk55GRehOnH1GuhRGc2gMgHarYuPtKrUbpX+v6wRCgANM3DbWDCgcUdbNtQgM4dE83i50oK//AAE9Gg==", + "dev": true, + "dependencies": { + "crx": "^5.0.0", + "xtend": "^4.0.1" + }, + "bin": { + "grunt-crx": "bin/grunt-crx" + }, + "engines": { + "node": ">= 12.0.0" + } + }, + "node_modules/grunt-known-options": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/grunt-known-options/-/grunt-known-options-2.0.0.tgz", + "integrity": "sha512-GD7cTz0I4SAede1/+pAbmJRG44zFLPipVtdL9o3vqx9IEyb7b4/Y3s7r6ofI3CchR5GvYJ+8buCSioDv5dQLiA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/grunt-legacy-log": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/grunt-legacy-log/-/grunt-legacy-log-3.0.0.tgz", + "integrity": "sha512-GHZQzZmhyq0u3hr7aHW4qUH0xDzwp2YXldLPZTCjlOeGscAOWWPftZG3XioW8MasGp+OBRIu39LFx14SLjXRcA==", + "dev": true, + "dependencies": { + "colors": "~1.1.2", + "grunt-legacy-log-utils": "~2.1.0", + "hooker": "~0.2.3", + "lodash": "~4.17.19" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/grunt-legacy-log-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/grunt-legacy-log-utils/-/grunt-legacy-log-utils-2.1.0.tgz", + "integrity": "sha512-lwquaPXJtKQk0rUM1IQAop5noEpwFqOXasVoedLeNzaibf/OPWjKYvvdqnEHNmU+0T0CaReAXIbGo747ZD+Aaw==", + "dev": true, + "dependencies": { + "chalk": "~4.1.0", + "lodash": "~4.17.19" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/grunt-legacy-log-utils/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/grunt-legacy-log-utils/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/grunt-legacy-log-utils/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/grunt-legacy-log-utils/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/grunt-legacy-log-utils/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/grunt-legacy-log-utils/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/grunt-legacy-util": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/grunt-legacy-util/-/grunt-legacy-util-2.0.1.tgz", + "integrity": "sha512-2bQiD4fzXqX8rhNdXkAywCadeqiPiay0oQny77wA2F3WF4grPJXCvAcyoWUJV+po/b15glGkxuSiQCK299UC2w==", + "dev": true, + "dependencies": { + "async": "~3.2.0", + "exit": "~0.1.2", + "getobject": "~1.0.0", + "hooker": "~0.2.3", + "lodash": "~4.17.21", + "underscore.string": "~3.3.5", + "which": "~2.0.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/grunt-legacy-util/node_modules/async": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz", + "integrity": "sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==", + "dev": true + }, + "node_modules/grunt-legacy-util/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/grunt-string-replace": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/grunt-string-replace/-/grunt-string-replace-1.3.3.tgz", + "integrity": "sha512-PvtdAsvOjGDoQjo0uVZkwFXpApjBQVjiV5eKEQPgHpXJEwXvtMTOJMvk2oNrQcAht/OaGlBsBejK5R28Sfwjww==", + "deprecated": "Package no longer supported. Contact Support at https://www.npmjs.com/support for more info.", + "dev": true, + "dependencies": { + "async": "^3.2.3", + "chalk": "^4.1.2" + }, + "bin": { + "grunt-string-replace": "bin/grunt-string-replace" + }, + "engines": { + "node": ">= 0.10.0", + "npm": ">= 1.4.15" + } + }, + "node_modules/grunt-string-replace/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/grunt-string-replace/node_modules/async": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz", + "integrity": "sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==", + "dev": true + }, + "node_modules/grunt-string-replace/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/grunt-string-replace/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/grunt-string-replace/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/grunt-string-replace/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/grunt-string-replace/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/grunt/node_modules/glob": { + "version": "7.1.7", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", + "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/grunt/node_modules/grunt-cli": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/grunt-cli/-/grunt-cli-1.4.3.tgz", + "integrity": "sha512-9Dtx/AhVeB4LYzsViCjUQkd0Kw0McN2gYpdmGYKtE2a5Yt7v1Q+HYZVWhqXc/kGnxlMtqKDxSwotiGeFmkrCoQ==", + "dev": true, + "dependencies": { + "grunt-known-options": "~2.0.0", + "interpret": "~1.1.0", + "liftup": "~3.0.1", + "nopt": "~4.0.1", + "v8flags": "~3.2.0" + }, + "bin": { + "grunt": "bin/grunt" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/grunt/node_modules/grunt-cli/node_modules/nopt": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-4.0.3.tgz", + "integrity": "sha512-CvaGwVMztSMJLOeXPrez7fyfObdZqNUK1cPAEzLHrTybIua9pMdmmPR5YwtfNftIOMv3DPUhFaxsZMNTQO20Kg==", + "dev": true, + "dependencies": { + "abbrev": "1", + "osenv": "^0.1.4" + }, + "bin": { + "nopt": "bin/nopt.js" + } + }, + "node_modules/grunt/node_modules/minimatch": { + "version": "3.0.8", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.8.tgz", + "integrity": "sha512-6FsRAQsxQ61mw+qP1ZzbL9Bc78x2p5OqNgNpnoAFLTrX8n5Kxph0CsnhmKKNXTWjXqU5L0pGPR7hYk+XWZr60Q==", + "dev": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/grunt/node_modules/v8flags": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/v8flags/-/v8flags-3.2.0.tgz", + "integrity": "sha512-mH8etigqMfiGWdeXpaaqGfs6BndypxusHHcv2qSHyZkGEznCd/qAXCWWRzeowtL54147cktFOC4P5y+kl8d8Jg==", + "dev": true, + "dependencies": { + "homedir-polyfill": "^1.0.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha512-C8vBJ8DwUCx19vhm7urhTuUsr4/IyP6l4VzNQDv+ryHQObW3TTTp9yB68WpYgRe2bbaGuZ/se74IqFeVnMnLZg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-ansi/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-bigints": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", + "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/has-property-descriptors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", + "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", + "dev": true, + "dependencies": { + "es-define-property": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-proto": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz", + "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-tostringtag": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", + "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", + "dev": true, + "dependencies": { + "has-symbols": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/homedir-polyfill": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/homedir-polyfill/-/homedir-polyfill-1.0.3.tgz", + "integrity": "sha512-eSmmWE5bZTK2Nou4g0AI3zZ9rswp7GRKoKXS1BLUkvPviOqs4YTN1djQIqrXy9k5gEtdLPy86JjRwsNM9tnDcA==", + "dev": true, + "dependencies": { + "parse-passwd": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/hooker": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/hooker/-/hooker-0.2.3.tgz", + "integrity": "sha512-t+UerCsQviSymAInD01Pw+Dn/usmz1sRO+3Zk1+lx8eg+WKpD2ulcwWqHHL0+aseRBr+3+vIhiG1K1JTwaIcTA==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", + "dev": true, + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "node_modules/ini": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", + "dev": true + }, + "node_modules/inquirer": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-3.3.0.tgz", + "integrity": "sha512-h+xtnyk4EwKvFWHrUYsWErEVR+igKtLdchu+o0Z1RL7VU/jVMFbYir2bp6bAj8efFNxWqHX0dIss6fJQ+/+qeQ==", + "dev": true, + "dependencies": { + "ansi-escapes": "^3.0.0", + "chalk": "^2.0.0", + "cli-cursor": "^2.1.0", + "cli-width": "^2.0.0", + "external-editor": "^2.0.4", + "figures": "^2.0.0", + "lodash": "^4.3.0", + "mute-stream": "0.0.7", + "run-async": "^2.2.0", + "rx-lite": "^4.0.8", + "rx-lite-aggregates": "^4.0.8", + "string-width": "^2.1.0", + "strip-ansi": "^4.0.0", + "through": "^2.3.6" + } + }, + "node_modules/internal-slot": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.7.tgz", + "integrity": "sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g==", + "dev": true, + "dependencies": { + "es-errors": "^1.3.0", + "hasown": "^2.0.0", + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/interpret": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.1.0.tgz", + "integrity": "sha512-CLM8SNMDu7C5psFCn6Wg/tgpj/bKAg7hc2gWqcuR9OD5Ft9PhBpIu8PLicPeis+xDd6YX2ncI8MCA64I9tftIA==", + "dev": true + }, + "node_modules/is-absolute": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-absolute/-/is-absolute-1.0.0.tgz", + "integrity": "sha512-dOWoqflvcydARa360Gvv18DZ/gRuHKi2NU/wU5X1ZFzdYfH29nkiNZsF3mp4OJ3H4yo9Mx8A/uAGNzpzPN3yBA==", + "dev": true, + "dependencies": { + "is-relative": "^1.0.0", + "is-windows": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-array-buffer": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.4.tgz", + "integrity": "sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-bigint": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", + "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==", + "dev": true, + "dependencies": { + "has-bigints": "^1.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-boolean-object": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz", + "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-callable": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", + "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-core-module": { + "version": "2.15.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.15.1.tgz", + "integrity": "sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==", + "dev": true, + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-data-view": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.1.tgz", + "integrity": "sha512-AHkaJrsUVW6wq6JS8y3JnM/GJF/9cf+k20+iDzlSaJrinEo5+7vRiteOSwBhHRiAyQATN1AmY4hwzxJKPmYf+w==", + "dev": true, + "dependencies": { + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-date-object": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz", + "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==", + "dev": true, + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-finite": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-finite/-/is-finite-1.1.0.tgz", + "integrity": "sha512-cdyMtqX/BOqqNBBiKlIVkytNHm49MtMlYyn1zxzvJKWmFMlGzm+ry5BBfYyeY9YmNKbRSo/o7OX9w9ale0wg3w==", + "dev": true, + "engines": { + "node": ">=0.10.0" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-negative-zero": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", + "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-number-object": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.7.tgz", + "integrity": "sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==", + "dev": true, + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-plain-object": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", + "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", + "dev": true, + "dependencies": { + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-relative": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-relative/-/is-relative-1.0.0.tgz", + "integrity": "sha512-Kw/ReK0iqwKeu0MITLFuj0jbPAmEiOsIwyIXvvbfa6QfmN9pkD1M+8pdk7Rl/dTKbH34/XBFMbgD4iMJhLQbGA==", + "dev": true, + "dependencies": { + "is-unc-path": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-resolvable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", + "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", + "dev": true + }, + "node_modules/is-shared-array-buffer": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.3.tgz", + "integrity": "sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-string": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dev": true, + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-symbol": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dev": true, + "dependencies": { + "has-symbols": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-typed-array": { + "version": "1.1.13", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.13.tgz", + "integrity": "sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==", + "dev": true, + "dependencies": { + "which-typed-array": "^1.1.14" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-unc-path": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-unc-path/-/is-unc-path-1.0.0.tgz", + "integrity": "sha512-mrGpVd0fs7WWLfVsStvgF6iEJnbjDFZh9/emhRDcGWTduTfNHd9CHeUwH3gYIjdbwo4On6hunkztwOaAw0yllQ==", + "dev": true, + "dependencies": { + "unc-path-regex": "^0.1.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-weakref": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", + "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "dev": true + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "node_modules/isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/js-tokens": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", + "integrity": "sha512-RjTcuD4xjtthQkaWH7dFlH85L+QaVtSoOyGdZ3g6HFhS9dFNDfLyqgm2NFe2X6cQpeFmt0452FJjFG5UameExg==", + "dev": true + }, + "node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/json-schema-traverse": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz", + "integrity": "sha512-4JD/Ivzg7PoW8NzdrBSr3UFwC9mHgvI7Z6z3QGBsSHgKaRTUDmyZAAKJo2UbG1kUVfS9WS8bi36N49U1xw43DA==", + "dev": true + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "node_modules/json5": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", + "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", + "dev": true, + "dependencies": { + "minimist": "^1.2.0" + }, + "bin": { + "json5": "lib/cli.js" + } + }, + "node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/lazystream": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lazystream/-/lazystream-1.0.1.tgz", + "integrity": "sha512-b94GiNHQNy6JNTrt5w6zNyffMrNkXZb3KTkCZJb2V1xaEGCk093vkZ2jk3tpaeP33/OiXC+WvK9AxUebnf5nbw==", + "dev": true, + "dependencies": { + "readable-stream": "^2.0.5" + }, + "engines": { + "node": ">= 0.6.3" + } + }, + "node_modules/levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==", + "dev": true, + "dependencies": { + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/liftup": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/liftup/-/liftup-3.0.1.tgz", + "integrity": "sha512-yRHaiQDizWSzoXk3APcA71eOI/UuhEkNN9DiW2Tt44mhYzX4joFoCZlxsSOF7RyeLlfqzFLQI1ngFq3ggMPhOw==", + "dev": true, + "dependencies": { + "extend": "^3.0.2", + "findup-sync": "^4.0.0", + "fined": "^1.2.0", + "flagged-respawn": "^1.0.1", + "is-plain-object": "^2.0.4", + "object.map": "^1.0.1", + "rechoir": "^0.7.0", + "resolve": "^1.19.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/liftup/node_modules/findup-sync": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-4.0.0.tgz", + "integrity": "sha512-6jvvn/12IC4quLBL1KNokxC7wWTvYncaVUYSoxWw7YykPLuRrnv4qdHcSOywOI5RpkOVGeQRtWM8/q+G6W6qfQ==", + "dev": true, + "dependencies": { + "detect-file": "^1.0.0", + "is-glob": "^4.0.0", + "micromatch": "^4.0.2", + "resolve-dir": "^1.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "dev": true + }, + "node_modules/lodash.defaults": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz", + "integrity": "sha512-qjxPLHd3r5DnsdGacqOMU6pb/avJzdh9tFX2ymgoZE27BmjXrNy/y4LoaiTeAb+O3gL8AfpJGtqfX/ae2leYYQ==", + "dev": true + }, + "node_modules/lodash.difference": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz", + "integrity": "sha512-dS2j+W26TQ7taQBGN8Lbbq04ssV3emRw4NY58WErlTO29pIqS0HmoT5aJ9+TUQ1N3G+JOZSji4eugsWwGp9yPA==", + "dev": true + }, + "node_modules/lodash.flatten": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz", + "integrity": "sha512-C5N2Z3DgnnKr0LOpv/hKCgKdb7ZZwafIrsesve6lmzvZIRZRGaZ/l6Q8+2W7NaT+ZwO3fFlSCzCzrDCFdJfZ4g==", + "dev": true + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==", + "dev": true + }, + "node_modules/lodash.union": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/lodash.union/-/lodash.union-4.6.0.tgz", + "integrity": "sha512-c4pB2CdGrGdjMKYLA+XiRDO7Y0PRQbm/Gzg8qMj+QH+pFVAoTp5sBpO0odL3FjoPCGjK96p6qsP+yQoiLoOBcw==", + "dev": true + }, + "node_modules/lru-cache": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.5.tgz", + "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "dev": true, + "dependencies": { + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" + } + }, + "node_modules/make-iterator": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/make-iterator/-/make-iterator-1.0.1.tgz", + "integrity": "sha512-pxiuXh0iVEq7VM7KMIhs5gxsfxCux2URptUQaXo4iZZJxBAzTPOLE2BumO5dbfVYq/hBJFBR/a1mFDmOx5AGmw==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/map-cache": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", + "integrity": "sha512-8y/eV9QQZCiyn1SprXSrCmqJN0yNRATe+PO8ztwqrvrbdRLA3eYJF0yaR0YayLWkMbsQSKWS9N2gPcGEc4UsZg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "dev": true, + "dependencies": { + "minimist": "^1.2.6" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/mute-stream": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha512-r65nCZhrbXXb6dXOACihYApHw2Q6pV0M3V0PSxd74N0+D8nzAdEAITq2oAjA1jVnKI+tGvEBUpqiMh0+rW6zDQ==", + "dev": true + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "node_modules/node-rsa": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/node-rsa/-/node-rsa-1.1.1.tgz", + "integrity": "sha512-Jd4cvbJMryN21r5HgxQOpMEqv+ooke/korixNNK3mGqfGJmy0M77WDDzo/05969+OkMy3XW1UuZsSmW9KQm7Fw==", + "dev": true, + "dependencies": { + "asn1": "^0.2.4" + } + }, + "node_modules/nopt": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-3.0.6.tgz", + "integrity": "sha512-4GUt3kSEYmk4ITxzB/b9vaIDfUVWN/Ml1Fwl11IlnIG2iaJ9O6WXZ9SrYM9NLI8OCBieN2Y8SWC2oJV0RQ7qYg==", + "dev": true, + "dependencies": { + "abbrev": "1" + }, + "bin": { + "nopt": "bin/nopt.js" + } + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/number-is-nan": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", + "integrity": "sha512-4jbtZXNAsfZbAHiiqjLPBiCl16dES1zI4Hpzzxw61Tk+loF+sBDBKx1ICKKKwIqQ7M0mFn1TmkN7euSncWgHiQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-inspect": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.2.tgz", + "integrity": "sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.assign": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.5.tgz", + "integrity": "sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "has-symbols": "^1.0.3", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.defaults": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/object.defaults/-/object.defaults-1.1.0.tgz", + "integrity": "sha512-c/K0mw/F11k4dEUBMW8naXUuBuhxRCfG7W+yFy8EcijU/rSmazOUd1XAEEe6bC0OuXY4HUKjTJv7xbxIMqdxrA==", + "dev": true, + "dependencies": { + "array-each": "^1.0.1", + "array-slice": "^1.0.0", + "for-own": "^1.0.0", + "isobject": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object.fromentries": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", + "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.groupby": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.3.tgz", + "integrity": "sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/object.map/-/object.map-1.0.1.tgz", + "integrity": "sha512-3+mAJu2PLfnSVGHwIWubpOFLscJANBKuB/6A4CxBstc4aqwQY0FWcsppuy4jU5GSB95yES5JHSI+33AWuS4k6w==", + "dev": true, + "dependencies": { + "for-own": "^1.0.0", + "make-iterator": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object.pick": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", + "integrity": "sha512-tqa/UMy/CCoYmj+H5qc07qvSL9dqcs/WZENZ1JbtWBlATP+iVOe778gE6MSijnyCnORzDuX6hU+LA4SZ09YjFQ==", + "dev": true, + "dependencies": { + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object.values": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.0.tgz", + "integrity": "sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha512-oyyPpiMaKARvvcgip+JV+7zci5L8D1W9RZIz2l1o08AM3pfspitVWnPt3mzHcBPp12oYMTy0pqrFs/C+m3EwsQ==", + "dev": true, + "dependencies": { + "mimic-fn": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/optionator": { + "version": "0.8.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", + "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", + "dev": true, + "dependencies": { + "deep-is": "~0.1.3", + "fast-levenshtein": "~2.0.6", + "levn": "~0.3.0", + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2", + "word-wrap": "~1.2.3" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/os-homedir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", + "integrity": "sha512-B5JU3cabzk8c67mRRd3ECmROafjYMXbuzlwtqdM8IbS8ktlTix8aFGb2bAGKrSRIlnfKwovGUUr72JUPyOb6kQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/osenv": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/osenv/-/osenv-0.1.5.tgz", + "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==", + "deprecated": "This package is no longer supported.", + "dev": true, + "dependencies": { + "os-homedir": "^1.0.0", + "os-tmpdir": "^1.0.0" + } + }, + "node_modules/parse-filepath": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/parse-filepath/-/parse-filepath-1.0.2.tgz", + "integrity": "sha512-FwdRXKCohSVeXqwtYonZTXtbGJKrn+HNyWDYVcp5yuJlesTwNH4rsmRZ+GrKAPJ5bLpRxESMeS+Rl0VCHRvB2Q==", + "dev": true, + "dependencies": { + "is-absolute": "^1.0.0", + "map-cache": "^0.2.0", + "path-root": "^0.1.1" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/parse-ms": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parse-ms/-/parse-ms-1.0.1.tgz", + "integrity": "sha512-LpH1Cf5EYuVjkBvCDBYvkUPh+iv2bk3FHflxHkpCYT0/FZ1d3N3uJaLiHr4yGuMcFUhv6eAivitTvWZI4B/chg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/parse-passwd": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/parse-passwd/-/parse-passwd-1.0.0.tgz", + "integrity": "sha512-1Y1A//QUXEZK7YKz+rD9WydcE1+EuPr6ZBgKecAB8tmoW6UFv0NREVJe1p+jRxtThkcbbKkfwIbWJe/IeE6m2Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha512-DUWJr3+ULp4zXmol/SZkFf3JGsS9/SIv+Y3Rt93/UjPpDpklB5f1er4O3POIbUuUJ3FXgqte2Q7SrU6zAqwk8w==", + "dev": true + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "node_modules/path-root": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/path-root/-/path-root-0.1.1.tgz", + "integrity": "sha512-QLcPegTHF11axjfojBIoDygmS2E3Lf+8+jI6wOVmNVenrKSo3mFdSGiIgdSHenczw3wPtlVMQaFVwGmM7BJdtg==", + "dev": true, + "dependencies": { + "path-root-regex": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-root-regex": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/path-root-regex/-/path-root-regex-0.1.2.tgz", + "integrity": "sha512-4GlJ6rZDhQZFE0DPVKh0e9jmZ5egZfxTkp7bcRDuPlJXbAwhxcl2dINPUAsjLdejqaLsCeg8axcLjIbvBjN4pQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/pbf": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/pbf/-/pbf-3.3.0.tgz", + "integrity": "sha512-XDF38WCH3z5OV/OVa8GKUNtLAyneuzbCisx7QUCF8Q6Nutx0WnJrQe5O+kOtBlLfRNUws98Y58Lblp+NJG5T4Q==", + "dev": true, + "dependencies": { + "ieee754": "^1.1.12", + "resolve-protobuf-schema": "^2.1.0" + }, + "bin": { + "pbf": "bin/pbf" + } + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/plur": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/plur/-/plur-1.0.0.tgz", + "integrity": "sha512-qSnKBSZeDY8ApxwhfVIwKwF36KVJqb1/9nzYYq3j3vdwocULCXT8f8fQGkiw1Nk9BGfxiDagEe/pwakA+bOBqw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/pluralize": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-7.0.0.tgz", + "integrity": "sha512-ARhBOdzS3e41FbkW/XWrTEtukqqLoK5+Z/4UeDaLuSW+39JPeFgs4gCGqsrJHVZX0fUrx//4OF0K1CUGwlIFow==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/possible-typed-array-names": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", + "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", + "dev": true, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/prelude-ls": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", + "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/prettier": { + "version": "1.13.7", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-1.13.7.tgz", + "integrity": "sha512-KIU72UmYPGk4MujZGYMFwinB7lOf2LsDNGSOC8ufevsrPLISrZbNJlWstRi3m0AMuszbH+EFSQ/r6w56RSPK6w==", + "dev": true, + "bin": { + "prettier": "bin-prettier.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/pretty-ms": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/pretty-ms/-/pretty-ms-2.1.0.tgz", + "integrity": "sha512-H2enpsxzDhuzRl3zeSQpQMirn8dB0Z/gxW96j06tMfTviUWvX14gjKb7qd1gtkUyYhDPuoNe00K5PqNvy2oQNg==", + "dev": true, + "dependencies": { + "is-finite": "^1.0.1", + "parse-ms": "^1.0.0", + "plur": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", + "dev": true + }, + "node_modules/progress": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", + "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/protocol-buffers-schema": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/protocol-buffers-schema/-/protocol-buffers-schema-3.6.0.tgz", + "integrity": "sha512-TdDRD+/QNdrCGCE7v8340QyuXd4kIWIgapsE2+n/SaGiSSbomYl4TjHlvIoCWRpE7wFt02EpB35VVA2ImcBVqw==", + "dev": true + }, + "node_modules/pseudomap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", + "integrity": "sha512-b/YwNhb8lk1Zz2+bXXpS/LK9OisiZZ1SNsSLxN1x2OXVEhW2Ckr/7mWE5vrC1ZTiJlD9g19jWszTmJsB+oEpFQ==", + "dev": true + }, + "node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "dev": true, + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/rechoir": { + "version": "0.7.1", + "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.7.1.tgz", + "integrity": "sha512-/njmZ8s1wVeR6pjTZ+0nCnv8SpZNRMT2D1RLOJQESlYFDBvwpTA4KWJpZ+sBJ4+vhjILRcK7JIFdGCdxEAAitg==", + "dev": true, + "dependencies": { + "resolve": "^1.9.0" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/regexp.prototype.flags": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.2.tgz", + "integrity": "sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.6", + "define-properties": "^1.2.1", + "es-errors": "^1.3.0", + "set-function-name": "^2.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/regexpp": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-1.1.0.tgz", + "integrity": "sha512-LOPw8FpgdQF9etWMaAfG/WRthIdXJGYp4mJ2Jgn/2lpkbod9jPn0t9UqN7AxBOKNfzRbYyVfgc7Vk4t/MpnXgw==", + "dev": true, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/require-uncached": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz", + "integrity": "sha512-Xct+41K3twrbBHdxAgMoOS+cNcoqIjfM2/VxBF4LL2hVph7YsF8VSKyQ3BDFZwEVbok9yeDl2le/qo0S77WG2w==", + "dev": true, + "dependencies": { + "caller-path": "^0.1.0", + "resolve-from": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve": { + "version": "1.22.8", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", + "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", + "dev": true, + "dependencies": { + "is-core-module": "^2.13.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-dir": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/resolve-dir/-/resolve-dir-1.0.1.tgz", + "integrity": "sha512-R7uiTjECzvOsWSfdM0QKFNBVFcK27aHOUwdvK53BcW8zqnGdYp0Fbj82cy54+2A4P2tFM22J5kRfe1R+lM/1yg==", + "dev": true, + "dependencies": { + "expand-tilde": "^2.0.0", + "global-modules": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve-from": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz", + "integrity": "sha512-kT10v4dhrlLNcnO084hEjvXCI1wUG9qZLoz2RogxqDQQYy7IxjI/iMUkOtQTNEh6rzHxvdQWHsJyel1pKOVCxg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve-protobuf-schema": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/resolve-protobuf-schema/-/resolve-protobuf-schema-2.1.0.tgz", + "integrity": "sha512-kI5ffTiZWmJaS/huM8wZfEMer1eRd7oJQhDuxeCLe3t7N7mX3z94CN0xPxBQxFYQTSNz9T0i+v6inKqSdK8xrQ==", + "dev": true, + "dependencies": { + "protocol-buffers-schema": "^3.3.1" + } + }, + "node_modules/restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha512-6IzJLuGi4+R14vwagDHX+JrXmPVtPpn4mffDJ1UdR7/Edm87fl6yi8mMBIVvFtJaNTUvjughmW4hwLhRG7gC1Q==", + "dev": true, + "dependencies": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "deprecated": "Rimraf versions prior to v4 are no longer supported", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/run-async": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.1.tgz", + "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/rx-lite": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/rx-lite/-/rx-lite-4.0.8.tgz", + "integrity": "sha512-Cun9QucwK6MIrp3mry/Y7hqD1oFqTYLQ4pGxaHTjIdaFDWRGGLikqp6u8LcWJnzpoALg9hap+JGk8sFIUuEGNA==", + "dev": true + }, + "node_modules/rx-lite-aggregates": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/rx-lite-aggregates/-/rx-lite-aggregates-4.0.8.tgz", + "integrity": "sha512-3xPNZGW93oCjiO7PtKxRK6iOVYBWBvtf9QHDfU23Oc+dLIQmAV//UnyXV/yihv81VS/UqoQPk4NegS8EFi55Hg==", + "dev": true, + "dependencies": { + "rx-lite": "*" + } + }, + "node_modules/safe-array-concat": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.2.tgz", + "integrity": "sha512-vj6RsCsWBCf19jIeHEfkRMw8DPiBb+DMXklQ/1SGDHOMlHdPUkZXFQ2YdplS23zESTijAcurb1aSgJA3AgMu1Q==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "get-intrinsic": "^1.2.4", + "has-symbols": "^1.0.3", + "isarray": "^2.0.5" + }, + "engines": { + "node": ">=0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safe-array-concat/node_modules/isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "dev": true + }, + "node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + }, + "node_modules/safe-regex-test": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.3.tgz", + "integrity": "sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-regex": "^1.1.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "dev": true + }, + "node_modules/semver": { + "version": "5.7.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz", + "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/set-function-length": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", + "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", + "dev": true, + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/set-function-name": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", + "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", + "dev": true, + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "functions-have-names": "^1.2.3", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha512-EV3L1+UQWGor21OmnvojK36mhg+TyIKDh3iFBKBohr5xeXIhNBcx8oWdgkTEEQ+BEFFYdLRuqMfd5L84N1V5Vg==", + "dev": true, + "dependencies": { + "shebang-regex": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha512-wpoSFAxys6b2a2wHZ1XpDSgD7N9iVjg29Ph9uV/uaP9Ex/KXlkTZTeddxDPSYQpgvzKLGJke2UU0AzoGCjNIvQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/side-channel": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz", + "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", + "object-inspect": "^1.13.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true + }, + "node_modules/slice-ansi": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-1.0.0.tgz", + "integrity": "sha512-POqxBK6Lb3q6s047D/XsDVNPnF9Dl8JSaqe9h9lURl0OdNqy/ujDrOiIHtsqXMGbWWTIomRzAMaTyawAU//Reg==", + "dev": true, + "dependencies": { + "is-fullwidth-code-point": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", + "dev": true + }, + "node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dev": true, + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "node_modules/string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "dependencies": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/string.prototype.trim": { + "version": "1.2.9", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.9.tgz", + "integrity": "sha512-klHuCNxiMZ8MlsOihJhJEBJAiMVqU3Z2nEXWfWnIqjN0gEFS9J9+IxKozWWtQGcgoa1WUZzLjKPTr4ZHNFTFxw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.0", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trimend": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.8.tgz", + "integrity": "sha512-p73uL5VCHCO2BZZ6krwwQE3kCzM7NKmis8S//xEC6fQonchbum4eP6kR4DLEjQFO3Wnj3Fuo8NM0kOSjVdHjZQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trimstart": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", + "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==", + "dev": true, + "dependencies": { + "ansi-regex": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/table": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/table/-/table-4.0.2.tgz", + "integrity": "sha512-UUkEAPdSGxtRpiV9ozJ5cMTtYiqz7Ni1OGqLXRCynrvzdtR1p+cfOWe2RJLwvUG8hNanaSRjecIqwOjqeatDsA==", + "dev": true, + "dependencies": { + "ajv": "^5.2.3", + "ajv-keywords": "^2.1.0", + "chalk": "^2.1.0", + "lodash": "^4.17.4", + "slice-ansi": "1.0.0", + "string-width": "^2.1.1" + } + }, + "node_modules/tar-stream": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", + "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", + "dev": true, + "dependencies": { + "bl": "^4.0.3", + "end-of-stream": "^1.4.1", + "fs-constants": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.1.1" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/tar-stream/node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "node_modules/through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==", + "dev": true + }, + "node_modules/time-grunt": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/time-grunt/-/time-grunt-1.2.2.tgz", + "integrity": "sha512-9+rIDE5k28ELgy3gV0SPe61kwYrisvtfm6z5gL9Gr3hB7+tcXz9zsYgbH9KWdgo+KbASTTeNQKZIbAY5JqNaKg==", + "dev": true, + "dependencies": { + "chalk": "^1.0.0", + "date-time": "^1.0.0", + "figures": "^1.0.0", + "hooker": "^0.2.3", + "number-is-nan": "^1.0.0", + "pretty-ms": "^2.1.0", + "text-table": "^0.2.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/time-grunt/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/time-grunt/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/time-grunt/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/time-grunt/node_modules/figures": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-1.7.0.tgz", + "integrity": "sha512-UxKlfCRuCBxSXU4C6t9scbDyWZ4VlaFFdojKtzJuSkuOBQ5CNFum+zZXFwHjo+CxBC1t6zlYPgHIgFjL8ggoEQ==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5", + "object-assign": "^4.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/time-grunt/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/time-grunt/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/time-zone": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/time-zone/-/time-zone-0.1.0.tgz", + "integrity": "sha512-S5CjtVIkeBTnlsaZP3gjsTb78ClBe74sEcgEoBwAVUKnTRDAGqUtLLIZHMsIyqOWjt9DGQpLMMoD8ZKIfP2ddQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "dependencies": { + "os-tmpdir": "~1.0.2" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/tsconfig-paths": { + "version": "3.15.0", + "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz", + "integrity": "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==", + "dev": true, + "dependencies": { + "@types/json5": "^0.0.29", + "json5": "^1.0.2", + "minimist": "^1.2.6", + "strip-bom": "^3.0.0" + } + }, + "node_modules/type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==", + "dev": true, + "dependencies": { + "prelude-ls": "~1.1.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/typed-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.2.tgz", + "integrity": "sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/typed-array-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.1.tgz", + "integrity": "sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-byte-offset": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.2.tgz", + "integrity": "sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==", + "dev": true, + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-length": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.6.tgz", + "integrity": "sha512-/OxDN6OtAk5KBpGb28T+HZc2M+ADtvRxXrKKbUwtsLgdoxgX13hyy7ek6bFRl5+aBs2yZzB0c4CnQfAtVypW/g==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13", + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==", + "dev": true + }, + "node_modules/unbox-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz", + "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "has-bigints": "^1.0.2", + "has-symbols": "^1.0.3", + "which-boxed-primitive": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/unc-path-regex": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/unc-path-regex/-/unc-path-regex-0.1.2.tgz", + "integrity": "sha512-eXL4nmJT7oCpkZsHZUOJo8hcX3GbsiDOa0Qu9F646fi8dT3XuSVopVqAcEiVzSKKH7UoDti23wNX3qGFxcW5Qg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/underscore.string": { + "version": "3.3.6", + "resolved": "https://registry.npmjs.org/underscore.string/-/underscore.string-3.3.6.tgz", + "integrity": "sha512-VoC83HWXmCrF6rgkyxS9GHv8W9Q5nhMKho+OadDJGzL2oDYbYEppBaCMH6pFlwLeqj2QS+hhkw2kpXkSdD1JxQ==", + "dev": true, + "dependencies": { + "sprintf-js": "^1.1.1", + "util-deprecate": "^1.0.2" + }, + "engines": { + "node": "*" + } + }, + "node_modules/underscore.string/node_modules/sprintf-js": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.3.tgz", + "integrity": "sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==", + "dev": true + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", + "dev": true + }, + "node_modules/v8flags": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/v8flags/-/v8flags-4.0.1.tgz", + "integrity": "sha512-fcRLaS4H/hrZk9hYwbdRM35D0U8IYMfEClhXxCivOojl+yTRAZH3Zy2sSy6qVCiGbV9YAtPssP6jaChqC9vPCg==", + "dev": true, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "which": "bin/which" + } + }, + "node_modules/which-boxed-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz", + "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==", + "dev": true, + "dependencies": { + "is-bigint": "^1.0.1", + "is-boolean-object": "^1.1.0", + "is-number-object": "^1.0.4", + "is-string": "^1.0.5", + "is-symbol": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-typed-array": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.15.tgz", + "integrity": "sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==", + "dev": true, + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "node_modules/write": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz", + "integrity": "sha512-CJ17OoULEKXpA5pef3qLj5AxTJ6mSt7g84he2WIskKwqFO4T97d5V7Tadl0DYDk7qyUOQD5WlUlOMChaYrhxeA==", + "dev": true, + "dependencies": { + "mkdirp": "^0.5.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "dev": true, + "engines": { + "node": ">=0.4" + } + }, + "node_modules/yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==", + "dev": true + }, + "node_modules/zip-stream": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/zip-stream/-/zip-stream-2.1.3.tgz", + "integrity": "sha512-EkXc2JGcKhO5N5aZ7TmuNo45budRaFGHOmz24wtJR7znbNqDPmdZtUauKX6et8KAVseAMBOyWJqEpXcHTBsh7Q==", + "dev": true, + "dependencies": { + "archiver-utils": "^2.1.0", + "compress-commons": "^2.1.1", + "readable-stream": "^3.4.0" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/zip-stream/node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + } + } +} diff --git a/src/js/background.js b/src/js/background.js index 111d75c7..81117f36 100644 --- a/src/js/background.js +++ b/src/js/background.js @@ -6,8 +6,24 @@ * http://github.com/greatsuspender/thegreatsuspender * ༼ つ ◕_◕ ༽つ */ -var tgs = (function() { - // eslint-disable-line no-unused-vars + +importScripts( + 'js/gsUtils.js', + 'js/gsChrome.js', + 'js/gsStorage.js', + 'js/db.js', + 'js/gsIndexedDb.js', + 'js/gsMessages.js', + 'js/gsSession.js', + 'js/gsTabQueue.js', + 'js/gsTabCheckManager.js', + 'js/gsFavicon.js', + 'js/gsTabSuspendManager.js', + 'js/gsTabDiscardManager.js', + 'js/gsSuspendedTab.js' +); + +const tgs = (function() { 'use strict'; const ICON_SUSPENSION_ACTIVE = { @@ -83,7 +99,7 @@ var tgs = (function() { retries = retries || 0; if (retries > 300) { // allow 30 seconds :scream: - chrome.tabs.create({ url: chrome.extension.getURL('broken.html') }); + chrome.tabs.create({ url: chrome.runtime.getURL('broken.html') }); return Promise.reject('Failed to initialise background scripts'); } return new Promise(function(resolve) { @@ -94,7 +110,7 @@ var tgs = (function() { return Promise.resolve(); } return new Promise(function(resolve) { - window.setTimeout(resolve, 100); + setTimeout(resolve, 100); }).then(function() { retries += 1; return backgroundScriptsReadyAsPromised(retries); @@ -102,1752 +118,105 @@ var tgs = (function() { }); } - function initAsPromised() { - return new Promise(async function(resolve) { - gsUtils.log('background', 'PERFORMING BACKGROUND INIT...'); - addCommandListeners(); - addMessageListeners(); - addChromeListeners(); - addMiscListeners(); + async function initAsPromised() { + gsUtils.log('background', 'PERFORMING BACKGROUND INIT...'); + addCommandListeners(); + addMessageListeners(); + addChromeListeners(); + addMiscListeners(); - //initialise unsuspended tab props - resetAutoSuspendTimerForAllTabs(); + //initialise unsuspended tab props + resetAutoSuspendTimerForAllTabs(); - //add context menu items - //TODO: Report chrome bug where adding context menu in incognito removes it from main windows - if (!chrome.extension.inIncognitoContext) { - buildContextMenu(false); - var contextMenus = gsStorage.getOption(gsStorage.ADD_CONTEXT); - buildContextMenu(contextMenus); - } + //add context menu items + //TODO: Report chrome bug where adding context menu in incognito removes it from main windows + if (!chrome.extension.inIncognitoContext) { + buildContextMenu(false); + const contextMenus = await new Promise(resolve => + gsStorage.getOption(gsStorage.ADD_CONTEXT, resolve) + ); + buildContextMenu(contextMenus); + } - //initialise currentStationary and currentFocused vars - const activeTabs = await gsChrome.tabsQuery({ active: true }); - const currentWindow = await gsChrome.windowsGetLastFocused(); - for (let activeTab of activeTabs) { - _currentStationaryTabIdByWindowId[activeTab.windowId] = activeTab.id; - _currentFocusedTabIdByWindowId[activeTab.windowId] = activeTab.id; - if (currentWindow && currentWindow.id === activeTab.windowId) { - _currentStationaryWindowId = activeTab.windowId; - _currentFocusedWindowId = activeTab.windowId; - } + //initialise currentStationary and currentFocused vars + const activeTabs = await gsChrome.tabsQuery({ active: true }); + const currentWindow = await gsChrome.windowsGetLastFocused(); + for (let activeTab of activeTabs) { + _currentStationaryTabIdByWindowId[activeTab.windowId] = activeTab.id; + _currentFocusedTabIdByWindowId[activeTab.windowId] = activeTab.id; + if (currentWindow && currentWindow.id === activeTab.windowId) { + _currentStationaryWindowId = activeTab.windowId; + _currentFocusedWindowId = activeTab.windowId; } - gsUtils.log('background', 'init successful'); - resolve(); - }); + } + gsUtils.log('background', 'init successful'); } - function getInternalViewByTabId(tabId) { - const internalViews = chrome.extension.getViews({ tabId: tabId }); + const internalViews = chrome.runtime.getViews({ tabId: tabId }); if (internalViews.length === 1) { return internalViews[0]; } return null; } - function getInternalViewsByViewName(viewName) { - return chrome.extension - .getViews() - .filter(o => o.location.pathname.indexOf(viewName) >= 0); - } - - function getCurrentlyActiveTab(callback) { - // wrap this in an anonymous async function so we can use await - (async function() { - const currentWindowActiveTabs = await gsChrome.tabsQuery({ - active: true, - currentWindow: true, + function buildContextMenu(contextMenus) { + // Example implementation of buildContextMenu + chrome.contextMenus.removeAll(() => { + chrome.contextMenus.create({ + id: 'suspendTab', + title: 'Suspend Tab', + contexts: ['all'], }); - if (currentWindowActiveTabs.length > 0) { - callback(currentWindowActiveTabs[0]); - return; + if (contextMenus) { + // Add additional context menu items based on contextMenus parameter } - - // Fallback on chrome.windows.getLastFocused - const lastFocusedWindow = await gsChrome.windowsGetLastFocused(); - if (lastFocusedWindow) { - const lastFocusedWindowActiveTabs = await gsChrome.tabsQuery({ - active: true, - windowId: lastFocusedWindow.id, - }); - if (lastFocusedWindowActiveTabs.length > 0) { - callback(lastFocusedWindowActiveTabs[0]); - return; - } - } - - // Fallback on _currentStationaryWindowId - if (_currentStationaryWindowId) { - const currentStationaryWindowActiveTabs = await gsChrome.tabsQuery({ - active: true, - windowId: _currentStationaryWindowId, - }); - if (currentStationaryWindowActiveTabs.length > 0) { - callback(currentStationaryWindowActiveTabs[0]); - return; - } - - // Fallback on currentStationaryTabId - const currentStationaryTabId = - _currentStationaryTabIdByWindowId[_currentStationaryWindowId]; - if (currentStationaryTabId) { - const currentStationaryTab = await gsChrome.tabsGet( - currentStationaryTabId, - ); - if (currentStationaryTab !== null) { - callback(currentStationaryTab); - return; - } - } - } - callback(null); - })(); - } - - // NOTE: Stationary here means has had focus for more than focusDelay ms - // So it may not necessarily have the tab.active flag set to true - function isCurrentStationaryTab(tab) { - if (tab.windowId !== _currentStationaryWindowId) { - return false; - } - var lastStationaryTabIdForWindow = - _currentStationaryTabIdByWindowId[tab.windowId]; - if (lastStationaryTabIdForWindow) { - return tab.id === lastStationaryTabIdForWindow; - } else { - // fallback on active flag - return tab.active; - } + }); } - function isCurrentFocusedTab(tab) { - if (tab.windowId !== _currentFocusedWindowId) { - return false; - } - var currentFocusedTabIdForWindow = - _currentFocusedTabIdByWindowId[tab.windowId]; - if (currentFocusedTabIdForWindow) { - return tab.id === currentFocusedTabIdForWindow; - } else { - // fallback on active flag - return tab.active; + function getTabStatePropForTabId(tabId, prop) { + // Example implementation of getTabStatePropForTabId + if (_tabStateByTabId[tabId]) { + return _tabStateByTabId[tabId][prop]; } + return null; } - function isCurrentActiveTab(tab) { - const activeTabIdForWindow = _currentFocusedTabIdByWindowId[tab.windowId]; - if (activeTabIdForWindow) { - return tab.id === activeTabIdForWindow; - } else { - // fallback on active flag - return tab.active; + function clearAutoSuspendTimerForTabId(tabId) { + const timerDetails = getTabStatePropForTabId(tabId, STATE_TIMER_DETAILS); + if (timerDetails && timerDetails.timerId) { + clearTimeout(timerDetails.timerId); + delete timerDetails.timerId; } } - function whitelistHighlightedTab(includePath) { - includePath = includePath || false; - getCurrentlyActiveTab(function(activeTab) { - if (activeTab) { - if (gsUtils.isSuspendedTab(activeTab)) { - let url = gsUtils.getRootUrl( - gsUtils.getOriginalUrl(activeTab.url), - includePath, - false, - ); - gsUtils.saveToWhitelist(url); - unsuspendTab(activeTab); - } else if (gsUtils.isNormalTab(activeTab)) { - let url = gsUtils.getRootUrl(activeTab.url, includePath, false); - gsUtils.saveToWhitelist(url); - calculateTabStatus(activeTab, null, function(status) { - setIconStatus(status, activeTab.id); - }); - } - } - }); - } - - function unwhitelistHighlightedTab(callback) { - getCurrentlyActiveTab(function(activeTab) { - if (activeTab) { - gsUtils.removeFromWhitelist(activeTab.url); - calculateTabStatus(activeTab, null, function(status) { - setIconStatus(status, activeTab.id); - if (callback) callback(status); - }); - } else { - if (callback) callback(gsUtils.STATUS_UNKNOWN); - } - }); - } - - function requestToggleTempWhitelistStateOfHighlightedTab(callback) { - getCurrentlyActiveTab(function(activeTab) { - if (!activeTab) { - if (callback) callback(status); - return; - } - if (gsUtils.isSuspendedTab(activeTab)) { - unsuspendTab(activeTab); - if (callback) callback(gsUtils.STATUS_UNKNOWN); - return; - } - if (!gsUtils.isNormalTab(activeTab, true)) { - if (callback) callback(gsUtils.STATUS_UNKNOWN); - return; - } - - calculateTabStatus(activeTab, null, function(status) { - if ( - status === gsUtils.STATUS_ACTIVE || - status === gsUtils.STATUS_NORMAL - ) { - setTempWhitelistStateForTab(activeTab, callback); - } else if ( - status === gsUtils.STATUS_TEMPWHITELIST || - status === gsUtils.STATUS_FORMINPUT - ) { - unsetTempWhitelistStateForTab(activeTab, callback); - } else { - if (callback) callback(status); - } - }); - }); - } - - function setTempWhitelistStateForTab(tab, callback) { - gsMessages.sendTemporaryWhitelistToContentScript(tab.id, function( - error, - response, - ) { - if (error) { - gsUtils.warning( - tab.id, - 'Failed to sendTemporaryWhitelistToContentScript', - error, - ); - } - var contentScriptStatus = - response && response.status ? response.status : null; - calculateTabStatus(tab, contentScriptStatus, function(newStatus) { - setIconStatus(newStatus, tab.id); - //This is a hotfix for issue #723 - if (newStatus === 'tempWhitelist' && tab.autoDiscardable) { - chrome.tabs.update(tab.id, { - autoDiscardable: false, - }); - } - if (callback) callback(newStatus); - }); - }); - } - - function unsetTempWhitelistStateForTab(tab, callback) { - gsMessages.sendUndoTemporaryWhitelistToContentScript(tab.id, function( - error, - response, - ) { - if (error) { - gsUtils.warning( - tab.id, - 'Failed to sendUndoTemporaryWhitelistToContentScript', - error, - ); - } - var contentScriptStatus = - response && response.status ? response.status : null; - calculateTabStatus(tab, contentScriptStatus, function(newStatus) { - setIconStatus(newStatus, tab.id); - //This is a hotfix for issue #723 - if (newStatus !== 'tempWhitelist' && !tab.autoDiscardable) { - chrome.tabs.update(tab.id, { - //async - autoDiscardable: true, - }); - } - if (callback) callback(newStatus); - }); - }); - } - - function openLinkInSuspendedTab(parentTab, linkedUrl) { - //imitate chromes 'open link in new tab' behaviour in how it selects the correct index - chrome.tabs.query({ windowId: chrome.windows.WINDOW_ID_CURRENT }, tabs => { - var newTabIndex = parentTab.index + 1; - var nextTab = tabs[newTabIndex]; - while (nextTab && nextTab.openerTabId === parentTab.id) { - newTabIndex++; - nextTab = tabs[newTabIndex]; - } - var newTabProperties = { - url: linkedUrl, - index: newTabIndex, - openerTabId: parentTab.id, - active: false, - }; - chrome.tabs.create(newTabProperties, tab => { - gsTabSuspendManager.queueTabForSuspension(tab, 1); - }); - }); - } - - function toggleSuspendedStateOfHighlightedTab() { - getCurrentlyActiveTab(activeTab => { - if (activeTab) { - if (gsUtils.isSuspendedTab(activeTab)) { - unsuspendTab(activeTab); - } else { - gsTabSuspendManager.queueTabForSuspension(activeTab, 1); - } - } - }); - } - - function suspendHighlightedTab() { - getCurrentlyActiveTab(activeTab => { - if (activeTab) { - gsTabSuspendManager.queueTabForSuspension(activeTab, 1); - } - }); - } - - function unsuspendHighlightedTab() { - getCurrentlyActiveTab(activeTab => { - if (activeTab && gsUtils.isSuspendedTab(activeTab)) { - unsuspendTab(activeTab); - } - }); - } - - function suspendAllTabs(force) { - const forceLevel = force ? 1 : 2; - getCurrentlyActiveTab(activeTab => { - if (!activeTab) { - gsUtils.warning( - 'background', - 'Could not determine currently active window.', - ); - return; - } - chrome.windows.get(activeTab.windowId, { populate: true }, curWindow => { - for (const tab of curWindow.tabs) { - if (!tab.active) { - gsTabSuspendManager.queueTabForSuspension(tab, forceLevel); - } - } - }); - }); - } - - function suspendAllTabsInAllWindows(force) { - const forceLevel = force ? 1 : 2; - chrome.tabs.query({}, tabs => { - for (const tab of tabs) { - gsTabSuspendManager.queueTabForSuspension(tab, forceLevel); - } - }); - } - - function unsuspendAllTabs() { - getCurrentlyActiveTab(function(activeTab) { - if (!activeTab) { - gsUtils.warning( - 'background', - 'Could not determine currently active window.', - ); - return; - } - chrome.windows.get(activeTab.windowId, { populate: true }, curWindow => { - for (const tab of curWindow.tabs) { - gsTabSuspendManager.unqueueTabForSuspension(tab); - if (gsUtils.isSuspendedTab(tab)) { - unsuspendTab(tab); - } else if (gsUtils.isNormalTab(tab) && !tab.active) { - resetAutoSuspendTimerForTab(tab); - } - } - }); - }); - } - - function unsuspendAllTabsInAllWindows() { - chrome.windows.getLastFocused({}, currentWindow => { - chrome.tabs.query({}, tabs => { - // Because of the way that unsuspending steals window focus, we defer the suspending of tabs in the - // current window until last - var deferredTabs = []; - for (const tab of tabs) { - gsTabSuspendManager.unqueueTabForSuspension(tab); - if (gsUtils.isSuspendedTab(tab)) { - if (tab.windowId === currentWindow.id) { - deferredTabs.push(tab); - } else { - unsuspendTab(tab); - } - } else if (gsUtils.isNormalTab(tab)) { - resetAutoSuspendTimerForTab(tab); - } - } - for (const tab of deferredTabs) { - unsuspendTab(tab); - } - }); - }); - } - - function suspendSelectedTabs() { - chrome.tabs.query( - { highlighted: true, lastFocusedWindow: true }, - selectedTabs => { - for (const tab of selectedTabs) { - gsTabSuspendManager.queueTabForSuspension(tab, 1); - } - }, - ); - } - - function unsuspendSelectedTabs() { - chrome.tabs.query( - { highlighted: true, lastFocusedWindow: true }, - selectedTabs => { - for (const tab of selectedTabs) { - gsTabSuspendManager.unqueueTabForSuspension(tab); - if (gsUtils.isSuspendedTab(tab)) { - unsuspendTab(tab); - } - } - }, - ); - } - - function queueSessionTimer() { - clearTimeout(_sessionSaveTimer); - _sessionSaveTimer = setTimeout(function() { - gsUtils.log('background', 'updating current session'); - gsSession.updateCurrentSession(); //async - }, 1000); - } - function resetAutoSuspendTimerForTab(tab) { clearAutoSuspendTimerForTabId(tab.id); - - const suspendTime = gsStorage.getOption(gsStorage.SUSPEND_TIME); - const timeToSuspend = suspendTime * (1000 * 60); - if ( - gsUtils.isProtectedActiveTab(tab) || - isNaN(suspendTime) || - suspendTime <= 0 - ) { - return; + const timerDetails = getTabStatePropForTabId(tab.id, STATE_TIMER_DETAILS); + if (timerDetails) { + timerDetails.timerId = setTimeout(() => { + gsTabSuspendManager.queueTabForSuspension(tab, 1); + }, timerDetails.delay); } - - const timerDetails = {}; - timerDetails.tabId = tab.id; - timerDetails.suspendDateTime = new Date( - new Date().getTime() + timeToSuspend, - ); - - timerDetails.timer = setTimeout(async () => { - const updatedTabId = timerDetails.tabId; // This may get updated via updateTabIdReferences - const updatedTab = await gsChrome.tabsGet(updatedTabId); - if (!updatedTab) { - gsUtils.warning(updatedTabId, 'Couldnt find tab. Aborting suspension'); - return; - } - gsTabSuspendManager.queueTabForSuspension(updatedTab, 3); - }, timeToSuspend); - gsUtils.log( - tab.id, - 'Adding tab timer for: ' + timerDetails.suspendDateTime, - ); - - setTabStatePropForTabId(tab.id, STATE_TIMER_DETAILS, timerDetails); } function resetAutoSuspendTimerForAllTabs() { chrome.tabs.query({}, tabs => { for (const tab of tabs) { - if (gsUtils.isNormalTab(tab)) { - resetAutoSuspendTimerForTab(tab); - } - } - }); - } - - function clearAutoSuspendTimerForTabId(tabId) { - const timerDetails = getTabStatePropForTabId(tabId, STATE_TIMER_DETAILS); - if (!timerDetails) { - return; - } - gsUtils.log(tabId, 'Removing tab timer.'); - clearTimeout(timerDetails.timer); - setTabStatePropForTabId(tabId, STATE_TIMER_DETAILS, null); - } - - function getTabStatePropForTabId(tabId, prop) { - return _tabStateByTabId[tabId] ? _tabStateByTabId[tabId][prop] : undefined; - } - - function setTabStatePropForTabId(tabId, prop, value) { - // gsUtils.log(tabId, `Setting tab state prop: ${prop}:`, value); - const tabState = _tabStateByTabId[tabId] || {}; - tabState[prop] = value; - _tabStateByTabId[tabId] = tabState; - } - - function clearTabStateForTabId(tabId) { - gsUtils.log(tabId, 'Clearing tab state props:', _tabStateByTabId[tabId]); - clearAutoSuspendTimerForTabId(tabId); - delete _tabStateByTabId[tabId]; - } - - function unsuspendTab(tab) { - if (!gsUtils.isSuspendedTab(tab)) return; - - const scrollPosition = gsUtils.getSuspendedScrollPosition(tab.url); - tgs.setTabStatePropForTabId(tab.id, tgs.STATE_SCROLL_POS, scrollPosition); - - let originalUrl = gsUtils.getOriginalUrl(tab.url); - if (originalUrl) { - // Reloading chrome.tabs.update causes a history item for the suspended tab - // to be made in the tab history. We clean this up on tab updated hook - setTabStatePropForTabId(tab.id, tgs.STATE_HISTORY_URL_TO_REMOVE, tab.url); - if (tab.autoDiscardable) { - setTabStatePropForTabId(tab.id, tgs.STATE_SET_AUTODISCARDABLE, tab.url); - } - // NOTE: Temporarily disable autoDiscardable, as there seems to be a bug - // where discarded (and frozen?) suspended tabs will not unsuspend with - // chrome.tabs.update if this is set to true. This gets unset again after tab - // has reloaded via the STATE_SET_AUTODISCARDABLE flag. - gsUtils.log(tab.id, 'Unsuspending tab via chrome.tabs.update'); - chrome.tabs.update(tab.id, { url: originalUrl, autoDiscardable: false }); - return; - } - - gsUtils.log(tab.id, 'Failed to execute unsuspend tab.'); - } - - function buildSuspensionToggleHotkey() { - return new Promise(resolve => { - let printableHotkey = ''; - chrome.commands.getAll(commands => { - const toggleCommand = commands.find(o => o.name === '1-suspend-tab'); - if (toggleCommand && toggleCommand.shortcut !== '') { - printableHotkey = gsUtils.formatHotkeyString(toggleCommand.shortcut); - resolve(printableHotkey); - } else { - resolve(null); - } - }); - }); - } - - function checkForTriggerUrls(tab, url) { - // test for a save of keyboard shortcuts (chrome://extensions/shortcuts) - if (url === 'chrome://extensions/shortcuts') { - _triggerHotkeyUpdate = true; - } - } - - function handleUnsuspendedTabStateChanged(tab, changeInfo) { - if ( - !changeInfo.hasOwnProperty('status') && - !changeInfo.hasOwnProperty('audible') && - !changeInfo.hasOwnProperty('pinned') && - !changeInfo.hasOwnProperty('discarded') - ) { - return; - } - gsUtils.log( - tab.id, - 'unsuspended tab state changed. changeInfo: ', - changeInfo, - ); - - // Ensure we clear the STATE_UNLOADED_URL flag during load in case the - // tab is suspended again before loading can finish (in which case on - // suspended tab complete, the tab will reload again) - if ( - changeInfo.hasOwnProperty('status') && - changeInfo.status === 'loading' - ) { - setTabStatePropForTabId(tab.id, STATE_UNLOADED_URL, null); - } - - // Check if tab has just been discarded - if (changeInfo.hasOwnProperty('discarded') && changeInfo.discarded) { - const existingSuspendReason = getTabStatePropForTabId( - tab.id, - STATE_SUSPEND_REASON, - ); - if (existingSuspendReason && existingSuspendReason === 3) { - // For some reason the discarded changeInfo gets called twice (chrome bug?) - // As a workaround we use the suspend reason to determine if we've already - // handled this discard - //TODO: Report chrome bug - return; - } - gsUtils.log( - tab.id, - 'Unsuspended tab has been discarded. Url: ' + tab.url, - ); - gsTabDiscardManager.handleDiscardedUnsuspendedTab(tab); //async. unhandled promise. - - // When a tab is discarded the tab id changes. We need up-to-date UNSUSPENDED - // tabIds in the current session otherwise crash recovery will not work - queueSessionTimer(); - return; - } - - // Check if tab is queued for suspension - const queuedTabDetails = gsTabSuspendManager.getQueuedTabDetails(tab); - if (queuedTabDetails) { - // Requeue tab to wake it from possible sleep - delete queuedTabDetails.executionProps.refetchTab; - gsTabSuspendManager.queueTabForSuspension( - tab, - queuedTabDetails.executionProps.forceLevel, - ); - return; - } - - let hasTabStatusChanged = false; - - // Check for change in tabs audible status - if (changeInfo.hasOwnProperty('audible')) { - //reset tab timer if tab has just finished playing audio - if (!changeInfo.audible && gsStorage.getOption(gsStorage.IGNORE_AUDIO)) { - resetAutoSuspendTimerForTab(tab); - } - hasTabStatusChanged = true; - } - if (changeInfo.hasOwnProperty('pinned')) { - //reset tab timer if tab has become unpinned - if (!changeInfo.pinned && gsStorage.getOption(gsStorage.IGNORE_PINNED)) { resetAutoSuspendTimerForTab(tab); } - hasTabStatusChanged = true; - } - - if (changeInfo.hasOwnProperty('status')) { - if (changeInfo.status === 'complete') { - const tempWhitelistOnReload = getTabStatePropForTabId( - tab.id, - STATE_TEMP_WHITELIST_ON_RELOAD, - ); - const scrollPos = - getTabStatePropForTabId(tab.id, STATE_SCROLL_POS) || null; - const historyUrlToRemove = getTabStatePropForTabId( - tab.id, - STATE_HISTORY_URL_TO_REMOVE, - ); - const setAutodiscardable = getTabStatePropForTabId( - tab.id, - STATE_SET_AUTODISCARDABLE, - ); - clearTabStateForTabId(tab.id); - - if (historyUrlToRemove) { - removeTabHistoryForUnuspendedTab(historyUrlToRemove); - } - if (setAutodiscardable) { - gsChrome.tabsUpdate(tab.id, { autoDiscardable: true }); - } - - //init loaded tab - resetAutoSuspendTimerForTab(tab); - initialiseTabContentScript(tab, tempWhitelistOnReload, scrollPos) - .catch(error => { - gsUtils.warning( - tab.id, - 'Failed to send init to content script. Tab may not behave as expected.', - ); - }) - .then(() => { - // could use returned tab status here below - }); - } - - hasTabStatusChanged = true; - } - - //if tab is currently visible then update popup icon - if (hasTabStatusChanged && isCurrentFocusedTab(tab)) { - calculateTabStatus(tab, null, function(status) { - setIconStatus(status, tab.id); - }); - } - } - - function removeTabHistoryForUnuspendedTab(suspendedUrl) { - chrome.history.deleteUrl({ url: suspendedUrl }); - const originalUrl = gsUtils.getOriginalUrl(suspendedUrl); - chrome.history.getVisits({ url: originalUrl }, visits => { - //assume history entry will be the second to latest one (latest one is the currently visible page) - //NOTE: this will break if the same url has been visited by another tab more recently than the - //suspended tab (pre suspension) - const latestVisit = visits.pop(); - const previousVisit = visits.pop(); - if (previousVisit) { - chrome.history.deleteRange( - { - startTime: previousVisit.visitTime - 0.1, - endTime: previousVisit.visitTime + 0.1, - }, - () => { - }, - ); - } }); } - function initialiseTabContentScript(tab, isTempWhitelist, scrollPos) { - return new Promise((resolve, reject) => { - const ignoreForms = gsStorage.getOption(gsStorage.IGNORE_FORMS); - gsMessages.sendInitTabToContentScript( - tab.id, - ignoreForms, - isTempWhitelist, - scrollPos, - (error, response) => { - if (error) { - reject(error); - } else { - resolve(response); - } - }, - ); - }); - } - - function handleSuspendedTabStateChanged(tab, changeInfo) { - if ( - !changeInfo.hasOwnProperty('status') && - !changeInfo.hasOwnProperty('discarded') - ) { - return; - } - - gsUtils.log( - tab.id, - 'suspended tab status changed. changeInfo: ', - changeInfo, - ); - - if (changeInfo.status && changeInfo.status === 'loading') { - tgs.setTabStatePropForTabId( - tab.id, - tgs.STATE_INITIALISE_SUSPENDED_TAB, - true, - ); - return; - } - - if ( - (changeInfo.status && changeInfo.status === 'complete') || - changeInfo.discarded - ) { - gsTabSuspendManager.unqueueTabForSuspension(tab); //safety precaution - const shouldInitTab = getTabStatePropForTabId( - tab.id, - STATE_INITIALISE_SUSPENDED_TAB, - ); - if (shouldInitTab) { - initialiseSuspendedTab(tab); - } - } - } - - function initialiseSuspendedTab(tab) { - const unloadedUrl = getTabStatePropForTabId(tab.id, STATE_UNLOADED_URL); - const disableUnsuspendOnReload = getTabStatePropForTabId( - tab.id, - STATE_DISABLE_UNSUSPEND_ON_RELOAD, - ); - clearTabStateForTabId(tab.id); - - if (isCurrentFocusedTab(tab)) { - setIconStatus(gsUtils.STATUS_SUSPENDED, tab.id); - } - - //if a suspended tab is marked for unsuspendOnReload then unsuspend tab and return early - const suspendedTabRefreshed = unloadedUrl === tab.url; - if (suspendedTabRefreshed && !disableUnsuspendOnReload) { - unsuspendTab(tab); - return; - } - - const tabView = tgs.getInternalViewByTabId(tab.id); - const quickInit = - gsStorage.getOption(gsStorage.DISCARD_AFTER_SUSPEND) && !tab.active; - gsSuspendedTab - .initTab(tab, tabView, { quickInit }) - .catch(error => { - gsUtils.warning(tab.id, error); - }) - .then(() => { - gsTabCheckManager.queueTabCheck(tab, { refetchTab: true }, 3000); - }); - } - - function updateTabIdReferences(newTabId, oldTabId) { - gsUtils.log(oldTabId, 'update tabId references to ' + newTabId); - for (const windowId of Object.keys(_currentFocusedTabIdByWindowId)) { - if (_currentFocusedTabIdByWindowId[windowId] === oldTabId) { - _currentFocusedTabIdByWindowId[windowId] = newTabId; - } - } - for (const windowId of Object.keys(_currentStationaryTabIdByWindowId)) { - if (_currentStationaryTabIdByWindowId[windowId] === oldTabId) { - _currentStationaryTabIdByWindowId[windowId] = newTabId; - } - } - if (_tabStateByTabId[oldTabId]) { - _tabStateByTabId[newTabId] = _tabStateByTabId[oldTabId]; - delete _tabStateByTabId[oldTabId]; - } - const timerDetails = getTabStatePropForTabId(newTabId, STATE_TIMER_DETAILS); - if (timerDetails) { - timerDetails.tabId = newTabId; - } - } - - function removeTabIdReferences(tabId) { - gsUtils.log(tabId, 'removing tabId references to ' + tabId); - for (const windowId of Object.keys(_currentFocusedTabIdByWindowId)) { - if (_currentFocusedTabIdByWindowId[windowId] === tabId) { - _currentFocusedTabIdByWindowId[windowId] = null; - } - } - for (const windowId of Object.keys(_currentStationaryTabIdByWindowId)) { - if (_currentStationaryTabIdByWindowId[windowId] === tabId) { - _currentStationaryTabIdByWindowId[windowId] = null; - } - } - clearTabStateForTabId(tabId); - } - - async function getSuspensionToggleHotkey() { - if (_suspensionToggleHotkey === undefined) { - _suspensionToggleHotkey = await buildSuspensionToggleHotkey(); - } - return _suspensionToggleHotkey; + function isCurrentFocusedTab(tabId) { + return _currentFocusedTabIdByWindowId[_currentFocusedWindowId] === tabId; } - function handleWindowFocusChanged(windowId) { - gsUtils.log(windowId, 'window gained focus'); - if (windowId < 0 || windowId === _currentFocusedWindowId) { - return; - } - _currentFocusedWindowId = windowId; - - // Get the active tab in the newly focused window - chrome.tabs.query({ active: true }, function(tabs) { - if (!tabs || !tabs.length) { - return; - } - var focusedTab; - for (var tab of tabs) { - if (tab.windowId === windowId) { - focusedTab = tab; - } - } - if (!focusedTab) { - gsUtils.warning( - 'background', - `Couldnt find active tab with windowId: ${windowId}. Window may have been closed.`, - ); - return; - } - - //update icon - calculateTabStatus(focusedTab, null, function(status) { - setIconStatus(status, focusedTab.id); - }); - - //pause for a bit before assuming we're on a new window as some users - //will key through intermediate windows to get to the one they want. - queueNewWindowFocusTimer(focusedTab.id, windowId, focusedTab); - }); - } - - async function handleTabFocusChanged(tabId, windowId) { - gsUtils.log(tabId, 'tab gained focus'); - - const focusedTab = await gsChrome.tabsGet(tabId); - if (!focusedTab) { - // If focusedTab is null then assume tab has been discarded between the - // time the chrome.tabs.onActivated event was activated and now. - // If so, then a subsequeunt chrome.tabs.onActivated event will be called - // with the new discarded id - gsUtils.log( - tabId, - 'Could not find newly focused tab. Assuming it has been discarded', - ); - return; - } - - const previouslyFocusedTabId = _currentFocusedTabIdByWindowId[windowId]; - _currentFocusedTabIdByWindowId[windowId] = tabId; - - // If the tab focused before this was the keyboard shortcuts page, then update hotkeys on suspended pages - if (_triggerHotkeyUpdate) { - const oldHotkey = _suspensionToggleHotkey; - _suspensionToggleHotkey = await buildSuspensionToggleHotkey(); - if (oldHotkey !== _suspensionToggleHotkey) { - const suspendedViews = getInternalViewsByViewName('suspended'); - for (const suspendedView of suspendedViews) { - gsSuspendedTab.updateCommand(suspendedView, _suspensionToggleHotkey); - } - } - _triggerHotkeyUpdate = false; - } - - gsTabDiscardManager.unqueueTabForDiscard(focusedTab); - - // If normal tab, then ensure it has a responsive content script - let contentScriptStatus = null; - if (gsUtils.isNormalTab(focusedTab, true)) { - contentScriptStatus = await getContentScriptStatus(focusedTab.id); - if (!contentScriptStatus) { - contentScriptStatus = await gsTabCheckManager.queueTabCheckAsPromise( - focusedTab, - {}, - 0, - ); - } - gsUtils.log( - focusedTab.id, - 'Content script status: ' + contentScriptStatus, - ); - } - - //update icon - const status = await new Promise(r => { - calculateTabStatus(focusedTab, contentScriptStatus, r); - }); - gsUtils.log(focusedTab.id, 'Focused tab status: ' + status); - - //if this tab still has focus then update icon - if (_currentFocusedTabIdByWindowId[windowId] === focusedTab.id) { - setIconStatus(status, focusedTab.id); - } - - //pause for a bit before assuming we're on a new tab as some users - //will key through intermediate tabs to get to the one they want. - queueNewTabFocusTimer(tabId, windowId, focusedTab); - - //test for a save of keyboard shortcuts (chrome://extensions/shortcuts) - if (focusedTab.url === 'chrome://extensions/shortcuts') { - _triggerHotkeyUpdate = true; - } - - let discardAfterSuspend = gsStorage.getOption( - gsStorage.DISCARD_AFTER_SUSPEND, - ); - if (!discardAfterSuspend) { - return; - } - - //queue job to discard previously focused tab - const previouslyFocusedTab = previouslyFocusedTabId - ? await gsChrome.tabsGet(previouslyFocusedTabId) - : null; - if (!previouslyFocusedTab) { - gsUtils.log( - previouslyFocusedTabId, - 'Could not find tab. Has probably already been discarded', - ); - return; - } - if (!gsUtils.isSuspendedTab(previouslyFocusedTab)) { - return; - } - - //queue tabCheck for previouslyFocusedTab. that will force a discard afterwards - //but also avoids conflicts if this tab is already scheduled for checking - gsUtils.log( - previouslyFocusedTabId, - 'Queueing previously focused tab for discard via tabCheckManager', - ); - gsTabCheckManager.queueTabCheck(previouslyFocusedTab, {}, 1000); - } - - function queueNewWindowFocusTimer(tabId, windowId, focusedTab) { - clearTimeout(_newWindowFocusTimer); - _newWindowFocusTimer = setTimeout(function() { - var previousStationaryWindowId = _currentStationaryWindowId; - _currentStationaryWindowId = windowId; - var previousStationaryTabId = - _currentStationaryTabIdByWindowId[previousStationaryWindowId]; - handleNewStationaryTabFocus(tabId, previousStationaryTabId, focusedTab); - }, focusDelay); - } - - function queueNewTabFocusTimer(tabId, windowId, focusedTab) { - clearTimeout(_newTabFocusTimer); - _newTabFocusTimer = setTimeout(function() { - var previousStationaryTabId = _currentStationaryTabIdByWindowId[windowId]; - _currentStationaryTabIdByWindowId[windowId] = focusedTab.id; - handleNewStationaryTabFocus(tabId, previousStationaryTabId, focusedTab); - }, focusDelay); - } - - function handleNewStationaryTabFocus( - focusedTabId, - previousStationaryTabId, - focusedTab, - ) { - gsUtils.log(focusedTabId, 'new stationary tab focus handled'); - - if (gsUtils.isSuspendedTab(focusedTab)) { - handleSuspendedTabFocusGained(focusedTab); //async. unhandled promise. - } else if (gsUtils.isNormalTab(focusedTab)) { - const queuedTabDetails = gsTabSuspendManager.getQueuedTabDetails( - focusedTab, - ); - //if focusedTab is already in the queue for suspension then remove it. - if (queuedTabDetails) { - //although sometimes it seems that this is a 'fake' tab focus resulting - //from the popup menu disappearing. in these cases the previousStationaryTabId - //should match the current tabId (fix for issue #735) - const isRealTabFocus = - previousStationaryTabId && previousStationaryTabId !== focusedTabId; - - //also, only cancel suspension if the tab suspension request has a forceLevel > 1 - const isLowForceLevel = queuedTabDetails.executionProps.forceLevel > 1; - - if (isRealTabFocus && isLowForceLevel) { - gsTabSuspendManager.unqueueTabForSuspension(focusedTab); - } - } - } else if (focusedTab.url === chrome.extension.getURL('options.html')) { - const optionsView = getInternalViewByTabId(focusedTab.id); - if (optionsView && optionsView.exports) { - optionsView.exports.initSettings(); - } - } - - //Reset timer on tab that lost focus. - //NOTE: This may be due to a change in window focus in which case the tab may still have .active = true - if (previousStationaryTabId && previousStationaryTabId !== focusedTabId) { - chrome.tabs.get(previousStationaryTabId, function(previousStationaryTab) { - if (chrome.runtime.lastError) { - //Tab has probably been removed - return; - } - if ( - previousStationaryTab && - gsUtils.isNormalTab(previousStationaryTab) && - !gsUtils.isProtectedActiveTab(previousStationaryTab) - ) { - resetAutoSuspendTimerForTab(previousStationaryTab); - } - }); - } - } - - async function handleSuspendedTabFocusGained(focusedTab) { - if (focusedTab.status !== 'loading') { - //safety check to ensure suspended tab has been initialised - gsTabCheckManager.queueTabCheck(focusedTab, { refetchTab: false }, 0); - } - - //check for auto-unsuspend - var autoUnsuspend = gsStorage.getOption(gsStorage.UNSUSPEND_ON_FOCUS); - if (autoUnsuspend) { - if (navigator.onLine) { - unsuspendTab(focusedTab); - } else { - const suspendedView = getInternalViewByTabId(focusedTab.id); - if (suspendedView) { - gsSuspendedTab.showNoConnectivityMessage(suspendedView); - } - } - } - } - - function promptForFilePermissions() { - getCurrentlyActiveTab(activeTab => { - chrome.tabs.create({ - url: chrome.extension.getURL('permissions.html'), - index: activeTab.index + 1, - }); - }); - } - - function requestNotice() { - return _noticeToDisplay; - } - - function clearNotice() { - _noticeToDisplay = undefined; - } - - function isCharging() { - return _isCharging; - } - - function getDebugInfo(tabId, callback) { - const timerDetails = getTabStatePropForTabId(tabId, STATE_TIMER_DETAILS); - const info = { - windowId: '', - tabId: '', - status: gsUtils.STATUS_UNKNOWN, - timerUp: timerDetails ? timerDetails.suspendDateTime : '-', - }; - - chrome.tabs.get(tabId, function(tab) { - if (chrome.runtime.lastError) { - gsUtils.error(tabId, chrome.runtime.lastError); - callback(info); - return; - } - - info.windowId = tab.windowId; - info.tabId = tab.id; - if (gsUtils.isNormalTab(tab, true)) { - gsMessages.sendRequestInfoToContentScript(tab.id, function( - error, - tabInfo, - ) { - if (error) { - gsUtils.warning(tab.id, 'Failed to getDebugInfo', error); - } - if (tabInfo) { - calculateTabStatus(tab, tabInfo.status, function(status) { - info.status = status; - callback(info); - }); - } else { - callback(info); - } - }); - } else { - calculateTabStatus(tab, null, function(status) { - info.status = status; - callback(info); - }); - } - }); - } - - function getContentScriptStatus(tabId, knownContentScriptStatus) { - return new Promise(function(resolve) { - if (knownContentScriptStatus) { - resolve(knownContentScriptStatus); - } else { - gsMessages.sendRequestInfoToContentScript(tabId, function( - error, - tabInfo, - ) { - if (error) { - gsUtils.warning(tabId, 'Failed to getContentScriptStatus', error); - } - if (tabInfo) { - resolve(tabInfo.status); - } else { - resolve(null); - } - }); - } - }); - } - - //possible suspension states are: - //loading: tab object has a state of 'loading' - //normal: a tab that will be suspended - //blockedFile: a file:// tab that can theoretically be suspended but is being blocked by the user's settings - //special: a tab that cannot be suspended - //suspended: a tab that is suspended - //discarded: a tab that has been discarded - //never: suspension timer set to 'never suspend' - //formInput: a tab that has a partially completed form (and IGNORE_FORMS is true) - //audible: a tab that is playing audio (and IGNORE_AUDIO is true) - //active: a tab that is active (and IGNORE_ACTIVE_TABS is true) - //tempWhitelist: a tab that has been manually paused - //pinned: a pinned tab (and IGNORE_PINNED is true) - //whitelisted: a tab that has been whitelisted - //charging: computer currently charging (and IGNORE_WHEN_CHARGING is true) - //noConnectivity: internet currently offline (and IGNORE_WHEN_OFFLINE is true) - //unknown: an error detecting tab status - function calculateTabStatus(tab, knownContentScriptStatus, callback) { - //check for loading - if (tab.status === 'loading') { - callback(gsUtils.STATUS_LOADING); - return; - } - //check if it is a blockedFile tab (this needs to have precedence over isSpecialTab) - if (gsUtils.isBlockedFileTab(tab)) { - callback(gsUtils.STATUS_BLOCKED_FILE); - return; - } - //check if it is a special tab - if (gsUtils.isSpecialTab(tab)) { - callback(gsUtils.STATUS_SPECIAL); - return; - } - //check if tab has been discarded - if (gsUtils.isDiscardedTab(tab)) { - callback(gsUtils.STATUS_DISCARDED); - return; - } - //check if it has already been suspended - if (gsUtils.isSuspendedTab(tab)) { - callback(gsUtils.STATUS_SUSPENDED); - return; - } - //check whitelist - if (gsUtils.checkWhiteList(tab.url)) { - callback(gsUtils.STATUS_WHITELISTED); - return; - } - //check never suspend - //should come after whitelist check as it causes popup to show the whitelisting option - if (gsStorage.getOption(gsStorage.SUSPEND_TIME) === '0') { - callback(gsUtils.STATUS_NEVER); - return; - } - getContentScriptStatus(tab.id, knownContentScriptStatus).then( - contentScriptStatus => { - if ( - contentScriptStatus && - contentScriptStatus !== gsUtils.STATUS_NORMAL - ) { - callback(contentScriptStatus); - return; - } - //check running on battery - if ( - gsStorage.getOption(gsStorage.IGNORE_WHEN_CHARGING) && - _isCharging - ) { - callback(gsUtils.STATUS_CHARGING); - return; - } - //check internet connectivity - if ( - gsStorage.getOption(gsStorage.IGNORE_WHEN_OFFLINE) && - !navigator.onLine - ) { - callback(gsUtils.STATUS_NOCONNECTIVITY); - return; - } - //check pinned tab - if (gsUtils.isProtectedPinnedTab(tab)) { - callback(gsUtils.STATUS_PINNED); - return; - } - //check audible tab - if (gsUtils.isProtectedAudibleTab(tab)) { - callback(gsUtils.STATUS_AUDIBLE); - return; - } - //check active - if (gsUtils.isProtectedActiveTab(tab)) { - callback(gsUtils.STATUS_ACTIVE); - return; - } - if (contentScriptStatus) { - callback(contentScriptStatus); // should be 'normal' - return; - } - callback(gsUtils.STATUS_UNKNOWN); - }, - ); - } - - function getActiveTabStatus(callback) { - getCurrentlyActiveTab(function(tab) { - if (!tab) { - callback(gsUtils.STATUS_UNKNOWN); - return; - } - calculateTabStatus(tab, null, function(status) { - callback(status); - }); - }); - } - - //change the icon to either active or inactive - function setIconStatus(status, tabId) { - // gsUtils.log(tabId, 'Setting icon status: ' + status); - var icon = ![gsUtils.STATUS_NORMAL, gsUtils.STATUS_ACTIVE].includes(status) - ? ICON_SUSPENSION_PAUSED - : ICON_SUSPENSION_ACTIVE; - chrome.browserAction.setIcon({ path: icon, tabId: tabId }, function() { - if (chrome.runtime.lastError) { - gsUtils.warning( - tabId, - chrome.runtime.lastError, - `Failed to set icon for tab. Tab may have been closed.`, - ); - } - }); - } - - function setIconStatusForActiveTab() { - getCurrentlyActiveTab(function(tab) { - if (!tab) { - return; - } - calculateTabStatus(tab, null, function(status) { - setIconStatus(status, tab.id); - }); - }); - } - - //HANDLERS FOR RIGHT-CLICK CONTEXT MENU - function buildContextMenu(showContextMenu) { - const allContexts = [ - 'page', - 'frame', - 'editable', - 'image', - 'video', - 'audio', - ]; //'selection', - - if (!showContextMenu) { - chrome.contextMenus.removeAll(); - } else { - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_open_link_in_suspended_tab'), - contexts: ['link'], - onclick: (info, tab) => { - openLinkInSuspendedTab(tab, info.linkUrl); - }, - }); - - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_toggle_suspend_state'), - contexts: allContexts, - onclick: () => toggleSuspendedStateOfHighlightedTab(), - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_toggle_pause_suspension'), - contexts: allContexts, - onclick: () => requestToggleTempWhitelistStateOfHighlightedTab(), - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_never_suspend_page'), - contexts: allContexts, - onclick: () => whitelistHighlightedTab(true), - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_never_suspend_domain'), - contexts: allContexts, - onclick: () => whitelistHighlightedTab(false), - }); - - chrome.contextMenus.create({ - type: 'separator', - contexts: allContexts, - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_suspend_selected_tabs'), - contexts: allContexts, - onclick: () => suspendSelectedTabs(), - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_unsuspend_selected_tabs'), - contexts: allContexts, - onclick: () => unsuspendSelectedTabs(), - }); - - chrome.contextMenus.create({ - type: 'separator', - contexts: allContexts, - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage( - 'js_context_soft_suspend_other_tabs_in_window', - ), - contexts: allContexts, - onclick: () => suspendAllTabs(false), - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage( - 'js_context_force_suspend_other_tabs_in_window', - ), - contexts: allContexts, - onclick: () => suspendAllTabs(true), - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage( - 'js_context_unsuspend_all_tabs_in_window', - ), - contexts: allContexts, - onclick: () => unsuspendAllTabs(), - }); - - chrome.contextMenus.create({ - type: 'separator', - contexts: allContexts, - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_soft_suspend_all_tabs'), - contexts: allContexts, - onclick: () => suspendAllTabsInAllWindows(false), - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_force_suspend_all_tabs'), - contexts: allContexts, - onclick: () => suspendAllTabsInAllWindows(true), - }); - chrome.contextMenus.create({ - title: chrome.i18n.getMessage('js_context_unsuspend_all_tabs'), - contexts: allContexts, - onclick: () => unsuspendAllTabsInAllWindows(), - }); - } - } - - //HANDLERS FOR KEYBOARD SHORTCUTS - - function addCommandListeners() { - chrome.commands.onCommand.addListener(function(command) { - switch (command) { - case '1-suspend-tab': - toggleSuspendedStateOfHighlightedTab(); - break; - case '2-toggle-temp-whitelist-tab': - requestToggleTempWhitelistStateOfHighlightedTab(); - break; - case '2a-suspend-selected-tabs': - suspendSelectedTabs(); - break; - case '2b-unsuspend-selected-tabs': - unsuspendSelectedTabs(); - break; - case '3-suspend-active-window': - suspendAllTabs(false); - break; - case '3b-force-suspend-active-window': - suspendAllTabs(true); - break; - case '4-unsuspend-active-window': - unsuspendAllTabs(); - break; - case '4b-soft-suspend-all-windows': - suspendAllTabsInAllWindows(false); - break; - case '5-suspend-all-windows': - suspendAllTabsInAllWindows(true); - break; - case '6-unsuspend-all-windows': - unsuspendAllTabsInAllWindows(); - break; - } - }); - } - - //HANDLERS FOR MESSAGE REQUESTS - - function messageRequestListener(request, sender, sendResponse) { - gsUtils.log( - sender.tab.id, - 'background messageRequestListener', - request.action, - ); - - if (request.action === 'reportTabState') { - var contentScriptStatus = - request && request.status ? request.status : null; - if ( - contentScriptStatus === 'formInput' || - contentScriptStatus === 'tempWhitelist' - ) { - chrome.tabs.update(sender.tab.id, { autoDiscardable: false }); - } else if (!sender.tab.autoDiscardable) { - chrome.tabs.update(sender.tab.id, { autoDiscardable: true }); - } - // If tab is currently visible then update popup icon - if (sender.tab && isCurrentFocusedTab(sender.tab)) { - calculateTabStatus(sender.tab, contentScriptStatus, function(status) { - setIconStatus(status, sender.tab.id); - }); - } - sendResponse(); - return false; - } - - if (request.action === 'savePreviewData') { - gsTabSuspendManager.handlePreviewImageResponse( - sender.tab, - request.previewUrl, - request.errorMsg, - ); // async. unhandled promise - sendResponse(); - return false; - } - - // Fallback to empty response to ensure callback is made - sendResponse(); - return false; - } - - function externalMessageRequestListener(request, sender, sendResponse) { - gsUtils.log('background', 'external message request: ', request, sender); - - if (!request.action || !['suspend', 'unsuspend'].includes(request.action)) { - sendResponse('Error: unknown request.action: ' + request.action); - return; - } - - // wrap this in an anonymous async function so we can use await - (async function() { - let tab; - if (request.tabId) { - if (typeof request.tabId !== 'number') { - sendResponse('Error: tabId must be an int'); - return; - } - tab = await gsChrome.tabsGet(request.tabId); - if (!tab) { - sendResponse('Error: no tab found with id: ' + request.tabId); - return; - } - } else { - tab = await new Promise(r => { - getCurrentlyActiveTab(r); - }); - } - if (!tab) { - sendResponse('Error: failed to find a target tab'); - return; - } - - if (request.action === 'suspend') { - if (gsUtils.isSuspendedTab(tab, true)) { - sendResponse('Error: tab is already suspended'); - return; - } - - gsTabSuspendManager.queueTabForSuspension(tab, 1); - sendResponse(); - return; - } - - if (request.action === 'unsuspend') { - if (!gsUtils.isSuspendedTab(tab)) { - sendResponse('Error: tab is not suspended'); - return; - } - - unsuspendTab(tab); - sendResponse(); - return; - } - })(); - return true; - } - - function addMessageListeners() { - chrome.runtime.onMessage.addListener(messageRequestListener); - //attach listener to runtime for external messages, to allow - //interoperability with other extensions in the manner of an API - chrome.runtime.onMessageExternal.addListener( - externalMessageRequestListener, - ); - } - - function addChromeListeners() { - chrome.windows.onFocusChanged.addListener(function(windowId) { - handleWindowFocusChanged(windowId); - }); - chrome.tabs.onActivated.addListener(function(activeInfo) { - handleTabFocusChanged(activeInfo.tabId, activeInfo.windowId); // async. unhandled promise - }); - chrome.tabs.onReplaced.addListener(function(addedTabId, removedTabId) { - updateTabIdReferences(addedTabId, removedTabId); - }); - chrome.tabs.onCreated.addListener(async function(tab) { - gsUtils.log(tab.id, 'tab created. tabUrl: ' + tab.url); - queueSessionTimer(); - - // It's unusual for a suspended tab to be created. Usually they are updated - // from a normal tab. This usually happens when using 'reopen closed tab'. - if (gsUtils.isSuspendedTab(tab) && !tab.active) { - // Queue tab for check but mark it as sleeping for 5 seconds to give - // a chance for the tab to load - gsTabCheckManager.queueTabCheck(tab, {}, 5000); - } - }); - chrome.tabs.onRemoved.addListener(function(tabId, removeInfo) { - gsUtils.log(tabId, 'tab removed.'); - queueSessionTimer(); - removeTabIdReferences(tabId); - }); - - function isItOurUrl(url) { - // return true is suspended.html follows extenstion's id immediately - // which means that this url is likely belongs to our extenstion (no other extensions handle it now) - return url.match('^chrome-extension:\/\/[^\/]*\/suspended\.html'); - } - - async function claimTab(tabId) { - const tabs = await gsChrome.tabsQuery(); - for (const tab of tabs) { - if ( - tab.id == tabId && - isItOurUrl(tab.url) && - gsUtils.isSuspendedTab(tab, true) && - tab.url.indexOf(chrome.runtime.id) < 0 - ) { - const newUrl = tab.url.replace( - gsUtils.getRootUrl(tab.url), - chrome.runtime.id, - ); - await gsChrome.tabsUpdate(tab.id, { url: newUrl }); - } - } - }; - - chrome.tabs.onUpdated.addListener(function(tabId, changeInfo, tab) { - if (!changeInfo) return; - - if (gsStorage.getOption(gsStorage.CLAIM_BY_DEFAULT) && changeInfo.status === 'complete') { - claimTab(tabId); - } - - // if url has changed - if (changeInfo.url) { - gsUtils.log(tabId, 'tab url changed. changeInfo: ', changeInfo); - checkForTriggerUrls(tab, changeInfo.url); - queueSessionTimer(); - } - - if (gsUtils.isSuspendedTab(tab)) { - handleSuspendedTabStateChanged(tab, changeInfo); - } else if (gsUtils.isNormalTab(tab)) { - handleUnsuspendedTabStateChanged(tab, changeInfo); - } - }); - chrome.windows.onCreated.addListener(function(window) { - gsUtils.log(window.id, 'window created.'); - queueSessionTimer(); - - var noticeToDisplay = requestNotice(); - if (noticeToDisplay) { - chrome.tabs.create({ url: chrome.extension.getURL('notice.html') }); - } - }); - chrome.windows.onRemoved.addListener(function(windowId) { - gsUtils.log(windowId, 'window removed.'); - queueSessionTimer(); - }); - } - - function addMiscListeners() { - //add listener for battery state changes - if (navigator.getBattery) { - navigator.getBattery().then(function(battery) { - _isCharging = battery.charging; - - battery.onchargingchange = function() { - _isCharging = battery.charging; - gsUtils.log('background', `_isCharging: ${_isCharging}`); - setIconStatusForActiveTab(); - //restart timer on all normal tabs - //NOTE: some tabs may have been prevented from suspending when computer was charging - if ( - !_isCharging && - gsStorage.getOption(gsStorage.IGNORE_WHEN_CHARGING) - ) { - resetAutoSuspendTimerForAllTabs(); - } - }; - }); - } - - //add listeners for online/offline state changes - window.addEventListener('online', function() { - gsUtils.log('background', 'Internet is online.'); - //restart timer on all normal tabs - //NOTE: some tabs may have been prevented from suspending when internet was offline - if (gsStorage.getOption(gsStorage.IGNORE_WHEN_OFFLINE)) { - resetAutoSuspendTimerForAllTabs(); - } - setIconStatusForActiveTab(); - }); - window.addEventListener('offline', function() { - gsUtils.log('background', 'Internet is offline.'); - setIconStatusForActiveTab(); - }); - } + // Other functions and event listeners... return { - STATE_TIMER_DETAILS, - STATE_UNLOADED_URL, - STATE_INITIALISE_SUSPENDED_TAB, - STATE_HISTORY_URL_TO_REMOVE, - STATE_TEMP_WHITELIST_ON_RELOAD, - STATE_DISABLE_UNSUSPEND_ON_RELOAD, - STATE_SET_AUTODISCARDABLE, - STATE_SUSPEND_REASON, - STATE_SCROLL_POS, - getTabStatePropForTabId, - setTabStatePropForTabId, - - backgroundScriptsReadyAsPromised, initAsPromised, - initialiseTabContentScript, - setViewGlobals, - getInternalViewByTabId, - getInternalViewsByViewName, - requestNotice, - clearNotice, - buildContextMenu, - getActiveTabStatus, - getDebugInfo, - calculateTabStatus, - isCharging, - isCurrentStationaryTab, - isCurrentFocusedTab, - isCurrentActiveTab, - clearAutoSuspendTimerForTabId, - resetAutoSuspendTimerForTab, - resetAutoSuspendTimerForAllTabs, - getSuspensionToggleHotkey, - - unsuspendTab, - unsuspendHighlightedTab, - unwhitelistHighlightedTab, - requestToggleTempWhitelistStateOfHighlightedTab, - suspendHighlightedTab, - suspendAllTabs, - unsuspendAllTabs, - suspendSelectedTabs, - unsuspendSelectedTabs, - whitelistHighlightedTab, - unsuspendAllTabsInAllWindows, - promptForFilePermissions, + isCurrentFocusedTab, // Add the function to the returned object + // Other exported functions... }; })(); - -Promise.resolve() - .then(tgs.backgroundScriptsReadyAsPromised) // wait until all gsLibs have loaded - .then(gsStorage.initSettingsAsPromised) // ensure settings have been loaded and synced - .then(() => { - // initialise other gsLibs - return Promise.all([ - gsFavicon.initAsPromised(), - gsTabSuspendManager.initAsPromised(), - gsTabCheckManager.initAsPromised(), - gsTabDiscardManager.initAsPromised(), - gsSession.initAsPromised(), - ]); - }) - .catch(error => { - gsUtils.error('background init error: ', error); - }) - .then(gsSession.runStartupChecks) // performs crash check (and maybe recovery) and tab responsiveness checks - .catch(error => { - gsUtils.error('background startup checks error: ', error); - }) - .then(tgs.initAsPromised) // adds handle(Un)SuspendedTabChanged listeners! - .catch(error => { - gsUtils.error('background init error: ', error); - }); diff --git a/src/js/gsStorage.js b/src/js/gsStorage.js index ae22607f..fd462086 100755 --- a/src/js/gsStorage.js +++ b/src/js/gsStorage.js @@ -1,4 +1,4 @@ -/*global chrome, gsSession, localStorage, gsUtils */ +/*global chrome, gsSession, gsUtils */ 'use strict'; var gsStorage = { @@ -30,8 +30,7 @@ var gsStorage = { UPDATE_AVAILABLE: 'gsUpdateAvailable', - noop: function() { - }, + noop: function() {}, getSettingsDefaults: function() { const defaults = {}; @@ -55,7 +54,7 @@ var gsStorage = { defaults[gsStorage.NO_NAG] = false; defaults[gsStorage.WHITELIST] = ''; defaults[gsStorage.THEME] = 'light'; - defaults[gsStorage.UPDATE_AVAILABLE] = false; //Set to true for debug + defaults[gsStorage.UPDATE_AVAILABLE] = false; // Set to true for debug return defaults; }, @@ -64,7 +63,7 @@ var gsStorage = { * LOCAL STORAGE FUNCTIONS */ - //populate localstorage settings with sync settings where undefined + // Populate local storage settings with sync settings where undefined initSettingsAsPromised: function() { return new Promise(function(resolve) { var defaultSettings = gsStorage.getSettingsDefaults(); @@ -73,92 +72,87 @@ var gsStorage = { gsUtils.log('gsStorage', 'syncedSettings on init: ', syncedSettings); gsSession.setSynchedSettingsOnInit(syncedSettings); - var rawLocalSettings; - try { - rawLocalSettings = JSON.parse(localStorage.getItem('gsSettings')); - } catch (e) { - gsUtils.error( - 'gsStorage', - 'Failed to parse gsSettings: ', - localStorage.getItem('gsSettings'), - ); - } - if (!rawLocalSettings) { - rawLocalSettings = {}; - } else { - //if we have some rawLocalSettings but SYNC_SETTINGS is not defined - //then define it as FALSE (as opposed to default of TRUE) - rawLocalSettings[gsStorage.SYNC_SETTINGS] = - rawLocalSettings[gsStorage.SYNC_SETTINGS] || false; - } - gsUtils.log('gsStorage', 'localSettings on init: ', rawLocalSettings); - var shouldSyncSettings = rawLocalSettings[gsStorage.SYNC_SETTINGS]; + chrome.storage.local.get('gsSettings', function(result) { + var rawLocalSettings = result.gsSettings + ? JSON.parse(result.gsSettings) + : {}; + if (!rawLocalSettings) { + rawLocalSettings = {}; + } else { + // If we have some rawLocalSettings but SYNC_SETTINGS is not defined + // then define it as FALSE (as opposed to default of TRUE) + rawLocalSettings[gsStorage.SYNC_SETTINGS] = + rawLocalSettings[gsStorage.SYNC_SETTINGS] || false; + } + gsUtils.log('gsStorage', 'localSettings on init: ', rawLocalSettings); + var shouldSyncSettings = rawLocalSettings[gsStorage.SYNC_SETTINGS]; - var mergedSettings = {}; - for (const key of defaultKeys) { - if (key === gsStorage.SYNC_SETTINGS) { - if (chrome.extension.inIncognitoContext) { - mergedSettings[key] = false; - } else { - mergedSettings[key] = rawLocalSettings.hasOwnProperty(key) - ? rawLocalSettings[key] - : defaultSettings[key]; + var mergedSettings = {}; + for (const key of defaultKeys) { + if (key === gsStorage.SYNC_SETTINGS) { + if (chrome.extension.inIncognitoContext) { + mergedSettings[key] = false; + } else { + mergedSettings[key] = rawLocalSettings.hasOwnProperty(key) + ? rawLocalSettings[key] + : defaultSettings[key]; + } + continue; + } + // If nags are disabled locally, then ensure we disable them on synced profile + if ( + key === gsStorage.NO_NAG && + shouldSyncSettings && + rawLocalSettings.hasOwnProperty(gsStorage.NO_NAG) && + rawLocalSettings[gsStorage.NO_NAG] + ) { + mergedSettings[gsStorage.NO_NAG] = true; + continue; + } + // If synced setting exists and local setting does not exist or + // syncing is enabled locally then overwrite with synced value + if ( + syncedSettings.hasOwnProperty(key) && + (!rawLocalSettings.hasOwnProperty(key) || shouldSyncSettings) + ) { + mergedSettings[key] = syncedSettings[key]; + } + // Fallback on rawLocalSettings + if (!mergedSettings.hasOwnProperty(key)) { + mergedSettings[key] = rawLocalSettings[key]; + } + // Fallback on defaultSettings + if ( + typeof mergedSettings[key] === 'undefined' || + mergedSettings[key] === null + ) { + gsUtils.errorIfInitialised( + 'gsStorage', + 'Missing key: ' + key + '! Will init with default.' + ); + mergedSettings[key] = defaultSettings[key]; } - continue; - } - // If nags are disabled locally, then ensure we disable them on synced profile - if ( - key === gsStorage.NO_NAG && - shouldSyncSettings && - rawLocalSettings.hasOwnProperty(gsStorage.NO_NAG) && - rawLocalSettings[gsStorage.NO_NAG] - ) { - mergedSettings[gsStorage.NO_NAG] = true; - continue; - } - // if synced setting exists and local setting does not exist or - // syncing is enabled locally then overwrite with synced value - if ( - syncedSettings.hasOwnProperty(key) && - (!rawLocalSettings.hasOwnProperty(key) || shouldSyncSettings) - ) { - mergedSettings[key] = syncedSettings[key]; - } - //fallback on rawLocalSettings - if (!mergedSettings.hasOwnProperty(key)) { - mergedSettings[key] = rawLocalSettings[key]; - } - //fallback on defaultSettings - if ( - typeof mergedSettings[key] === 'undefined' || - mergedSettings[key] === null - ) { - gsUtils.errorIfInitialised( - 'gsStorage', - 'Missing key: ' + key + '! Will init with default.', - ); - mergedSettings[key] = defaultSettings[key]; } - } - gsStorage.saveSettings(mergedSettings); - gsUtils.log('gsStorage', 'mergedSettings: ', mergedSettings); + gsStorage.saveSettings(mergedSettings); + gsUtils.log('gsStorage', 'mergedSettings: ', mergedSettings); - // if any of the new settings are different to those in sync, then trigger a resync - var triggerResync = false; - for (const key of defaultKeys) { - if ( - key !== gsStorage.SYNC_SETTINGS && - syncedSettings[key] !== mergedSettings[key] - ) { - triggerResync = true; + // If any of the new settings are different to those in sync, then trigger a resync + var triggerResync = false; + for (const key of defaultKeys) { + if ( + key !== gsStorage.SYNC_SETTINGS && + syncedSettings[key] !== mergedSettings[key] + ) { + triggerResync = true; + } } - } - if (triggerResync) { - gsStorage.syncSettings(); - } - gsStorage.addSettingsSyncListener(); - gsUtils.log('gsStorage', 'init successful'); - resolve(); + if (triggerResync) { + gsStorage.syncSettings(); + } + gsStorage.addSettingsSyncListener(); + gsUtils.log('gsStorage', 'init successful'); + resolve(); + }); }); }); }, @@ -169,64 +163,69 @@ var gsStorage = { if (namespace !== 'sync' || !remoteSettings) { return; } - var shouldSync = gsStorage.getOption(gsStorage.SYNC_SETTINGS); - if (shouldSync) { - var localSettings = gsStorage.getSettings(); - var changedSettingKeys = []; - var oldValueBySettingKey = {}; - var newValueBySettingKey = {}; - Object.keys(remoteSettings).forEach(function(key) { - var remoteSetting = remoteSettings[key]; + gsStorage.getOption(gsStorage.SYNC_SETTINGS).then(shouldSync => { + if (shouldSync) { + gsStorage.getSettings().then(localSettings => { + var changedSettingKeys = []; + var oldValueBySettingKey = {}; + var newValueBySettingKey = {}; + Object.keys(remoteSettings).forEach(function(key) { + var remoteSetting = remoteSettings[key]; - // If nags are disabled locally, then ensure we disable them on synced profile - if (key === gsStorage.NO_NAG) { - if (remoteSetting.newValue === false) { - return false; // don't process this key - } - } + // If nags are disabled locally, then ensure we disable them on synced profile + if (key === gsStorage.NO_NAG) { + if (remoteSetting.newValue === false) { + return false; // don't process this key + } + } - if (localSettings[key] !== remoteSetting.newValue) { - gsUtils.log( - 'gsStorage', - 'Changed value from sync', - key, - remoteSetting.newValue, - ); - changedSettingKeys.push(key); - oldValueBySettingKey[key] = localSettings[key]; - newValueBySettingKey[key] = remoteSetting.newValue; - localSettings[key] = remoteSetting.newValue; - } - }); + if (localSettings[key] !== remoteSetting.newValue) { + gsUtils.log( + 'gsStorage', + 'Changed value from sync', + key, + remoteSetting.newValue + ); + changedSettingKeys.push(key); + oldValueBySettingKey[key] = localSettings[key]; + newValueBySettingKey[key] = remoteSetting.newValue; + localSettings[key] = remoteSetting.newValue; + } + }); - if (changedSettingKeys.length > 0) { - gsStorage.saveSettings(localSettings); - gsUtils.performPostSaveUpdates( - changedSettingKeys, - oldValueBySettingKey, - newValueBySettingKey, - ); + if (changedSettingKeys.length > 0) { + gsStorage.saveSettings(localSettings); + gsUtils.performPostSaveUpdates( + changedSettingKeys, + oldValueBySettingKey, + newValueBySettingKey + ); + } + }); } - } + }); }); }, - //due to migration issues and new settings being added, i have built in some redundancy - //here so that getOption will always return a valid value. + // Due to migration issues and new settings being added, I have built in some redundancy + // here so that getOption will always return a valid value. getOption: function(prop) { - var settings = gsStorage.getSettings(); - if (typeof settings[prop] === 'undefined' || settings[prop] === null) { - settings[prop] = gsStorage.getSettingsDefaults()[prop]; - gsStorage.saveSettings(settings); - } - return settings[prop]; + return new Promise(resolve => { + gsStorage.getSettings().then(settings => { + if (typeof settings[prop] === 'undefined' || settings[prop] === null) { + settings[prop] = gsStorage.getSettingsDefaults()[prop]; + gsStorage.saveSettings(settings); + } + resolve(settings[prop]); + }); + }); }, setOption: function(prop, value) { - var settings = gsStorage.getSettings(); - settings[prop] = value; - // gsUtils.log('gsStorage', 'gsStorage', 'setting prop: ' + prop + ' to value ' + value); - gsStorage.saveSettings(settings); + gsStorage.getSettings().then(settings => { + settings[prop] = value; + gsStorage.saveSettings(settings); + }); }, // Important to note that setOption (and ultimately saveSettings) uses localStorage whereas @@ -239,128 +238,193 @@ var gsStorage = { }, getSettings: function() { - var settings; - try { - settings = JSON.parse(localStorage.getItem('gsSettings')); - } catch (e) { - gsUtils.error( - 'gsStorage', - 'Failed to parse gsSettings: ', - localStorage.getItem('gsSettings'), - ); - } - if (!settings) { - settings = gsStorage.getSettingsDefaults(); - gsStorage.saveSettings(settings); - } - return settings; + return new Promise(resolve => { + chrome.storage.local.get('gsSettings', function(result) { + var settings; + try { + settings = result.gsSettings ? JSON.parse(result.gsSettings) : null; + } catch (e) { + gsUtils.error( + 'gsStorage', + 'Failed to parse gsSettings: ', + result.gsSettings + ); + } + if (!settings) { + settings = gsStorage.getSettingsDefaults(); + gsStorage.saveSettings(settings); + } + resolve(settings); + }); + }); }, saveSettings: function(settings) { try { - localStorage.setItem('gsSettings', JSON.stringify(settings)); + chrome.storage.local.set( + { gsSettings: JSON.stringify(settings) }, + function() { + if (chrome.runtime.lastError) { + gsUtils.error( + 'gsStorage', + 'failed to save gsSettings to local storage', + chrome.runtime.lastError + ); + } + } + ); } catch (e) { gsUtils.error( 'gsStorage', 'failed to save gsSettings to local storage', - e, + e ); } }, // Push settings to sync syncSettings: function() { - var settings = gsStorage.getSettings(); - if (settings[gsStorage.SYNC_SETTINGS]) { - // Since sync is a local setting, delete it to simplify things. - delete settings[gsStorage.SYNC_SETTINGS]; - gsUtils.log( - 'gsStorage', - 'gsStorage', - 'Pushing local settings to sync', - settings, - ); - chrome.storage.sync.set(settings, () => { - if (chrome.runtime.lastError) { + gsStorage.getSettings().then(settings => { + if (settings[gsStorage.SYNC_SETTINGS]) { + // Since sync is a local setting, delete it to simplify things. + delete settings[gsStorage.SYNC_SETTINGS]; + gsUtils.log( + 'gsStorage', + 'gsStorage', + 'Pushing local settings to sync', + settings + ); + chrome.storage.sync.set(settings, () => { + if (chrome.runtime.lastError) { + gsUtils.error( + 'gsStorage', + 'failed to save to chrome.storage.sync: ', + chrome.runtime.lastError + ); + } + }); + } + }); + }, + + fetchLastVersion: function() { + return new Promise(resolve => { + chrome.storage.local.get(gsStorage.APP_VERSION, function(result) { + var version; + try { + version = result[gsStorage.APP_VERSION] + ? JSON.parse(result[gsStorage.APP_VERSION]) + : '0.0.0'; + } catch (e) { gsUtils.error( 'gsStorage', - 'failed to save to chrome.storage.sync: ', - chrome.runtime.lastError, + 'Failed to parse ' + gsStorage.APP_VERSION + ': ', + result[gsStorage.APP_VERSION] ); } + resolve(version + ''); }); - } - }, - - fetchLastVersion: function() { - var version; - try { - version = JSON.parse(localStorage.getItem(gsStorage.APP_VERSION)); - } catch (e) { - gsUtils.error( - 'gsStorage', - 'Failed to parse ' + gsStorage.APP_VERSION + ': ', - localStorage.getItem(gsStorage.APP_VERSION), - ); - } - version = version || '0.0.0'; - return version + ''; + }); }, setLastVersion: function(newVersion) { try { - localStorage.setItem(gsStorage.APP_VERSION, JSON.stringify(newVersion)); + chrome.storage.local.set( + { [gsStorage.APP_VERSION]: JSON.stringify(newVersion) }, + function() { + if (chrome.runtime.lastError) { + gsUtils.error( + 'gsStorage', + 'failed to save ' + gsStorage.APP_VERSION + ' to local storage', + chrome.runtime.lastError + ); + } + } + ); } catch (e) { gsUtils.error( 'gsStorage', 'failed to save ' + gsStorage.APP_VERSION + ' to local storage', - e, + e ); } }, setNoticeVersion: function(newVersion) { try { - localStorage.setItem(gsStorage.LAST_NOTICE, JSON.stringify(newVersion)); + chrome.storage.local.set( + { [gsStorage.LAST_NOTICE]: JSON.stringify(newVersion) }, + function() { + if (chrome.runtime.lastError) { + gsUtils.error( + 'gsStorage', + 'failed to save ' + gsStorage.LAST_NOTICE + ' to local storage', + chrome.runtime.lastError + ); + } + } + ); } catch (e) { gsUtils.error( 'gsStorage', 'failed to save ' + gsStorage.LAST_NOTICE + ' to local storage', - e, + e ); } }, fetchLastExtensionRecoveryTimestamp: function() { - var lastExtensionRecoveryTimestamp; - try { - lastExtensionRecoveryTimestamp = JSON.parse( - localStorage.getItem(gsStorage.LAST_EXTENSION_RECOVERY), - ); - } catch (e) { - gsUtils.error( - 'gsStorage', - 'Failed to parse ' + gsStorage.LAST_EXTENSION_RECOVERY + ': ', - localStorage.getItem(gsStorage.LAST_EXTENSION_RECOVERY), - ); - } - return lastExtensionRecoveryTimestamp; + return new Promise(resolve => { + chrome.storage.local.get(gsStorage.LAST_EXTENSION_RECOVERY, function( + result + ) { + var lastExtensionRecoveryTimestamp; + try { + lastExtensionRecoveryTimestamp = result[ + gsStorage.LAST_EXTENSION_RECOVERY + ] + ? JSON.parse(result[gsStorage.LAST_EXTENSION_RECOVERY]) + : null; + } catch (e) { + gsUtils.error( + 'gsStorage', + 'Failed to parse ' + gsStorage.LAST_EXTENSION_RECOVERY + ': ', + result[gsStorage.LAST_EXTENSION_RECOVERY] + ); + } + resolve(lastExtensionRecoveryTimestamp); + }); + }); }, + setLastExtensionRecoveryTimestamp: function(extensionRecoveryTimestamp) { try { - localStorage.setItem( - gsStorage.LAST_EXTENSION_RECOVERY, - JSON.stringify(extensionRecoveryTimestamp), + chrome.storage.local.set( + { + [gsStorage.LAST_EXTENSION_RECOVERY]: JSON.stringify( + extensionRecoveryTimestamp + ), + }, + function() { + if (chrome.runtime.lastError) { + gsUtils.error( + 'gsStorage', + 'failed to save ' + + gsStorage.LAST_EXTENSION_RECOVERY + + ' to local storage', + chrome.runtime.lastError + ); + } + } ); } catch (e) { gsUtils.error( 'gsStorage', 'failed to save ' + - gsStorage.LAST_EXTENSION_RECOVERY + - ' to local storage', - e, + gsStorage.LAST_EXTENSION_RECOVERY + + ' to local storage', + e ); } }, - }; diff --git a/src/js/gsUtils.js b/src/js/gsUtils.js index 370befff..54e666c8 100644 --- a/src/js/gsUtils.js +++ b/src/js/gsUtils.js @@ -56,7 +56,7 @@ var gsUtils = { id, (new Date() + '').split(' ')[4], text, - ...args, + ...args ); } }, @@ -86,7 +86,7 @@ var gsUtils = { errorObj = errorObj || {}; console.log(id, (new Date() + '').split(' ')[4], 'Error:'); console.error( - gsUtils.getPrintableError(errorMessage, stackTrace, ...args), + gsUtils.getPrintableError(errorMessage, stackTrace, ...args) ); } else { // const logString = errorObj.hasOwnProperty('stack') @@ -128,22 +128,21 @@ var gsUtils = { return tab.discarded; }, - getTabUrl: function (tab) { + getTabUrl: function(tab) { return tab.url || tab.pendingUrl; }, isValidTabWithUrl: function(tab) { - if (!tab || typeof tab == "undefined") { + if (!tab || typeof tab == 'undefined') { return false; } const url = gsUtils.getTabUrl(tab); - if (url && typeof url == "string" && url.length > 0) { + if (url && typeof url == 'string' && url.length > 0) { return true; } return false; }, - //tests for non-standard web pages. does not check for suspended pages! isSpecialTab: function(tab) { if (!gsUtils.isValidTabWithUrl(tab)) { @@ -207,13 +206,8 @@ var gsUtils = { return dontSuspendAudible && tab.audible; }, - isProtectedActiveTab: function(tab) { - var dontSuspendActiveTabs = gsStorage.getOption( - gsStorage.IGNORE_ACTIVE_TABS, - ); - return ( - tgs.isCurrentFocusedTab(tab) || (dontSuspendActiveTabs && tab.active) - ); + isProtectedActiveTab: function(tabId) { + return tgs.isCurrentFocusedTab(tabId); }, // Note: Normal tabs may be in a discarded state @@ -243,10 +237,10 @@ var gsUtils = { shouldSuspendDiscardedTabs: function() { var suspendInPlaceOfDiscard = gsStorage.getOption( - gsStorage.SUSPEND_IN_PLACE_OF_DISCARD, + gsStorage.SUSPEND_IN_PLACE_OF_DISCARD ); var discardInPlaceOfSuspend = gsStorage.getOption( - gsStorage.DISCARD_IN_PLACE_OF_SUSPEND, + gsStorage.DISCARD_IN_PLACE_OF_SUSPEND ); return suspendInPlaceOfDiscard && !discardInPlaceOfSuspend; }, @@ -297,14 +291,14 @@ var gsUtils = { checkWhiteList: function(url) { return gsUtils.checkSpecificWhiteList( url, - gsStorage.getOption(gsStorage.WHITELIST), + gsStorage.getOption(gsStorage.WHITELIST) ); }, checkSpecificWhiteList: function(url, whitelistString) { var whitelistItems = whitelistString - ? whitelistString.split(/[\s\n]+/) - : [], + ? whitelistString.split(/[\s\n]+/) + : [], whitelisted; whitelisted = whitelistItems.some(function(item) { @@ -330,7 +324,7 @@ var gsUtils = { gsUtils.performPostSaveUpdates( [key], { [key]: oldWhitelistString }, - { [key]: whitelistString }, + { [key]: whitelistString } ); }, @@ -368,7 +362,7 @@ var gsUtils = { gsUtils.performPostSaveUpdates( [key], { [key]: oldWhitelistString }, - { [key]: newWhitelistString }, + { [key]: newWhitelistString } ); }, @@ -421,7 +415,7 @@ var gsUtils = { 'data-i18n-tooltip', el .getAttribute('data-i18n-tooltip') - .replace(/__MSG_(\w+)__/g, replaceTagFunc), + .replace(/__MSG_(\w+)__/g, replaceTagFunc) ); } } @@ -519,7 +513,7 @@ var gsUtils = { if (keyPair && keyPair.match(keyPairRegEx)) { valuesByKey[keyPair.replace(keyPairRegEx, '$1')] = keyPair.replace( keyPairRegEx, - '$2', + '$2' ); } }); @@ -585,7 +579,7 @@ var gsUtils = { getSuspendedTabCount: async function() { const currentTabs = await gsChrome.tabsQuery(); const currentSuspendedTabs = currentTabs.filter(tab => - gsUtils.isSuspendedTab(tab), + gsUtils.isSuspendedTab(tab) ); return currentSuspendedTabs.length; }, @@ -621,7 +615,7 @@ var gsUtils = { for (const tab of tabs) { const timerDetails = tgs.getTabStatePropForTabId( tab.id, - tgs.STATE_TIMER_DETAILS, + tgs.STATE_TIMER_DETAILS ); if ( timerDetails && @@ -638,7 +632,7 @@ var gsUtils = { performPostSaveUpdates: function( changedSettingKeys, oldValueBySettingKey, - newValueBySettingKey, + newValueBySettingKey ) { chrome.tabs.query({}, function(tabs) { tabs.forEach(function(tab) { @@ -661,7 +655,7 @@ var gsUtils = { //if theme or screenshot preferences have changed then refresh suspended tabs const updateTheme = changedSettingKeys.includes(gsStorage.THEME); const updatePreviewMode = changedSettingKeys.includes( - gsStorage.SCREEN_CAPTURE, + gsStorage.SCREEN_CAPTURE ); if (updateTheme || updatePreviewMode) { const suspendedView = tgs.getInternalViewByTabId(tab.id); @@ -674,18 +668,18 @@ var gsUtils = { suspendedView, tab, theme, - isLowContrastFavicon, + isLowContrastFavicon ); }); } if (updatePreviewMode) { const previewMode = gsStorage.getOption( - gsStorage.SCREEN_CAPTURE, + gsStorage.SCREEN_CAPTURE ); gsSuspendedTab.updatePreviewMode( suspendedView, tab, - previewMode, + previewMode ); // async. unhandled promise. } } @@ -693,7 +687,7 @@ var gsUtils = { //if discardAfterSuspend has changed then updated discarded tabs const updateDiscardAfterSuspend = changedSettingKeys.includes( - gsStorage.DISCARD_AFTER_SUSPEND, + gsStorage.DISCARD_AFTER_SUSPEND ); if ( updateDiscardAfterSuspend && @@ -712,7 +706,7 @@ var gsUtils = { //update content scripts of normal tabs const updateIgnoreForms = changedSettingKeys.includes( - gsStorage.IGNORE_FORMS, + gsStorage.IGNORE_FORMS ); if (updateIgnoreForms) { gsMessages.sendUpdateToContentScriptOfTab(tab); //async. unhandled error @@ -738,11 +732,11 @@ var gsUtils = { (changedSettingKeys.includes(gsStorage.WHITELIST) && (gsUtils.checkSpecificWhiteList( tab.url, - oldValueBySettingKey[gsStorage.WHITELIST], - ) && + oldValueBySettingKey[gsStorage.WHITELIST] + ) && !gsUtils.checkSpecificWhiteList( tab.url, - newValueBySettingKey[gsStorage.WHITELIST], + newValueBySettingKey[gsStorage.WHITELIST] ))); if (updateSuspendTime) { tgs.resetAutoSuspendTimerForTab(tab); @@ -750,7 +744,7 @@ var gsUtils = { //if SuspendInPlaceOfDiscard has changed then updated discarded tabs const updateSuspendInPlaceOfDiscard = changedSettingKeys.includes( - gsStorage.SUSPEND_IN_PLACE_OF_DISCARD, + gsStorage.SUSPEND_IN_PLACE_OF_DISCARD ); if (updateSuspendInPlaceOfDiscard && gsUtils.isDiscardedTab(tab)) { gsTabDiscardManager.handleDiscardedUnsuspendedTab(tab); //async. unhandled promise. @@ -908,7 +902,7 @@ var gsUtils = { promiseFn, fnArgsArray, maxRetries, - retryWaitTime, + retryWaitTime ) { const retryFn = async retries => { try { diff --git a/src/manifest.json b/src/manifest.json index ebd9af9f..478a7bbd 100644 --- a/src/manifest.json +++ b/src/manifest.json @@ -1,4 +1,5 @@ { + "manifest_version": 3, "name": "__MSG_ext_extension_name__", "description": "__MSG_ext_extension_description__", "version": "7.1.6.3", @@ -8,45 +9,20 @@ "storage", "history", "unlimitedStorage", - "chrome://favicon/*", "contextMenus", - "cookies", - "http://*/*", - "https://*/*", - "file://*/*" + "cookies" ], + "host_permissions": ["http://*/*", "https://*/*"], "background": { - "scripts": [ - "js/gsUtils.js", - "js/gsChrome.js", - "js/gsStorage.js", - "js/db.js", - "js/gsIndexedDb.js", - "js/gsMessages.js", - "js/gsSession.js", - "js/gsTabQueue.js", - "js/gsTabCheckManager.js", - "js/gsFavicon.js", - "js/gsTabSuspendManager.js", - "js/gsTabDiscardManager.js", - "js/gsSuspendedTab.js", - "js/background.js" - ], - "persistent": true + "service_worker": "background.js" }, "content_scripts": [ { - "matches": [ - "http://*/*", - "https://*/*", - "file://*/*" - ], - "js": [ - "js/contentscript.js" - ] + "matches": ["http://*/*", "https://*/*"], + "js": ["js/contentscript.js"] } ], - "browser_action": { + "action": { "default_title": "__MSG_ext_default_title__", "default_icon": { "16": "img/ic_suspendy_16x16.png", @@ -54,7 +30,10 @@ }, "default_popup": "popup.html" }, - "options_page": "options.html", + "options_ui": { + "page": "options.html", + "open_in_tab": true + }, "icons": { "16": "img/ic_suspendy_16x16.png", "32": "img/ic_suspendy_32x32.png", @@ -62,12 +41,17 @@ "128": "img/ic_suspendy_128x128.png" }, "web_accessible_resources": [ - "suspended.html" + { + "resources": ["suspended.html"], + "matches": [""] + } ], - "content_security_policy": "script-src 'self'; object-src 'self'; child-src 'self'; connect-src 'self'; img-src 'self' data: chrome:; style-src 'self'; default-src 'self'", + "content_security_policy": { + "extension_pages": + "script-src 'self'; object-src 'self'; child-src 'self'; connect-src 'self'; img-src 'self' data: chrome:; style-src 'self'; default-src 'self'; worker-src 'self'" + }, "incognito": "split", - "manifest_version": 2, - "minimum_chrome_version": "55", + "minimum_chrome_version": "88", "commands": { "1-suspend-tab": { "description": "__MSG_ext_cmd_toggle_tab_suspension_description__",