├── .github └── dependabot.yml ├── .gitignore ├── .travis.yml ├── LICENSE ├── README.md ├── package-lock.json ├── package.json ├── src ├── breaker.test.ts ├── helpers.ts ├── index.ts └── types.ts ├── tsconfig.json └── tslint.json /.github/dependabot.yml: -------------------------------------------------------------------------------- 1 | version: 2 2 | updates: 3 | - package-ecosystem: npm 4 | directory: "/" 5 | schedule: 6 | interval: monthly 7 | time: "02:00" 8 | open-pull-requests-limit: 10 9 | target-branch: develop 10 | ignore: 11 | - dependency-name: chai 12 | versions: 13 | - 4.3.0 14 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | 8 | # Runtime data 9 | pids 10 | *.pid 11 | *.seed 12 | *.pid.lock 13 | 14 | # Directory for instrumented libs generated by jscoverage/JSCover 15 | lib-cov 16 | 17 | # Coverage directory used by tools like istanbul 18 | coverage 19 | 20 | # nyc test coverage 21 | .nyc_output 22 | 23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 24 | .grunt 25 | 26 | # Bower dependency directory (https://bower.io/) 27 | bower_components 28 | 29 | # node-waf configuration 30 | .lock-wscript 31 | 32 | # Compiled binary addons (https://nodejs.org/api/addons.html) 33 | build/Release 34 | 35 | # Dependency directories 36 | node_modules/ 37 | jspm_packages/ 38 | 39 | # TypeScript v1 declaration files 40 | typings/ 41 | 42 | # Optional npm cache directory 43 | .npm 44 | 45 | # Optional eslint cache 46 | .eslintcache 47 | 48 | # Optional REPL history 49 | .node_repl_history 50 | 51 | # Output of 'npm pack' 52 | *.tgz 53 | 54 | # Yarn Integrity file 55 | .yarn-integrity 56 | 57 | # dotenv environment variables file 58 | .env 59 | 60 | # next.js build output 61 | .next 62 | 63 | # vuepress build output 64 | .vuepress/dist 65 | 66 | # Serverless directories 67 | .serverless 68 | 69 | .vscode/ 70 | dist/ 71 | trash/ 72 | lib/ -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | language: node_js 2 | node_js: 3 | - '10' 4 | 5 | stages: 6 | - test 7 | - name: npm release 8 | if: tag IS present 9 | 10 | jobs: 11 | include: 12 | - stage: test 13 | script: npm test 14 | - stage: npm release 15 | if: tag IS present 16 | node_js: "10" 17 | script: npm run build 18 | deploy: 19 | provider: npm 20 | email: yuriy.bogomolov@gmail.com 21 | api_key: 22 | secure: qRGnKupEfkYgYmS6bFwFKCv/0ctHez3xj5xKkYFBeIky8OcNE6RSFfZPXH5ZAeaLiyLC7V1VfU7JbKbYWn+wqKly/NromGyg1na96yDxlMxanVNeNca/C2QFc6lScNqu/PE5NN1ot3lclDtJ/V8R5TAa8CKYHj0Wp55MHLonxZ5/Fl0gadc3GZrTJzg3N8fLTRPylJv24RrvR+28Hnai3tO6zOAqNmAAKIl1MjjAXN59zHmqVhIP4wvdaRQaWvt7IqcFBI9AzdqXHlvUEDekKPwBvd6vFgTbybhl3Vm9y2QX27yEbR1O7X+6mxlkikMSbYUaVLIpRDikDFFWmSmhe6gCxX/ZVc7B9M7Dy5W6EfHc/lNEzL5mslD6QaF4xD8Bmvw48ZxSk7Q5Heh4qJytU3TjfY/5iNN/prZioiIizJPz/E0nVHSv3dv2jBeM6h1FY7+H+QozlLADNK9HwXli0hwtmjxu7kFoHjR/kL645eekw1sp8auBp+pdECV7dQn5CDln6nES4qmPWclwJO7vWzp+WJm7qHUVAFloGn1bwQ9jdw+eE/UJNsLeDaRL9fm7tEqwH2I7SHC7OqhSBZ3QnAGnq++en0hdOpf//t7TkmhcNv9Bme3k1juA/fMuRCWSLinKEav5B6/B9kVEAtmuy303b4WLCfJs3kz52bX9ndI= 23 | skip_cleanup: true 24 | on: 25 | tags: true -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2018-2019 Yuriy Bogomolov 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Circuit Breaker pattern as a monad 2 | 3 | [![npm](https://img.shields.io/npm/v/circuit-breaker-monad.svg)](https://www.npmjs.com/package/circuit-breaker-monad) 4 | [![Build Status](https://travis-ci.org/YBogomolov/circuit-breaker-monad.svg?branch=master)](https://travis-ci.org/YBogomolov/circuit-breaker-monad) 5 | 6 | Part of [fp-ts](https://github.com/gcanti/fp-ts) ecosystem. 7 | 8 | TypeScript implementation of Circuit Breaker pattern. Adaptation of [Glue.CircuitBreaker](https://hackage.haskell.org/package/glue-core-0.4.2/docs/Glue-CircuitBreaker.html) module from Haskell. 9 | 10 | You can read a bit more abou the pattern and this implementation [in my article](https://medium.com/@yuriybogomolov/circuit-breaker-in-a-functional-world-9c555c8e9527). 11 | 12 | ## Usage example 13 | 14 | First of all, you need to install the package: 15 | 16 | ```sh 17 | npm install circuit-breaker-monad 18 | ``` 19 | 20 | Then import the main `circuitBreaker` function: 21 | 22 | ```ts 23 | import { circuitBreaker } from 'circuit-breaker-monad/lib'; 24 | ``` 25 | 26 | This function returns a `Reader`, which, given the corresponding `BreakerOptions`, creates an enhanced fetcher – a function which takes any `Lazy>` instance and returns a tuple of `IORef` and `TaskEither`. The first element of this tuple is current circuit breaker status, implemented via `IORef` (mutable reference in the IO monad), and the second element of the tuple is ready-to-be-called `TaskEither`. 27 | 28 | Let's look at the usage example: 29 | 30 | ```ts 31 | import { fold } from 'fp-ts/lib/Either'; 32 | import { IORef } from 'fp-ts/lib/IORef'; 33 | 34 | import { circuitBreaker, defaultBreakerOptions } from 'circuit-breaker-monad/lib'; 35 | import { BreakerClosed } from 'circuit-breaker-monad/lib/types'; 36 | 37 | const fetcher = circuitBreaker()(defaultBreakerOptions); 38 | 39 | const main = async () => { 40 | const request = () => fetch('http://my-domain.com/my-data.json').then((res) => res.json()); 41 | const breakerState = new IORef(new BreakerClosed(0)); // initial breaker state 42 | const [result, ref] = fetcher({ request, breakerState }); 43 | const response = await result(); 44 | 45 | fold( 46 | (e: Error) => { }, 47 | (myData) => { }, 48 | )(response); 49 | 50 | // ref :: BreakerClosed { errorCount: 0 } 51 | // result :: TaskEither 52 | // response :: Either 53 | // myData :: TMyJsonData 54 | }; 55 | ``` 56 | 57 | The `ref` variable is resulting circuit breaker status, which can be passed to the next call, so you take the full control over what's going on inside the circuit breaker: 58 | 59 | ```ts 60 | const [ref, result] = fetcher(promise); 61 | const myData1 = await result(); 62 | const [, result2] = fetcher(promise, ref); 63 | const myData2 = await result2(); 64 | // here `ref` may be 'Open' if the second call to the service has failed 65 | ``` 66 | 67 | ### Configuration 68 | 69 | Circuit breaker may be configured by passing these parameters to the Reader: 70 | 71 | - `maxBreakerFailures` - how many times the underlying service must fail in the given window before the circuit opens; 72 | - `resetTimeoutSecs` - the window of time in which the underlying service must fail for the circuit to open, seconds; 73 | - `breakerDescription` - description that is attached to the failure so as to identify the particular circuit. 74 | 75 | Default options are: 76 | 77 | ```ts 78 | export const defaultBreakerOptions: BreakerOptions = { 79 | maxBreakerFailures: 3, 80 | resetTimeoutSecs: 60, 81 | breakerDescription: 'Circuit breaker is closed', 82 | }; 83 | ``` 84 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "circuit-breaker-monad", 3 | "version": "1.0.1", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@babel/code-frame": { 8 | "version": "7.5.5", 9 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.5.5.tgz", 10 | "integrity": "sha512-27d4lZoomVyo51VegxI20xZPuSHusqbQag/ztrBC7wegWoQ1nLREPVSKSW8byhTlzTKyNE4ifaTA6lCp7JjpFw==", 11 | "dev": true, 12 | "requires": { 13 | "@babel/highlight": "^7.0.0" 14 | } 15 | }, 16 | "@babel/core": { 17 | "version": "7.12.9", 18 | "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.12.9.tgz", 19 | "integrity": "sha512-gTXYh3M5wb7FRXQy+FErKFAv90BnlOuNn1QkCK2lREoPAjrQCO49+HVSrFoe5uakFAF5eenS75KbO2vQiLrTMQ==", 20 | "dev": true, 21 | "requires": { 22 | "@babel/code-frame": "^7.10.4", 23 | "@babel/generator": "^7.12.5", 24 | "@babel/helper-module-transforms": "^7.12.1", 25 | "@babel/helpers": "^7.12.5", 26 | "@babel/parser": "^7.12.7", 27 | "@babel/template": "^7.12.7", 28 | "@babel/traverse": "^7.12.9", 29 | "@babel/types": "^7.12.7", 30 | "convert-source-map": "^1.7.0", 31 | "debug": "^4.1.0", 32 | "gensync": "^1.0.0-beta.1", 33 | "json5": "^2.1.2", 34 | "lodash": "^4.17.19", 35 | "resolve": "^1.3.2", 36 | "semver": "^5.4.1", 37 | "source-map": "^0.5.0" 38 | }, 39 | "dependencies": { 40 | "@babel/code-frame": { 41 | "version": "7.10.4", 42 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz", 43 | "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==", 44 | "dev": true, 45 | "requires": { 46 | "@babel/highlight": "^7.10.4" 47 | } 48 | }, 49 | "@babel/highlight": { 50 | "version": "7.10.4", 51 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", 52 | "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", 53 | "dev": true, 54 | "requires": { 55 | "@babel/helper-validator-identifier": "^7.10.4", 56 | "chalk": "^2.0.0", 57 | "js-tokens": "^4.0.0" 58 | } 59 | }, 60 | "source-map": { 61 | "version": "0.5.7", 62 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", 63 | "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", 64 | "dev": true 65 | } 66 | } 67 | }, 68 | "@babel/generator": { 69 | "version": "7.12.5", 70 | "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.12.5.tgz", 71 | "integrity": "sha512-m16TQQJ8hPt7E+OS/XVQg/7U184MLXtvuGbCdA7na61vha+ImkyyNM/9DDA0unYCVZn3ZOhng+qz48/KBOT96A==", 72 | "dev": true, 73 | "requires": { 74 | "@babel/types": "^7.12.5", 75 | "jsesc": "^2.5.1", 76 | "source-map": "^0.5.0" 77 | }, 78 | "dependencies": { 79 | "source-map": { 80 | "version": "0.5.7", 81 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", 82 | "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", 83 | "dev": true 84 | } 85 | } 86 | }, 87 | "@babel/helper-function-name": { 88 | "version": "7.10.4", 89 | "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.10.4.tgz", 90 | "integrity": "sha512-YdaSyz1n8gY44EmN7x44zBn9zQ1Ry2Y+3GTA+3vH6Mizke1Vw0aWDM66FOYEPw8//qKkmqOckrGgTYa+6sceqQ==", 91 | "dev": true, 92 | "requires": { 93 | "@babel/helper-get-function-arity": "^7.10.4", 94 | "@babel/template": "^7.10.4", 95 | "@babel/types": "^7.10.4" 96 | } 97 | }, 98 | "@babel/helper-get-function-arity": { 99 | "version": "7.10.4", 100 | "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.10.4.tgz", 101 | "integrity": "sha512-EkN3YDB+SRDgiIUnNgcmiD361ti+AVbL3f3Henf6dqqUyr5dMsorno0lJWJuLhDhkI5sYEpgj6y9kB8AOU1I2A==", 102 | "dev": true, 103 | "requires": { 104 | "@babel/types": "^7.10.4" 105 | } 106 | }, 107 | "@babel/helper-member-expression-to-functions": { 108 | "version": "7.12.7", 109 | "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.12.7.tgz", 110 | "integrity": "sha512-DCsuPyeWxeHgh1Dus7APn7iza42i/qXqiFPWyBDdOFtvS581JQePsc1F/nD+fHrcswhLlRc2UpYS1NwERxZhHw==", 111 | "dev": true, 112 | "requires": { 113 | "@babel/types": "^7.12.7" 114 | } 115 | }, 116 | "@babel/helper-module-imports": { 117 | "version": "7.12.5", 118 | "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.12.5.tgz", 119 | "integrity": "sha512-SR713Ogqg6++uexFRORf/+nPXMmWIn80TALu0uaFb+iQIUoR7bOC7zBWyzBs5b3tBBJXuyD0cRu1F15GyzjOWA==", 120 | "dev": true, 121 | "requires": { 122 | "@babel/types": "^7.12.5" 123 | } 124 | }, 125 | "@babel/helper-module-transforms": { 126 | "version": "7.12.1", 127 | "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.12.1.tgz", 128 | "integrity": "sha512-QQzehgFAZ2bbISiCpmVGfiGux8YVFXQ0abBic2Envhej22DVXV9nCFaS5hIQbkyo1AdGb+gNME2TSh3hYJVV/w==", 129 | "dev": true, 130 | "requires": { 131 | "@babel/helper-module-imports": "^7.12.1", 132 | "@babel/helper-replace-supers": "^7.12.1", 133 | "@babel/helper-simple-access": "^7.12.1", 134 | "@babel/helper-split-export-declaration": "^7.11.0", 135 | "@babel/helper-validator-identifier": "^7.10.4", 136 | "@babel/template": "^7.10.4", 137 | "@babel/traverse": "^7.12.1", 138 | "@babel/types": "^7.12.1", 139 | "lodash": "^4.17.19" 140 | } 141 | }, 142 | "@babel/helper-optimise-call-expression": { 143 | "version": "7.12.7", 144 | "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.12.7.tgz", 145 | "integrity": "sha512-I5xc9oSJ2h59OwyUqjv95HRyzxj53DAubUERgQMrpcCEYQyToeHA+NEcUEsVWB4j53RDeskeBJ0SgRAYHDBckw==", 146 | "dev": true, 147 | "requires": { 148 | "@babel/types": "^7.12.7" 149 | } 150 | }, 151 | "@babel/helper-replace-supers": { 152 | "version": "7.12.5", 153 | "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.12.5.tgz", 154 | "integrity": "sha512-5YILoed0ZyIpF4gKcpZitEnXEJ9UoDRki1Ey6xz46rxOzfNMAhVIJMoune1hmPVxh40LRv1+oafz7UsWX+vyWA==", 155 | "dev": true, 156 | "requires": { 157 | "@babel/helper-member-expression-to-functions": "^7.12.1", 158 | "@babel/helper-optimise-call-expression": "^7.10.4", 159 | "@babel/traverse": "^7.12.5", 160 | "@babel/types": "^7.12.5" 161 | } 162 | }, 163 | "@babel/helper-simple-access": { 164 | "version": "7.12.1", 165 | "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.12.1.tgz", 166 | "integrity": "sha512-OxBp7pMrjVewSSC8fXDFrHrBcJATOOFssZwv16F3/6Xtc138GHybBfPbm9kfiqQHKhYQrlamWILwlDCeyMFEaA==", 167 | "dev": true, 168 | "requires": { 169 | "@babel/types": "^7.12.1" 170 | } 171 | }, 172 | "@babel/helper-split-export-declaration": { 173 | "version": "7.11.0", 174 | "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.11.0.tgz", 175 | "integrity": "sha512-74Vejvp6mHkGE+m+k5vHY93FX2cAtrw1zXrZXRlG4l410Nm9PxfEiVTn1PjDPV5SnmieiueY4AFg2xqhNFuuZg==", 176 | "dev": true, 177 | "requires": { 178 | "@babel/types": "^7.11.0" 179 | } 180 | }, 181 | "@babel/helper-validator-identifier": { 182 | "version": "7.10.4", 183 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz", 184 | "integrity": "sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==", 185 | "dev": true 186 | }, 187 | "@babel/helpers": { 188 | "version": "7.12.5", 189 | "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.12.5.tgz", 190 | "integrity": "sha512-lgKGMQlKqA8meJqKsW6rUnc4MdUk35Ln0ATDqdM1a/UpARODdI4j5Y5lVfUScnSNkJcdCRAaWkspykNoFg9sJA==", 191 | "dev": true, 192 | "requires": { 193 | "@babel/template": "^7.10.4", 194 | "@babel/traverse": "^7.12.5", 195 | "@babel/types": "^7.12.5" 196 | } 197 | }, 198 | "@babel/highlight": { 199 | "version": "7.5.0", 200 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.5.0.tgz", 201 | "integrity": "sha512-7dV4eu9gBxoM0dAnj/BCFDW9LFU0zvTrkq0ugM7pnHEgguOEeOz1so2ZghEdzviYzQEED0r4EAgpsBChKy1TRQ==", 202 | "dev": true, 203 | "requires": { 204 | "chalk": "^2.0.0", 205 | "esutils": "^2.0.2", 206 | "js-tokens": "^4.0.0" 207 | } 208 | }, 209 | "@babel/parser": { 210 | "version": "7.12.7", 211 | "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.12.7.tgz", 212 | "integrity": "sha512-oWR02Ubp4xTLCAqPRiNIuMVgNO5Aif/xpXtabhzW2HWUD47XJsAB4Zd/Rg30+XeQA3juXigV7hlquOTmwqLiwg==", 213 | "dev": true 214 | }, 215 | "@babel/runtime": { 216 | "version": "7.12.5", 217 | "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.12.5.tgz", 218 | "integrity": "sha512-plcc+hbExy3McchJCEQG3knOsuh3HH+Prx1P6cLIkET/0dLuQDEnrT+s27Axgc9bqfsmNUNHfscgMUdBpC9xfg==", 219 | "dev": true, 220 | "requires": { 221 | "regenerator-runtime": "^0.13.4" 222 | } 223 | }, 224 | "@babel/template": { 225 | "version": "7.12.7", 226 | "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.12.7.tgz", 227 | "integrity": "sha512-GkDzmHS6GV7ZeXfJZ0tLRBhZcMcY0/Lnb+eEbXDBfCAcZCjrZKe6p3J4we/D24O9Y8enxWAg1cWwof59yLh2ow==", 228 | "dev": true, 229 | "requires": { 230 | "@babel/code-frame": "^7.10.4", 231 | "@babel/parser": "^7.12.7", 232 | "@babel/types": "^7.12.7" 233 | }, 234 | "dependencies": { 235 | "@babel/code-frame": { 236 | "version": "7.10.4", 237 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz", 238 | "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==", 239 | "dev": true, 240 | "requires": { 241 | "@babel/highlight": "^7.10.4" 242 | } 243 | }, 244 | "@babel/highlight": { 245 | "version": "7.10.4", 246 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", 247 | "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", 248 | "dev": true, 249 | "requires": { 250 | "@babel/helper-validator-identifier": "^7.10.4", 251 | "chalk": "^2.0.0", 252 | "js-tokens": "^4.0.0" 253 | } 254 | } 255 | } 256 | }, 257 | "@babel/traverse": { 258 | "version": "7.12.9", 259 | "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.12.9.tgz", 260 | "integrity": "sha512-iX9ajqnLdoU1s1nHt36JDI9KG4k+vmI8WgjK5d+aDTwQbL2fUnzedNedssA645Ede3PM2ma1n8Q4h2ohwXgMXw==", 261 | "dev": true, 262 | "requires": { 263 | "@babel/code-frame": "^7.10.4", 264 | "@babel/generator": "^7.12.5", 265 | "@babel/helper-function-name": "^7.10.4", 266 | "@babel/helper-split-export-declaration": "^7.11.0", 267 | "@babel/parser": "^7.12.7", 268 | "@babel/types": "^7.12.7", 269 | "debug": "^4.1.0", 270 | "globals": "^11.1.0", 271 | "lodash": "^4.17.19" 272 | }, 273 | "dependencies": { 274 | "@babel/code-frame": { 275 | "version": "7.10.4", 276 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz", 277 | "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==", 278 | "dev": true, 279 | "requires": { 280 | "@babel/highlight": "^7.10.4" 281 | } 282 | }, 283 | "@babel/highlight": { 284 | "version": "7.10.4", 285 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", 286 | "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", 287 | "dev": true, 288 | "requires": { 289 | "@babel/helper-validator-identifier": "^7.10.4", 290 | "chalk": "^2.0.0", 291 | "js-tokens": "^4.0.0" 292 | } 293 | } 294 | } 295 | }, 296 | "@babel/types": { 297 | "version": "7.12.7", 298 | "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.7.tgz", 299 | "integrity": "sha512-MNyI92qZq6jrQkXvtIiykvl4WtoRrVV9MPn+ZfsoEENjiWcBQ3ZSHrkxnJWgWtLX3XXqX5hrSQ+X69wkmesXuQ==", 300 | "dev": true, 301 | "requires": { 302 | "@babel/helper-validator-identifier": "^7.10.4", 303 | "lodash": "^4.17.19", 304 | "to-fast-properties": "^2.0.0" 305 | } 306 | }, 307 | "@types/chai": { 308 | "version": "4.2.22", 309 | "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.2.22.tgz", 310 | "integrity": "sha512-tFfcE+DSTzWAgifkjik9AySNqIyNoYwmR+uecPwwD/XRNfvOjmC/FjCxpiUGDkDVDphPfCUecSQVFw+lN3M3kQ==", 311 | "dev": true 312 | }, 313 | "@types/fetch-mock": { 314 | "version": "7.3.5", 315 | "resolved": "https://registry.npmjs.org/@types/fetch-mock/-/fetch-mock-7.3.5.tgz", 316 | "integrity": "sha512-sLecm9ohBdGIpYUP9rWk5/XIKY2xHMYTBJIcJuBBM8IJWnYoQ1DAj8F4OVjnfD0API1drlkWEV0LPNk+ACuhsg==", 317 | "dev": true 318 | }, 319 | "@types/mocha": { 320 | "version": "9.0.0", 321 | "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-9.0.0.tgz", 322 | "integrity": "sha512-scN0hAWyLVAvLR9AyW7HoFF5sJZglyBsbPuHO4fv7JRvfmPBMfp1ozWqOf/e4wwPNxezBZXRfWzMb6iFLgEVRA==", 323 | "dev": true 324 | }, 325 | "@ungap/promise-all-settled": { 326 | "version": "1.1.2", 327 | "resolved": "https://registry.npmjs.org/@ungap/promise-all-settled/-/promise-all-settled-1.1.2.tgz", 328 | "integrity": "sha512-sL/cEvJWAnClXw0wHk85/2L0G6Sj8UB0Ctc1TEMbKSsmpRosqhwj9gWgFRZSrBr2f9tiXISwNhCPmlfqUqyb9Q==", 329 | "dev": true 330 | }, 331 | "ansi-colors": { 332 | "version": "4.1.1", 333 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", 334 | "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", 335 | "dev": true 336 | }, 337 | "ansi-regex": { 338 | "version": "5.0.1", 339 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 340 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 341 | "dev": true 342 | }, 343 | "ansi-styles": { 344 | "version": "3.2.1", 345 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 346 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 347 | "dev": true, 348 | "requires": { 349 | "color-convert": "^1.9.0" 350 | } 351 | }, 352 | "anymatch": { 353 | "version": "3.1.2", 354 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 355 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 356 | "dev": true, 357 | "requires": { 358 | "normalize-path": "^3.0.0", 359 | "picomatch": "^2.0.4" 360 | } 361 | }, 362 | "arg": { 363 | "version": "4.1.3", 364 | "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", 365 | "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", 366 | "dev": true 367 | }, 368 | "argparse": { 369 | "version": "1.0.10", 370 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 371 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 372 | "dev": true, 373 | "requires": { 374 | "sprintf-js": "~1.0.2" 375 | } 376 | }, 377 | "assertion-error": { 378 | "version": "1.1.0", 379 | "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz", 380 | "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==", 381 | "dev": true 382 | }, 383 | "balanced-match": { 384 | "version": "1.0.0", 385 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 386 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 387 | "dev": true 388 | }, 389 | "binary-extensions": { 390 | "version": "2.2.0", 391 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 392 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 393 | "dev": true 394 | }, 395 | "brace-expansion": { 396 | "version": "1.1.11", 397 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 398 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 399 | "dev": true, 400 | "requires": { 401 | "balanced-match": "^1.0.0", 402 | "concat-map": "0.0.1" 403 | } 404 | }, 405 | "braces": { 406 | "version": "3.0.2", 407 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 408 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 409 | "dev": true, 410 | "requires": { 411 | "fill-range": "^7.0.1" 412 | } 413 | }, 414 | "browser-stdout": { 415 | "version": "1.3.1", 416 | "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", 417 | "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", 418 | "dev": true 419 | }, 420 | "buffer-from": { 421 | "version": "1.1.1", 422 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", 423 | "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", 424 | "dev": true 425 | }, 426 | "builtin-modules": { 427 | "version": "1.1.1", 428 | "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", 429 | "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", 430 | "dev": true 431 | }, 432 | "camelcase": { 433 | "version": "6.2.0", 434 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.2.0.tgz", 435 | "integrity": "sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg==", 436 | "dev": true 437 | }, 438 | "chai": { 439 | "version": "4.3.4", 440 | "resolved": "https://registry.npmjs.org/chai/-/chai-4.3.4.tgz", 441 | "integrity": "sha512-yS5H68VYOCtN1cjfwumDSuzn/9c+yza4f3reKXlE5rUg7SFcCEy90gJvydNgOYtblyf4Zi6jIWRnXOgErta0KA==", 442 | "dev": true, 443 | "requires": { 444 | "assertion-error": "^1.1.0", 445 | "check-error": "^1.0.2", 446 | "deep-eql": "^3.0.1", 447 | "get-func-name": "^2.0.0", 448 | "pathval": "^1.1.1", 449 | "type-detect": "^4.0.5" 450 | } 451 | }, 452 | "chalk": { 453 | "version": "2.4.2", 454 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 455 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 456 | "dev": true, 457 | "requires": { 458 | "ansi-styles": "^3.2.1", 459 | "escape-string-regexp": "^1.0.5", 460 | "supports-color": "^5.3.0" 461 | } 462 | }, 463 | "check-error": { 464 | "version": "1.0.2", 465 | "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz", 466 | "integrity": "sha1-V00xLt2Iu13YkS6Sht1sCu1KrII=", 467 | "dev": true 468 | }, 469 | "chokidar": { 470 | "version": "3.5.2", 471 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", 472 | "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", 473 | "dev": true, 474 | "requires": { 475 | "anymatch": "~3.1.2", 476 | "braces": "~3.0.2", 477 | "fsevents": "~2.3.2", 478 | "glob-parent": "~5.1.2", 479 | "is-binary-path": "~2.1.0", 480 | "is-glob": "~4.0.1", 481 | "normalize-path": "~3.0.0", 482 | "readdirp": "~3.6.0" 483 | } 484 | }, 485 | "cliui": { 486 | "version": "7.0.4", 487 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", 488 | "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", 489 | "dev": true, 490 | "requires": { 491 | "string-width": "^4.2.0", 492 | "strip-ansi": "^6.0.0", 493 | "wrap-ansi": "^7.0.0" 494 | } 495 | }, 496 | "color-convert": { 497 | "version": "1.9.3", 498 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 499 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 500 | "dev": true, 501 | "requires": { 502 | "color-name": "1.1.3" 503 | } 504 | }, 505 | "color-name": { 506 | "version": "1.1.3", 507 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 508 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 509 | "dev": true 510 | }, 511 | "commander": { 512 | "version": "2.20.3", 513 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 514 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", 515 | "dev": true 516 | }, 517 | "concat-map": { 518 | "version": "0.0.1", 519 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 520 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 521 | "dev": true 522 | }, 523 | "convert-source-map": { 524 | "version": "1.7.0", 525 | "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.7.0.tgz", 526 | "integrity": "sha512-4FJkXzKXEDB1snCFZlLP4gpC3JILicCpGbzG9f9G7tGqGCzETQ2hWPrcinA9oU4wtf2biUaEH5065UnMeR33oA==", 527 | "dev": true, 528 | "requires": { 529 | "safe-buffer": "~5.1.1" 530 | }, 531 | "dependencies": { 532 | "safe-buffer": { 533 | "version": "5.1.2", 534 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 535 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 536 | "dev": true 537 | } 538 | } 539 | }, 540 | "core-js": { 541 | "version": "3.8.0", 542 | "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.8.0.tgz", 543 | "integrity": "sha512-W2VYNB0nwQQE7tKS7HzXd7r2y/y2SVJl4ga6oH/dnaLFzM0o2lB2P3zCkWj5Wc/zyMYjtgd5Hmhk0ObkQFZOIA==", 544 | "dev": true 545 | }, 546 | "create-require": { 547 | "version": "1.1.1", 548 | "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", 549 | "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", 550 | "dev": true 551 | }, 552 | "data-uri-to-buffer": { 553 | "version": "3.0.1", 554 | "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-3.0.1.tgz", 555 | "integrity": "sha512-WboRycPNsVw3B3TL559F7kuBUM4d8CgMEvk6xEJlOp7OBPjt6G7z8WMWlD2rOFZLk6OYfFIUGsCOWzcQH9K2og==", 556 | "dev": true 557 | }, 558 | "debug": { 559 | "version": "4.2.0", 560 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.2.0.tgz", 561 | "integrity": "sha512-IX2ncY78vDTjZMFUdmsvIRFY2Cf4FnD0wRs+nQwJU8Lu99/tPFdb0VybiiMTPe3I6rQmwsqQqRBvxU+bZ/I8sg==", 562 | "dev": true, 563 | "requires": { 564 | "ms": "2.1.2" 565 | }, 566 | "dependencies": { 567 | "ms": { 568 | "version": "2.1.2", 569 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 570 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 571 | "dev": true 572 | } 573 | } 574 | }, 575 | "decamelize": { 576 | "version": "4.0.0", 577 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-4.0.0.tgz", 578 | "integrity": "sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==", 579 | "dev": true 580 | }, 581 | "deep-eql": { 582 | "version": "3.0.1", 583 | "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz", 584 | "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==", 585 | "dev": true, 586 | "requires": { 587 | "type-detect": "^4.0.0" 588 | } 589 | }, 590 | "diff": { 591 | "version": "4.0.2", 592 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", 593 | "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", 594 | "dev": true 595 | }, 596 | "emoji-regex": { 597 | "version": "8.0.0", 598 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 599 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 600 | "dev": true 601 | }, 602 | "escalade": { 603 | "version": "3.1.1", 604 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 605 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 606 | "dev": true 607 | }, 608 | "escape-string-regexp": { 609 | "version": "1.0.5", 610 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 611 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 612 | "dev": true 613 | }, 614 | "esprima": { 615 | "version": "4.0.1", 616 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 617 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 618 | "dev": true 619 | }, 620 | "esutils": { 621 | "version": "2.0.3", 622 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 623 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 624 | "dev": true 625 | }, 626 | "fetch-blob": { 627 | "version": "3.1.2", 628 | "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.1.2.tgz", 629 | "integrity": "sha512-hunJbvy/6OLjCD0uuhLdp0mMPzP/yd2ssd1t2FCJsaA7wkWhpbp9xfuNVpv7Ll4jFhzp6T4LAupSiV9uOeg0VQ==", 630 | "dev": true, 631 | "requires": { 632 | "web-streams-polyfill": "^3.0.3" 633 | } 634 | }, 635 | "fetch-mock": { 636 | "version": "9.11.0", 637 | "resolved": "https://registry.npmjs.org/fetch-mock/-/fetch-mock-9.11.0.tgz", 638 | "integrity": "sha512-PG1XUv+x7iag5p/iNHD4/jdpxL9FtVSqRMUQhPab4hVDt80T1MH5ehzVrL2IdXO9Q2iBggArFvPqjUbHFuI58Q==", 639 | "dev": true, 640 | "requires": { 641 | "@babel/core": "^7.0.0", 642 | "@babel/runtime": "^7.0.0", 643 | "core-js": "^3.0.0", 644 | "debug": "^4.1.1", 645 | "glob-to-regexp": "^0.4.0", 646 | "is-subset": "^0.1.1", 647 | "lodash.isequal": "^4.5.0", 648 | "path-to-regexp": "^2.2.1", 649 | "querystring": "^0.2.0", 650 | "whatwg-url": "^6.5.0" 651 | } 652 | }, 653 | "fill-range": { 654 | "version": "7.0.1", 655 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 656 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 657 | "dev": true, 658 | "requires": { 659 | "to-regex-range": "^5.0.1" 660 | } 661 | }, 662 | "find-up": { 663 | "version": "5.0.0", 664 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", 665 | "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", 666 | "dev": true, 667 | "requires": { 668 | "locate-path": "^6.0.0", 669 | "path-exists": "^4.0.0" 670 | } 671 | }, 672 | "flat": { 673 | "version": "5.0.2", 674 | "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", 675 | "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", 676 | "dev": true 677 | }, 678 | "fp-ts": { 679 | "version": "2.11.5", 680 | "resolved": "https://registry.npmjs.org/fp-ts/-/fp-ts-2.11.5.tgz", 681 | "integrity": "sha512-OqlwJq1BdpB83BZXTqI+dNcA6uYk6qk4u9Cgnt64Y+XS7dwdbp/mobx8S2KXf2AXH+scNmA/UVK3SEFHR3vHZA==", 682 | "dev": true 683 | }, 684 | "fs.realpath": { 685 | "version": "1.0.0", 686 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 687 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 688 | "dev": true 689 | }, 690 | "fsevents": { 691 | "version": "2.3.2", 692 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 693 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 694 | "dev": true, 695 | "optional": true 696 | }, 697 | "gensync": { 698 | "version": "1.0.0-beta.2", 699 | "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", 700 | "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", 701 | "dev": true 702 | }, 703 | "get-caller-file": { 704 | "version": "2.0.5", 705 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 706 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 707 | "dev": true 708 | }, 709 | "get-func-name": { 710 | "version": "2.0.0", 711 | "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.0.tgz", 712 | "integrity": "sha1-6td0q+5y4gQJQzoGY2YCPdaIekE=", 713 | "dev": true 714 | }, 715 | "glob": { 716 | "version": "7.1.6", 717 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", 718 | "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", 719 | "dev": true, 720 | "requires": { 721 | "fs.realpath": "^1.0.0", 722 | "inflight": "^1.0.4", 723 | "inherits": "2", 724 | "minimatch": "^3.0.4", 725 | "once": "^1.3.0", 726 | "path-is-absolute": "^1.0.0" 727 | } 728 | }, 729 | "glob-parent": { 730 | "version": "5.1.2", 731 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 732 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 733 | "dev": true, 734 | "requires": { 735 | "is-glob": "^4.0.1" 736 | } 737 | }, 738 | "glob-to-regexp": { 739 | "version": "0.4.1", 740 | "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz", 741 | "integrity": "sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==", 742 | "dev": true 743 | }, 744 | "globals": { 745 | "version": "11.12.0", 746 | "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", 747 | "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", 748 | "dev": true 749 | }, 750 | "growl": { 751 | "version": "1.10.5", 752 | "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz", 753 | "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==", 754 | "dev": true 755 | }, 756 | "has-flag": { 757 | "version": "3.0.0", 758 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 759 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 760 | "dev": true 761 | }, 762 | "he": { 763 | "version": "1.2.0", 764 | "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", 765 | "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", 766 | "dev": true 767 | }, 768 | "immutable": { 769 | "version": "3.8.2", 770 | "resolved": "https://registry.npmjs.org/immutable/-/immutable-3.8.2.tgz", 771 | "integrity": "sha1-wkOZUUVbs5kT2vKBN28VMOEErfM=", 772 | "dev": true 773 | }, 774 | "inflight": { 775 | "version": "1.0.6", 776 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 777 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 778 | "dev": true, 779 | "requires": { 780 | "once": "^1.3.0", 781 | "wrappy": "1" 782 | } 783 | }, 784 | "inherits": { 785 | "version": "2.0.3", 786 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 787 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", 788 | "dev": true 789 | }, 790 | "is-binary-path": { 791 | "version": "2.1.0", 792 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 793 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 794 | "dev": true, 795 | "requires": { 796 | "binary-extensions": "^2.0.0" 797 | } 798 | }, 799 | "is-extglob": { 800 | "version": "2.1.1", 801 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 802 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 803 | "dev": true 804 | }, 805 | "is-fullwidth-code-point": { 806 | "version": "3.0.0", 807 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 808 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 809 | "dev": true 810 | }, 811 | "is-glob": { 812 | "version": "4.0.3", 813 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 814 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 815 | "dev": true, 816 | "requires": { 817 | "is-extglob": "^2.1.1" 818 | } 819 | }, 820 | "is-number": { 821 | "version": "7.0.0", 822 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 823 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 824 | "dev": true 825 | }, 826 | "is-plain-obj": { 827 | "version": "2.1.0", 828 | "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-2.1.0.tgz", 829 | "integrity": "sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==", 830 | "dev": true 831 | }, 832 | "is-subset": { 833 | "version": "0.1.1", 834 | "resolved": "https://registry.npmjs.org/is-subset/-/is-subset-0.1.1.tgz", 835 | "integrity": "sha1-ilkRfZMt4d4A8kX83TnOQ/HpOaY=", 836 | "dev": true 837 | }, 838 | "is-unicode-supported": { 839 | "version": "0.1.0", 840 | "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", 841 | "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", 842 | "dev": true 843 | }, 844 | "isexe": { 845 | "version": "2.0.0", 846 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 847 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 848 | "dev": true 849 | }, 850 | "js-tokens": { 851 | "version": "4.0.0", 852 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 853 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 854 | "dev": true 855 | }, 856 | "js-yaml": { 857 | "version": "3.13.1", 858 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", 859 | "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", 860 | "dev": true, 861 | "requires": { 862 | "argparse": "^1.0.7", 863 | "esprima": "^4.0.0" 864 | } 865 | }, 866 | "jsesc": { 867 | "version": "2.5.2", 868 | "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", 869 | "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", 870 | "dev": true 871 | }, 872 | "json5": { 873 | "version": "2.1.3", 874 | "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz", 875 | "integrity": "sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==", 876 | "dev": true, 877 | "requires": { 878 | "minimist": "^1.2.5" 879 | }, 880 | "dependencies": { 881 | "minimist": { 882 | "version": "1.2.5", 883 | "resolved": "http://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 884 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", 885 | "dev": true 886 | } 887 | } 888 | }, 889 | "locate-path": { 890 | "version": "6.0.0", 891 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", 892 | "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", 893 | "dev": true, 894 | "requires": { 895 | "p-locate": "^5.0.0" 896 | } 897 | }, 898 | "lodash": { 899 | "version": "4.17.21", 900 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 901 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 902 | "dev": true 903 | }, 904 | "lodash.isequal": { 905 | "version": "4.5.0", 906 | "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", 907 | "integrity": "sha1-QVxEePK8wwEgwizhDtMib30+GOA=", 908 | "dev": true 909 | }, 910 | "lodash.sortby": { 911 | "version": "4.7.0", 912 | "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz", 913 | "integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg=", 914 | "dev": true 915 | }, 916 | "log-symbols": { 917 | "version": "4.1.0", 918 | "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", 919 | "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", 920 | "dev": true, 921 | "requires": { 922 | "chalk": "^4.1.0", 923 | "is-unicode-supported": "^0.1.0" 924 | }, 925 | "dependencies": { 926 | "ansi-styles": { 927 | "version": "4.3.0", 928 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 929 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 930 | "dev": true, 931 | "requires": { 932 | "color-convert": "^2.0.1" 933 | } 934 | }, 935 | "chalk": { 936 | "version": "4.1.2", 937 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 938 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 939 | "dev": true, 940 | "requires": { 941 | "ansi-styles": "^4.1.0", 942 | "supports-color": "^7.1.0" 943 | } 944 | }, 945 | "color-convert": { 946 | "version": "2.0.1", 947 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 948 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 949 | "dev": true, 950 | "requires": { 951 | "color-name": "~1.1.4" 952 | } 953 | }, 954 | "color-name": { 955 | "version": "1.1.4", 956 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 957 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 958 | "dev": true 959 | }, 960 | "has-flag": { 961 | "version": "4.0.0", 962 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 963 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 964 | "dev": true 965 | }, 966 | "supports-color": { 967 | "version": "7.2.0", 968 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 969 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 970 | "dev": true, 971 | "requires": { 972 | "has-flag": "^4.0.0" 973 | } 974 | } 975 | } 976 | }, 977 | "make-error": { 978 | "version": "1.3.6", 979 | "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", 980 | "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", 981 | "dev": true 982 | }, 983 | "minimatch": { 984 | "version": "3.0.4", 985 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 986 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 987 | "dev": true, 988 | "requires": { 989 | "brace-expansion": "^1.1.7" 990 | } 991 | }, 992 | "minimist": { 993 | "version": "0.0.8", 994 | "resolved": "http://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 995 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", 996 | "dev": true 997 | }, 998 | "mkdirp": { 999 | "version": "0.5.1", 1000 | "resolved": "http://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 1001 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 1002 | "dev": true, 1003 | "requires": { 1004 | "minimist": "0.0.8" 1005 | } 1006 | }, 1007 | "mocha": { 1008 | "version": "9.1.3", 1009 | "resolved": "https://registry.npmjs.org/mocha/-/mocha-9.1.3.tgz", 1010 | "integrity": "sha512-Xcpl9FqXOAYqI3j79pEtHBBnQgVXIhpULjGQa7DVb0Po+VzmSIK9kanAiWLHoRR/dbZ2qpdPshuXr8l1VaHCzw==", 1011 | "dev": true, 1012 | "requires": { 1013 | "@ungap/promise-all-settled": "1.1.2", 1014 | "ansi-colors": "4.1.1", 1015 | "browser-stdout": "1.3.1", 1016 | "chokidar": "3.5.2", 1017 | "debug": "4.3.2", 1018 | "diff": "5.0.0", 1019 | "escape-string-regexp": "4.0.0", 1020 | "find-up": "5.0.0", 1021 | "glob": "7.1.7", 1022 | "growl": "1.10.5", 1023 | "he": "1.2.0", 1024 | "js-yaml": "4.1.0", 1025 | "log-symbols": "4.1.0", 1026 | "minimatch": "3.0.4", 1027 | "ms": "2.1.3", 1028 | "nanoid": "3.1.25", 1029 | "serialize-javascript": "6.0.0", 1030 | "strip-json-comments": "3.1.1", 1031 | "supports-color": "8.1.1", 1032 | "which": "2.0.2", 1033 | "workerpool": "6.1.5", 1034 | "yargs": "16.2.0", 1035 | "yargs-parser": "20.2.4", 1036 | "yargs-unparser": "2.0.0" 1037 | }, 1038 | "dependencies": { 1039 | "argparse": { 1040 | "version": "2.0.1", 1041 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", 1042 | "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", 1043 | "dev": true 1044 | }, 1045 | "debug": { 1046 | "version": "4.3.2", 1047 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.2.tgz", 1048 | "integrity": "sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==", 1049 | "dev": true, 1050 | "requires": { 1051 | "ms": "2.1.2" 1052 | }, 1053 | "dependencies": { 1054 | "ms": { 1055 | "version": "2.1.2", 1056 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1057 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1058 | "dev": true 1059 | } 1060 | } 1061 | }, 1062 | "diff": { 1063 | "version": "5.0.0", 1064 | "resolved": "https://registry.npmjs.org/diff/-/diff-5.0.0.tgz", 1065 | "integrity": "sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==", 1066 | "dev": true 1067 | }, 1068 | "escape-string-regexp": { 1069 | "version": "4.0.0", 1070 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", 1071 | "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", 1072 | "dev": true 1073 | }, 1074 | "glob": { 1075 | "version": "7.1.7", 1076 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", 1077 | "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==", 1078 | "dev": true, 1079 | "requires": { 1080 | "fs.realpath": "^1.0.0", 1081 | "inflight": "^1.0.4", 1082 | "inherits": "2", 1083 | "minimatch": "^3.0.4", 1084 | "once": "^1.3.0", 1085 | "path-is-absolute": "^1.0.0" 1086 | } 1087 | }, 1088 | "has-flag": { 1089 | "version": "4.0.0", 1090 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1091 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 1092 | "dev": true 1093 | }, 1094 | "js-yaml": { 1095 | "version": "4.1.0", 1096 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", 1097 | "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", 1098 | "dev": true, 1099 | "requires": { 1100 | "argparse": "^2.0.1" 1101 | } 1102 | }, 1103 | "supports-color": { 1104 | "version": "8.1.1", 1105 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", 1106 | "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", 1107 | "dev": true, 1108 | "requires": { 1109 | "has-flag": "^4.0.0" 1110 | } 1111 | } 1112 | } 1113 | }, 1114 | "ms": { 1115 | "version": "2.1.3", 1116 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1117 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1118 | "dev": true 1119 | }, 1120 | "nanoid": { 1121 | "version": "3.1.25", 1122 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.1.25.tgz", 1123 | "integrity": "sha512-rdwtIXaXCLFAQbnfqDRnI6jaRHp9fTcYBjtFKE8eezcZ7LuLjhUaQGNeMXf1HmRoCH32CLz6XwX0TtxEOS/A3Q==", 1124 | "dev": true 1125 | }, 1126 | "node-fetch": { 1127 | "version": "3.0.0", 1128 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.0.0.tgz", 1129 | "integrity": "sha512-bKMI+C7/T/SPU1lKnbQbwxptpCrG9ashG+VkytmXCPZyuM9jB6VU+hY0oi4lC8LxTtAeWdckNCTa3nrGsAdA3Q==", 1130 | "dev": true, 1131 | "requires": { 1132 | "data-uri-to-buffer": "^3.0.1", 1133 | "fetch-blob": "^3.1.2" 1134 | } 1135 | }, 1136 | "normalize-path": { 1137 | "version": "3.0.0", 1138 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1139 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1140 | "dev": true 1141 | }, 1142 | "once": { 1143 | "version": "1.4.0", 1144 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1145 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1146 | "dev": true, 1147 | "requires": { 1148 | "wrappy": "1" 1149 | } 1150 | }, 1151 | "p-limit": { 1152 | "version": "3.1.0", 1153 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", 1154 | "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", 1155 | "dev": true, 1156 | "requires": { 1157 | "yocto-queue": "^0.1.0" 1158 | } 1159 | }, 1160 | "p-locate": { 1161 | "version": "5.0.0", 1162 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", 1163 | "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", 1164 | "dev": true, 1165 | "requires": { 1166 | "p-limit": "^3.0.2" 1167 | } 1168 | }, 1169 | "path-exists": { 1170 | "version": "4.0.0", 1171 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 1172 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 1173 | "dev": true 1174 | }, 1175 | "path-is-absolute": { 1176 | "version": "1.0.1", 1177 | "resolved": "http://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1178 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 1179 | "dev": true 1180 | }, 1181 | "path-parse": { 1182 | "version": "1.0.7", 1183 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 1184 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", 1185 | "dev": true 1186 | }, 1187 | "path-to-regexp": { 1188 | "version": "2.4.0", 1189 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-2.4.0.tgz", 1190 | "integrity": "sha512-G6zHoVqC6GGTQkZwF4lkuEyMbVOjoBKAEybQUypI1WTkqinCOrq2x6U2+phkJ1XsEMTy4LjtwPI7HW+NVrRR2w==", 1191 | "dev": true 1192 | }, 1193 | "pathval": { 1194 | "version": "1.1.1", 1195 | "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.1.tgz", 1196 | "integrity": "sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==", 1197 | "dev": true 1198 | }, 1199 | "picomatch": { 1200 | "version": "2.3.0", 1201 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", 1202 | "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", 1203 | "dev": true 1204 | }, 1205 | "punycode": { 1206 | "version": "2.1.1", 1207 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 1208 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 1209 | "dev": true 1210 | }, 1211 | "querystring": { 1212 | "version": "0.2.0", 1213 | "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", 1214 | "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=", 1215 | "dev": true 1216 | }, 1217 | "randombytes": { 1218 | "version": "2.1.0", 1219 | "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", 1220 | "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", 1221 | "dev": true, 1222 | "requires": { 1223 | "safe-buffer": "^5.1.0" 1224 | } 1225 | }, 1226 | "readdirp": { 1227 | "version": "3.6.0", 1228 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 1229 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 1230 | "dev": true, 1231 | "requires": { 1232 | "picomatch": "^2.2.1" 1233 | } 1234 | }, 1235 | "regenerator-runtime": { 1236 | "version": "0.13.7", 1237 | "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.7.tgz", 1238 | "integrity": "sha512-a54FxoJDIr27pgf7IgeQGxmqUNYrcV338lf/6gH456HZ/PhX+5BcwHXG9ajESmwe6WRO0tAzRUrRmNONWgkrew==", 1239 | "dev": true 1240 | }, 1241 | "require-directory": { 1242 | "version": "2.1.1", 1243 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 1244 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", 1245 | "dev": true 1246 | }, 1247 | "resolve": { 1248 | "version": "1.13.1", 1249 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.13.1.tgz", 1250 | "integrity": "sha512-CxqObCX8K8YtAhOBRg+lrcdn+LK+WYOS8tSjqSFbjtrI5PnS63QPhZl4+yKfrU9tdsbMu9Anr/amegT87M9Z6w==", 1251 | "dev": true, 1252 | "requires": { 1253 | "path-parse": "^1.0.6" 1254 | } 1255 | }, 1256 | "safe-buffer": { 1257 | "version": "5.2.1", 1258 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 1259 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 1260 | "dev": true 1261 | }, 1262 | "semver": { 1263 | "version": "5.7.0", 1264 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.0.tgz", 1265 | "integrity": "sha512-Ya52jSX2u7QKghxeoFGpLwCtGlt7j0oY9DYb5apt9nPlJ42ID+ulTXESnt/qAQcoSERyZ5sl3LDIOw0nAn/5DA==", 1266 | "dev": true 1267 | }, 1268 | "serialize-javascript": { 1269 | "version": "6.0.0", 1270 | "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.0.tgz", 1271 | "integrity": "sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==", 1272 | "dev": true, 1273 | "requires": { 1274 | "randombytes": "^2.1.0" 1275 | } 1276 | }, 1277 | "source-map": { 1278 | "version": "0.6.1", 1279 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 1280 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 1281 | "dev": true 1282 | }, 1283 | "source-map-support": { 1284 | "version": "0.5.19", 1285 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz", 1286 | "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==", 1287 | "dev": true, 1288 | "requires": { 1289 | "buffer-from": "^1.0.0", 1290 | "source-map": "^0.6.0" 1291 | } 1292 | }, 1293 | "sprintf-js": { 1294 | "version": "1.0.3", 1295 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 1296 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 1297 | "dev": true 1298 | }, 1299 | "string-width": { 1300 | "version": "4.2.3", 1301 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 1302 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 1303 | "dev": true, 1304 | "requires": { 1305 | "emoji-regex": "^8.0.0", 1306 | "is-fullwidth-code-point": "^3.0.0", 1307 | "strip-ansi": "^6.0.1" 1308 | } 1309 | }, 1310 | "strip-ansi": { 1311 | "version": "6.0.1", 1312 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 1313 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 1314 | "dev": true, 1315 | "requires": { 1316 | "ansi-regex": "^5.0.1" 1317 | } 1318 | }, 1319 | "strip-json-comments": { 1320 | "version": "3.1.1", 1321 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", 1322 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", 1323 | "dev": true 1324 | }, 1325 | "supports-color": { 1326 | "version": "5.5.0", 1327 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1328 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1329 | "dev": true, 1330 | "requires": { 1331 | "has-flag": "^3.0.0" 1332 | } 1333 | }, 1334 | "to-fast-properties": { 1335 | "version": "2.0.0", 1336 | "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", 1337 | "integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=", 1338 | "dev": true 1339 | }, 1340 | "to-regex-range": { 1341 | "version": "5.0.1", 1342 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1343 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1344 | "dev": true, 1345 | "requires": { 1346 | "is-number": "^7.0.0" 1347 | } 1348 | }, 1349 | "tr46": { 1350 | "version": "1.0.1", 1351 | "resolved": "https://registry.npmjs.org/tr46/-/tr46-1.0.1.tgz", 1352 | "integrity": "sha1-qLE/1r/SSJUZZ0zN5VujaTtwbQk=", 1353 | "dev": true, 1354 | "requires": { 1355 | "punycode": "^2.1.0" 1356 | } 1357 | }, 1358 | "ts-node": { 1359 | "version": "9.1.1", 1360 | "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-9.1.1.tgz", 1361 | "integrity": "sha512-hPlt7ZACERQGf03M253ytLY3dHbGNGrAq9qIHWUY9XHYl1z7wYngSr3OQ5xmui8o2AaxsONxIzjafLUiWBo1Fg==", 1362 | "dev": true, 1363 | "requires": { 1364 | "arg": "^4.1.0", 1365 | "create-require": "^1.1.0", 1366 | "diff": "^4.0.1", 1367 | "make-error": "^1.1.1", 1368 | "source-map-support": "^0.5.17", 1369 | "yn": "3.1.1" 1370 | } 1371 | }, 1372 | "tslib": { 1373 | "version": "1.10.0", 1374 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.10.0.tgz", 1375 | "integrity": "sha512-qOebF53frne81cf0S9B41ByenJ3/IuH8yJKngAX35CmiZySA0khhkovshKK+jGCaMnVomla7gVlIcc3EvKPbTQ==", 1376 | "dev": true 1377 | }, 1378 | "tslint": { 1379 | "version": "5.20.1", 1380 | "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.20.1.tgz", 1381 | "integrity": "sha512-EcMxhzCFt8k+/UP5r8waCf/lzmeSyVlqxqMEDQE7rWYiQky8KpIBz1JAoYXfROHrPZ1XXd43q8yQnULOLiBRQg==", 1382 | "dev": true, 1383 | "requires": { 1384 | "@babel/code-frame": "^7.0.0", 1385 | "builtin-modules": "^1.1.1", 1386 | "chalk": "^2.3.0", 1387 | "commander": "^2.12.1", 1388 | "diff": "^4.0.1", 1389 | "glob": "^7.1.1", 1390 | "js-yaml": "^3.13.1", 1391 | "minimatch": "^3.0.4", 1392 | "mkdirp": "^0.5.1", 1393 | "resolve": "^1.3.2", 1394 | "semver": "^5.3.0", 1395 | "tslib": "^1.8.0", 1396 | "tsutils": "^2.29.0" 1397 | }, 1398 | "dependencies": { 1399 | "diff": { 1400 | "version": "4.0.1", 1401 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.1.tgz", 1402 | "integrity": "sha512-s2+XdvhPCOF01LRQBC8hf4vhbVmI2CGS5aZnxLJlT5FtdhPCDFq80q++zK2KlrVorVDdL5BOGZ/VfLrVtYNF+Q==", 1403 | "dev": true 1404 | } 1405 | } 1406 | }, 1407 | "tslint-sonarts": { 1408 | "version": "1.9.0", 1409 | "resolved": "https://registry.npmjs.org/tslint-sonarts/-/tslint-sonarts-1.9.0.tgz", 1410 | "integrity": "sha512-CJWt+IiYI8qggb2O/JPkS6CkC5DY1IcqRsm9EHJ+AxoWK70lvtP7jguochyNDMP2vIz/giGdWCfEM39x/I/Vnw==", 1411 | "dev": true, 1412 | "requires": { 1413 | "immutable": "^3.8.2" 1414 | } 1415 | }, 1416 | "tsutils": { 1417 | "version": "2.29.0", 1418 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", 1419 | "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", 1420 | "dev": true, 1421 | "requires": { 1422 | "tslib": "^1.8.1" 1423 | } 1424 | }, 1425 | "type-detect": { 1426 | "version": "4.0.8", 1427 | "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", 1428 | "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", 1429 | "dev": true 1430 | }, 1431 | "typescript": { 1432 | "version": "3.9.10", 1433 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", 1434 | "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", 1435 | "dev": true 1436 | }, 1437 | "web-streams-polyfill": { 1438 | "version": "3.1.0", 1439 | "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.1.0.tgz", 1440 | "integrity": "sha512-wO9r1YnYe7kFBLHyyVEhV1H8VRWoNiNnuP+v/HUUmSTaRF8F93Kmd3JMrETx0f11GXxRek6OcL2QtjFIdc5WYw==", 1441 | "dev": true 1442 | }, 1443 | "webidl-conversions": { 1444 | "version": "4.0.2", 1445 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-4.0.2.tgz", 1446 | "integrity": "sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==", 1447 | "dev": true 1448 | }, 1449 | "whatwg-url": { 1450 | "version": "6.5.0", 1451 | "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-6.5.0.tgz", 1452 | "integrity": "sha512-rhRZRqx/TLJQWUpQ6bmrt2UV4f0HCQ463yQuONJqC6fO2VoEb1pTYddbe59SkYq87aoM5A3bdhMZiUiVws+fzQ==", 1453 | "dev": true, 1454 | "requires": { 1455 | "lodash.sortby": "^4.7.0", 1456 | "tr46": "^1.0.1", 1457 | "webidl-conversions": "^4.0.2" 1458 | } 1459 | }, 1460 | "which": { 1461 | "version": "2.0.2", 1462 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 1463 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 1464 | "dev": true, 1465 | "requires": { 1466 | "isexe": "^2.0.0" 1467 | } 1468 | }, 1469 | "workerpool": { 1470 | "version": "6.1.5", 1471 | "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.1.5.tgz", 1472 | "integrity": "sha512-XdKkCK0Zqc6w3iTxLckiuJ81tiD/o5rBE/m+nXpRCB+/Sq4DqkfXZ/x0jW02DG1tGsfUGXbTJyZDP+eu67haSw==", 1473 | "dev": true 1474 | }, 1475 | "wrap-ansi": { 1476 | "version": "7.0.0", 1477 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", 1478 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", 1479 | "dev": true, 1480 | "requires": { 1481 | "ansi-styles": "^4.0.0", 1482 | "string-width": "^4.1.0", 1483 | "strip-ansi": "^6.0.0" 1484 | }, 1485 | "dependencies": { 1486 | "ansi-styles": { 1487 | "version": "4.3.0", 1488 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 1489 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 1490 | "dev": true, 1491 | "requires": { 1492 | "color-convert": "^2.0.1" 1493 | } 1494 | }, 1495 | "color-convert": { 1496 | "version": "2.0.1", 1497 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 1498 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 1499 | "dev": true, 1500 | "requires": { 1501 | "color-name": "~1.1.4" 1502 | } 1503 | }, 1504 | "color-name": { 1505 | "version": "1.1.4", 1506 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 1507 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 1508 | "dev": true 1509 | } 1510 | } 1511 | }, 1512 | "wrappy": { 1513 | "version": "1.0.2", 1514 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1515 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1516 | "dev": true 1517 | }, 1518 | "y18n": { 1519 | "version": "5.0.8", 1520 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", 1521 | "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", 1522 | "dev": true 1523 | }, 1524 | "yargs": { 1525 | "version": "16.2.0", 1526 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", 1527 | "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", 1528 | "dev": true, 1529 | "requires": { 1530 | "cliui": "^7.0.2", 1531 | "escalade": "^3.1.1", 1532 | "get-caller-file": "^2.0.5", 1533 | "require-directory": "^2.1.1", 1534 | "string-width": "^4.2.0", 1535 | "y18n": "^5.0.5", 1536 | "yargs-parser": "^20.2.2" 1537 | } 1538 | }, 1539 | "yargs-parser": { 1540 | "version": "20.2.4", 1541 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz", 1542 | "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==", 1543 | "dev": true 1544 | }, 1545 | "yargs-unparser": { 1546 | "version": "2.0.0", 1547 | "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-2.0.0.tgz", 1548 | "integrity": "sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==", 1549 | "dev": true, 1550 | "requires": { 1551 | "camelcase": "^6.0.0", 1552 | "decamelize": "^4.0.0", 1553 | "flat": "^5.0.2", 1554 | "is-plain-obj": "^2.1.0" 1555 | } 1556 | }, 1557 | "yn": { 1558 | "version": "3.1.1", 1559 | "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", 1560 | "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", 1561 | "dev": true 1562 | }, 1563 | "yocto-queue": { 1564 | "version": "0.1.0", 1565 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", 1566 | "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", 1567 | "dev": true 1568 | } 1569 | } 1570 | } 1571 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "circuit-breaker-monad", 3 | "version": "1.0.1", 4 | "description": "Circuit Breaker pattern as a monad", 5 | "files": [ 6 | "lib" 7 | ], 8 | "main": "lib/index.js", 9 | "typings": "lib/index.d.ts", 10 | "scripts": { 11 | "test": "mocha --timeout 5000 -r ts-node/register src/**/*.test.ts", 12 | "build": "tsc" 13 | }, 14 | "repository": { 15 | "type": "git", 16 | "url": "git+https://github.com/YBogomolov/circuit-breaker-monad.git" 17 | }, 18 | "keywords": [ 19 | "circuit", 20 | "breaker", 21 | "typescript", 22 | "monad" 23 | ], 24 | "author": "Yuriy Bogomolov ", 25 | "license": "MIT", 26 | "bugs": { 27 | "url": "https://github.com/YBogomolov/circuit-breaker-monad/issues" 28 | }, 29 | "homepage": "https://github.com/YBogomolov/circuit-breaker-monad#readme", 30 | "peerDependencies": { 31 | "fp-ts": "^1.18.2 || ^2" 32 | }, 33 | "devDependencies": { 34 | "@types/chai": "^4.1.7", 35 | "@types/fetch-mock": "^7.3.0", 36 | "@types/mocha": "^9.0.0", 37 | "chai": "^4.2.0", 38 | "fetch-mock": "^9.0.0", 39 | "fp-ts": "^2.0.5", 40 | "mocha": "^9.0.3", 41 | "node-fetch": "^3.0.0", 42 | "ts-node": "^9.0.0", 43 | "tslint": "^5.11.0", 44 | "tslint-sonarts": "^1.8.0", 45 | "typescript": "^3.5.1" 46 | } 47 | } -------------------------------------------------------------------------------- /src/breaker.test.ts: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import fetchMock from 'fetch-mock'; 3 | import { either, fold } from 'fp-ts/lib/Either'; 4 | import { constant } from 'fp-ts/lib/function'; 5 | import { IORef } from 'fp-ts/lib/IORef'; 6 | import { delay, task } from 'fp-ts/lib/Task'; 7 | import 'mocha'; 8 | 9 | import { breakerClosed } from './helpers'; 10 | import { circuitBreaker, defaultBreakerOptions, singletonBreaker } from './index'; 11 | import { BreakerClosed } from './types'; 12 | 13 | const ADDRESS = 'http://example.com'; 14 | const yep = constant('yep'); 15 | const nope = constant('nope'); 16 | 17 | describe('Circuit Breaker', () => { 18 | after(() => { 19 | fetchMock.reset(); 20 | }); 21 | 22 | it('should handle promise resolution', (done) => { 23 | const fetcher = circuitBreaker()(defaultBreakerOptions); 24 | const [result] = fetcher({ 25 | request: () => Promise.resolve('ok'), 26 | breakerState: new IORef(new BreakerClosed(0)), 27 | }); 28 | result().then( 29 | (res) => either.bimap( 30 | res, 31 | (err) => done(err), 32 | (_) => done(), 33 | ), 34 | ); 35 | }); 36 | 37 | it('should handle promise rejection', (done) => { 38 | const fetcher = circuitBreaker()(defaultBreakerOptions); 39 | const [result] = fetcher({ 40 | request: () => Promise.reject('not ok'), 41 | breakerState: new IORef(new BreakerClosed(0)), 42 | }); 43 | result().then( 44 | (res) => either.bimap( 45 | res, 46 | (_) => done(), 47 | (s) => done(`should not resolve: ${s}`), 48 | ), 49 | ); 50 | }); 51 | 52 | it('should handle promise rejection with Error', (done) => { 53 | const error = new Error('not ok'); 54 | const fetcher = circuitBreaker()(defaultBreakerOptions); 55 | const [result] = fetcher({ 56 | request: () => Promise.reject(error), 57 | breakerState: new IORef(new BreakerClosed(0)), 58 | }); 59 | result().then( 60 | (res) => either.bimap( 61 | res, 62 | (l: Error) => { 63 | expect(l).to.equal(error); 64 | done(); 65 | }, 66 | (s) => done(`should not resolve: ${s}`), 67 | ), 68 | ); 69 | }); 70 | 71 | it('should resolve after a series of failing calls', async () => { 72 | const fetch = fetchMock 73 | .sandbox() 74 | .mock(ADDRESS, Promise.reject('oops'), { repeat: 2 }) 75 | .mock(ADDRESS, Promise.resolve('ok'), { overwriteRoutes: false }); 76 | const options = { maxBreakerFailures: 1, resetTimeoutSecs: 1, breakerDescription: '' }; 77 | const fetcher = circuitBreaker()(options); 78 | 79 | const [result1, { breakerState }] = fetcher({ 80 | request: () => fetch(ADDRESS), 81 | breakerState: new IORef(breakerClosed(0)), 82 | }); 83 | await result1(); 84 | expect(breakerState.read().tag).to.equal('Closed'); 85 | const [result2] = fetcher({ request: () => fetch(ADDRESS), breakerState }); 86 | await result2(); 87 | expect(breakerState.read().tag).to.equal('Open'); 88 | await delay(1000)(task.of(''))(); 89 | const [result] = fetcher({ request: () => fetch(ADDRESS), breakerState }); 90 | const r3 = await result(); 91 | 92 | expect(fold(nope, yep)(r3)).to.equal('yep'); 93 | expect(breakerState.read().tag).to.equal('Closed'); 94 | }); 95 | 96 | it('should mutate internal singleton breaker state', async () => { 97 | const fetch = fetchMock 98 | .sandbox() 99 | .mock(ADDRESS, Promise.reject('oops'), { repeat: 2 }) 100 | .mock(ADDRESS, Promise.resolve('ok'), { overwriteRoutes: false }); 101 | const options = { maxBreakerFailures: 1, resetTimeoutSecs: 2, breakerDescription: 'open' }; 102 | const promise = () => fetch(ADDRESS); 103 | 104 | // This is a fetcher with MUTABLE internal state using a clojure: 105 | const fetcher = singletonBreaker(options); 106 | 107 | const resultTE1 = fetcher(promise); 108 | const resultE1 = await resultTE1(); 109 | expect(fold(nope, yep)(resultE1)).to.equal('nope'); 110 | 111 | const resultTE2 = fetcher(promise); 112 | const resultE2 = await resultTE2(); 113 | expect(fold(nope, yep)(resultE2)).to.equal('nope'); 114 | 115 | await delay(1000)(task.of(''))(); 116 | const resultTE3 = fetcher(promise); 117 | const resultE3 = await resultTE3(); 118 | expect(fold((e) => e.message, yep)(resultE3)).to.equal('open'); 119 | 120 | await delay(1000)(task.of(''))(); 121 | const resultTE4 = fetcher(promise); 122 | const resultE4 = await resultTE4(); 123 | expect(fold((e) => e.message, yep)(resultE4)).to.equal('yep'); 124 | }); 125 | }); 126 | -------------------------------------------------------------------------------- /src/helpers.ts: -------------------------------------------------------------------------------- 1 | import { IORef } from 'fp-ts/lib/IORef'; 2 | 3 | import { BreakerClosed, BreakerOpen } from './types'; 4 | 5 | /** 6 | * Creates a new instance of BreakerClosed class 7 | * @param errorCount Number of errors encountered up to this time 8 | */ 9 | export const breakerClosed = (errorCount: number) => new BreakerClosed(errorCount); 10 | 11 | /** 12 | * Creates a new instance of BreakerOpen class 13 | * @param openEndTime Time when breaker opening ends 14 | */ 15 | export const breakerOpen = (openEndTime: number) => new BreakerOpen(openEndTime); 16 | 17 | /** 18 | * Gets current time as UTC timestamp 19 | */ 20 | export const getCurrentTime = () => new IORef(Date.now()); 21 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import { left } from 'fp-ts/lib/Either'; 2 | import { constFalse, constTrue, Lazy } from 'fp-ts/lib/function'; 3 | import * as IO from 'fp-ts/lib/IO'; 4 | import { IORef, newIORef } from 'fp-ts/lib/IORef'; 5 | import { pipe } from 'fp-ts/lib/pipeable'; 6 | import { State } from 'fp-ts/lib/State'; 7 | import * as TE from 'fp-ts/lib/TaskEither'; 8 | 9 | import { breakerClosed, breakerOpen, getCurrentTime } from './helpers'; 10 | import { BreakerEnvironment, BreakerOptions, BreakerState, CircuitBreaker } from './types'; 11 | 12 | /** 13 | * Default circuit breaker options 14 | */ 15 | export const defaultBreakerOptions: BreakerOptions = { 16 | maxBreakerFailures: 3, 17 | resetTimeoutSecs: 60, 18 | breakerDescription: 'Circuit breaker is open', 19 | }; 20 | 21 | export const circuitBreaker = (): CircuitBreaker => (opts: BreakerOptions) => { 22 | const failingCall = (): TE.TaskEither => TE.fromEither(left(new Error(opts.breakerDescription))); 23 | 24 | const incErrors = (ref: IORef) => 25 | pipe( 26 | getCurrentTime().read, 27 | IO.chain((currentTime) => pipe( 28 | ref.read, 29 | IO.chain((state) => { 30 | switch (state.tag) { 31 | case 'Closed': { 32 | const errorCount = state.errorCount; 33 | if (errorCount >= opts.maxBreakerFailures) { 34 | return ref.write(breakerOpen(currentTime + (opts.resetTimeoutSecs * 1000))); 35 | } else { 36 | return ref.write(breakerClosed(errorCount + 1)); 37 | } 38 | } 39 | case 'Open': { 40 | return IO.of(undefined); 41 | } 42 | } 43 | }), 44 | )), 45 | ); 46 | 47 | const callIfClosed = (request: Lazy>, ref: IORef): TE.TaskEither => 48 | TE.tryCatch( 49 | request, 50 | (reason) => pipe( 51 | incErrors(ref), 52 | IO.map(() => (reason instanceof Error) ? reason : new Error(String(reason))), 53 | )(), 54 | ); 55 | 56 | const canaryCall = (request: Lazy>, ref: IORef): TE.TaskEither => 57 | pipe( 58 | callIfClosed(request, ref), 59 | TE.chain((result: T) => pipe( 60 | ref.write(breakerClosed(0)), 61 | IO.chain(() => IO.of(result)), 62 | (a) => TE.taskEither.fromIO(a), 63 | )), 64 | ); 65 | 66 | const callIfOpen = (request: Lazy>, ref: IORef): TE.TaskEither => 67 | pipe( 68 | getCurrentTime().read, 69 | IO.chain((currentTime) => pipe( 70 | ref.read, 71 | IO.chain((state) => { 72 | switch (state.tag) { 73 | case 'Closed': 74 | return pipe(ref.write(state), IO.map(constFalse)); 75 | case 'Open': { 76 | if (currentTime > state.openEndTime) { 77 | return pipe(ref.write(breakerOpen(currentTime + (opts.resetTimeoutSecs * 1000))), IO.map(constTrue)); 78 | } 79 | return pipe(ref.write(state), IO.map(constFalse)); 80 | } 81 | } 82 | })), 83 | ), 84 | (a) => TE.taskEither.fromIO(a), 85 | TE.chain((canaryRequest) => canaryRequest ? canaryCall(request, ref) : failingCall()), 86 | ); 87 | 88 | const breakerService: State, TE.TaskEither> = 89 | ({ breakerState, request }) => [ 90 | pipe( 91 | breakerState.read, 92 | (a) => TE.taskEither.fromIO(a), 93 | TE.chain((state: BreakerState) => { 94 | switch (state.tag) { 95 | case 'Closed': 96 | return callIfClosed(request, breakerState); 97 | case 'Open': 98 | return callIfOpen(request, breakerState); 99 | } 100 | }), 101 | ), 102 | { breakerState, request }, 103 | ]; 104 | 105 | return breakerService; 106 | }; 107 | 108 | /** 109 | * Gets a singleton circuit breaker with clojure-bound state. 110 | * @param opts Breaker options 111 | */ 112 | export const singletonBreaker = (opts: BreakerOptions) => { 113 | const breakerState = newIORef(breakerClosed(0))(); 114 | return (request: Lazy>): TE.TaskEither => 115 | circuitBreaker()(opts)({ request, breakerState })[0]; 116 | }; 117 | -------------------------------------------------------------------------------- /src/types.ts: -------------------------------------------------------------------------------- 1 | // tslint:disable:max-classes-per-file 2 | 3 | import { Lazy } from 'fp-ts/lib/function'; 4 | import { IORef } from 'fp-ts/lib/IORef'; 5 | import { Reader } from 'fp-ts/lib/Reader'; 6 | import { State } from 'fp-ts/lib/State'; 7 | import { TaskEither } from 'fp-ts/lib/TaskEither'; 8 | 9 | /** 10 | * Options for determining behaviour of circuit breaking services. 11 | * 12 | * @export 13 | * @interface CircuitBreakerOptions 14 | */ 15 | export interface BreakerOptions { 16 | /** 17 | * How many times the underlying service must fail in the given window before the circuit opens. 18 | * 19 | * @type {number} 20 | * @memberof CircuitBreakerOptions 21 | */ 22 | maxBreakerFailures: number; 23 | 24 | /** 25 | * The window of time in which the underlying service must fail for the circuit to open. 26 | * 27 | * @type {number} 28 | * @memberof CircuitBreakerOptions 29 | */ 30 | resetTimeoutSecs: number; 31 | 32 | /** 33 | * Description that is attached to the failure so as to identify the particular circuit. 34 | * 35 | * @type {string} 36 | * @memberof CircuitBreakerOptions 37 | */ 38 | breakerDescription: string; 39 | } 40 | 41 | /** 42 | * Closed circuit breaker status (normal operation mode) 43 | * 44 | * @export 45 | * @class BreakerClosed 46 | */ 47 | export class BreakerClosed { 48 | public readonly tag = 'Closed'; 49 | constructor(public readonly errorCount: number) { } 50 | } 51 | 52 | /** 53 | * Open circuit break status (failure) 54 | * 55 | * @export 56 | * @class BreakerOpen 57 | */ 58 | export class BreakerOpen { 59 | public readonly tag = 'Open'; 60 | constructor(public readonly openEndTime: number) { } 61 | } 62 | 63 | /** 64 | * Sum type corresponding to possible circuit breaker statuses: open or closed 65 | * 66 | * @export 67 | * @type BreakerState 68 | */ 69 | export type BreakerState = BreakerClosed | BreakerOpen; 70 | 71 | /** 72 | * Breaker environment for State monad 73 | * 74 | * @export 75 | * @interface BreakerEnvironment 76 | * @template T Target data type 77 | */ 78 | export interface BreakerEnvironment { 79 | request: Lazy>; 80 | breakerState: IORef; 81 | } 82 | 83 | /** 84 | * Circuit breaker generator type. 85 | * It takes options required to instantiate a breaker and returns a State monad of result. 86 | */ 87 | export type CircuitBreaker = Reader, TaskEither>>; 88 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compileOnSave": true, 3 | "compilerOptions": { 4 | "outDir": "./lib", 5 | "rootDir": "./src", 6 | "module": "commonjs", 7 | "target": "es5", 8 | "moduleResolution": "node", 9 | "strict": true, 10 | "allowJs": false, 11 | "strictNullChecks": true, 12 | "experimentalDecorators": true, 13 | "allowSyntheticDefaultImports": true, 14 | "esModuleInterop": true, 15 | "noImplicitAny": true, 16 | "noUnusedLocals": true, 17 | "noUnusedParameters": true, 18 | "strictFunctionTypes": true, 19 | "forceConsistentCasingInFileNames": true, 20 | "suppressImplicitAnyIndexErrors": true, 21 | "declaration": true, 22 | "typeRoots": [ 23 | "./src/@types", 24 | ], 25 | "lib": [ 26 | "scripthost", 27 | "es2017", 28 | "dom" 29 | ] 30 | }, 31 | "include": [ 32 | "./src/**/*.ts" 33 | ], 34 | "exclude": [ 35 | "node_modules" 36 | ] 37 | } -------------------------------------------------------------------------------- /tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": [ 3 | "tslint:latest", 4 | "tslint-sonarts" 5 | ], 6 | "defaultSeverity": "error", 7 | "rules": { 8 | "arrow-return-shorthand": true, 9 | "callable-types": true, 10 | "class-name": true, 11 | "comment-format": [ 12 | true, 13 | "check-space" 14 | ], 15 | "curly": true, 16 | "eofline": true, 17 | "forin": true, 18 | "import-spacing": true, 19 | "indent": [ 20 | true, 21 | "spaces" 22 | ], 23 | "interface-over-type-literal": true, 24 | "label-position": true, 25 | "member-access": false, 26 | "member-ordering": [ 27 | true, 28 | { 29 | "order": [ 30 | "static-field", 31 | "instance-field", 32 | "static-method", 33 | "instance-method" 34 | ] 35 | } 36 | ], 37 | "no-arg": true, 38 | "no-bitwise": true, 39 | "no-console": [ 40 | true, 41 | "debug", 42 | "info", 43 | "time", 44 | "timeEnd", 45 | "trace" 46 | ], 47 | "no-construct": true, 48 | "no-debugger": true, 49 | "no-duplicate-super": true, 50 | "no-empty": false, 51 | "no-empty-interface": false, 52 | "no-eval": true, 53 | "no-inferrable-types": [ 54 | true, 55 | "ignore-params" 56 | ], 57 | "no-misused-new": true, 58 | "no-non-null-assertion": true, 59 | "no-shadowed-variable": true, 60 | "no-string-literal": false, 61 | "no-string-throw": true, 62 | "no-switch-case-fall-through": true, 63 | "no-trailing-whitespace": true, 64 | "no-unnecessary-initializer": true, 65 | "no-unused-expression": true, 66 | "no-var-keyword": true, 67 | "no-any": true, 68 | "ban-types": { 69 | "options": [ 70 | [ 71 | "Object", 72 | "Avoid using the `Object` type. Did you mean `object`?" 73 | ], 74 | [ 75 | "Function", 76 | "Avoid using the `Function` type. Prefer a specific function type, like `() => void`." 77 | ], 78 | [ 79 | "Boolean", 80 | "Avoid using the `Boolean` type. Did you mean `boolean`?" 81 | ], 82 | [ 83 | "Number", 84 | "Avoid using the `Number` type. Did you mean `number`?" 85 | ], 86 | [ 87 | "String", 88 | "Avoid using the `String` type. Did you mean `string`?" 89 | ], 90 | [ 91 | "Symbol", 92 | "Avoid using the `Symbol` type. Did you mean `symbol`?" 93 | ] 94 | ] 95 | }, 96 | "object-literal-sort-keys": false, 97 | "one-line": [ 98 | true, 99 | "check-open-brace", 100 | "check-catch", 101 | "check-else", 102 | "check-whitespace" 103 | ], 104 | "prefer-const": true, 105 | "quotemark": [ 106 | true, 107 | "single" 108 | ], 109 | "radix": true, 110 | "semicolon": [ 111 | true, 112 | "always" 113 | ], 114 | "triple-equals": [ 115 | true, 116 | "allow-null-check" 117 | ], 118 | "typedef-whitespace": [ 119 | true, 120 | { 121 | "call-signature": "nospace", 122 | "index-signature": "nospace", 123 | "parameter": "nospace", 124 | "property-declaration": "nospace", 125 | "variable-declaration": "nospace" 126 | } 127 | ], 128 | "unified-signatures": true, 129 | "variable-name": false, 130 | "whitespace": [ 131 | true, 132 | "check-branch", 133 | "check-decl", 134 | "check-operator", 135 | "check-separator", 136 | "check-type" 137 | ], 138 | "no-var-requires": false, 139 | "no-submodule-imports": false, 140 | "jsx-no-multiline-js": false, 141 | "interface-name": false, 142 | "no-implicit-dependencies": [ 143 | true, 144 | "dev" 145 | ], 146 | "ordered-imports": [ 147 | true, 148 | { 149 | "import-sources-order": "case-insensitive", 150 | "grouped-imports": true, 151 | "groups": [ 152 | { 153 | "name": "react", 154 | "match": "^react$", 155 | "order": 10 156 | }, 157 | { 158 | "name": "relative", 159 | "match": "^\\.\\./", 160 | "order": 30 161 | }, 162 | { 163 | "name": "local", 164 | "match": "(^\\./)|(^\\.$)", 165 | "order": 40 166 | }, 167 | { 168 | "name": "modules", 169 | "match": ".*", 170 | "order": 20 171 | } 172 | ] 173 | } 174 | ] 175 | } 176 | } --------------------------------------------------------------------------------