├── .gitignore ├── .vscode ├── launch.json ├── settings.json └── tasks.json ├── .vscodeignore ├── CHANGELOG.md ├── LICENSE ├── README.md ├── docs ├── 2-example-inlineLet.gif ├── 4-example-extractLet_expression.gif ├── 5-example-extractLet_lambda.gif └── 6-example-extractLet_string.gif ├── example.fs ├── package-lock.json ├── package.json ├── src ├── extension-commands.ts ├── extension.ts ├── general-funcs.ts ├── test-unit │ └── general-funcs.test.ts ├── test │ ├── 0-extension.test.ts │ ├── 1-extension-command-extractLet.test.ts │ ├── 2-extension-command-inlineLet.test.ts │ ├── index.ts │ └── utils.ts └── vscode-funcs.ts ├── todos.txt ├── tsconfig.json └── tslint.json /.gitignore: -------------------------------------------------------------------------------- 1 | out 2 | node_modules 3 | .vscode-test/ 4 | *.vsix 5 | .vs/ 6 | .ionide -------------------------------------------------------------------------------- /.vscode/launch.json: -------------------------------------------------------------------------------- 1 | // A launch configuration that compiles the extension and then opens it inside a new window 2 | { 3 | "version": "0.1.0", 4 | "configurations": [ 5 | { 6 | "name": "Extension", 7 | "type": "extensionHost", 8 | "request": "launch", 9 | "runtimeExecutable": "${execPath}", 10 | "args": ["--extensionDevelopmentPath=${workspaceRoot}" ], 11 | "stopOnEntry": false, 12 | "sourceMaps": true, 13 | "outFiles": [ "${workspaceRoot}/out/**/*.js" ], 14 | "preLaunchTask": "npm: watch" 15 | }, 16 | { 17 | "name": "Extension Tests", 18 | "type": "extensionHost", 19 | "request": "launch", 20 | "runtimeExecutable": "${execPath}", 21 | "args": ["--extensionDevelopmentPath=${workspaceRoot}", "--extensionTestsPath=${workspaceRoot}/out/test" ], 22 | "stopOnEntry": false, 23 | "sourceMaps": true, 24 | "outFiles": [ "${workspaceRoot}/out/test/**/*.js" ], 25 | "preLaunchTask": "npm: watch" 26 | } 27 | ] 28 | } 29 | -------------------------------------------------------------------------------- /.vscode/settings.json: -------------------------------------------------------------------------------- 1 | // Place your settings in this file to overwrite default and user settings. 2 | { 3 | "files.exclude": { 4 | "out": false // set this to true to hide the "out" folder with the compiled JS files 5 | }, 6 | "search.exclude": { 7 | "out": true // set this to false to include "out" folder in search results 8 | }, 9 | "typescript.tsdk": "node_modules\\typescript\\lib" 10 | } -------------------------------------------------------------------------------- /.vscode/tasks.json: -------------------------------------------------------------------------------- 1 | // See https://go.microsoft.com/fwlink/?LinkId=733558 2 | // for the documentation about the tasks.json format 3 | { 4 | "version": "2.0.0", 5 | "tasks": [ 6 | { 7 | "type": "npm", 8 | "script": "watch", 9 | "problemMatcher": "$tsc-watch", 10 | "isBackground": true, 11 | "presentation": { 12 | "reveal": "never" 13 | }, 14 | "group": { 15 | "kind": "build", 16 | "isDefault": true 17 | } 18 | } 19 | ] 20 | } -------------------------------------------------------------------------------- /.vscodeignore: -------------------------------------------------------------------------------- 1 | .vscode/** 2 | .vscode-test/** 3 | out/test/** 4 | out/**/*.map 5 | src/** 6 | .gitignore 7 | tsconfig.json 8 | vsc-extension-quickstart.md 9 | todos.txt 10 | *.fs -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Change Log 2 | 3 | ## 0.6.2 4 | - Fixed Issue #6 - extracting selection with a lambda now checks if the whole selection is a lambda 5 | 6 | ## 0.6.1 7 | - Fixed feature added by Issue #5 - can now inline binding with type annotations from usage 8 | 9 | ## 0.6.0 10 | - Issue Fixed #3 - Inline binding doesn't add brackets for single expressions 11 | 12 | ## 0.5.9 13 | - Issue Fixed #5 - Inline binding with type annotations 14 | 15 | ## 0.5.7 16 | - Issue Fixed #2 - mac keybinding for inlining 17 | - Issue Fixed #4 - empty line preventing inlining 18 | - packages security audit & fixes 19 | 20 | ## 0.5.5 21 | - Issue Fixed #1 - Extracting a snippet with a string closing string lteral symbol at the end 22 | 23 | ## 0.0.1 24 | - Initial release -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2018 Dan Mannock 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. -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Vscode-Fsharp-Refactor 2 | Additional F# refactoring tools for vscode. 3 | Less keystrokes, more F#, more fun(ctional programming). 4 | 5 | Available in the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=danmannock.vscode-fsharp-refactor) 6 | 7 | ## Features 8 | ### Extract 9 | Extract expression to let binding 10 | 1. Select the expression to extract 11 | 2. Use command (ctrl+shift+R) 12 | 13 | #### Extracts expressions 14 | 15 | ![4-example-extractLet_expression](https://raw.githubusercontent.com/dmannock/vscode-fsharp-refactor/master/docs/4-example-extractLet_expression.gif) 16 | 17 | #### Extracts lambdas 18 | ![5-example-extractLet_lambda](https://raw.githubusercontent.com/dmannock/vscode-fsharp-refactor/master/docs/5-example-extractLet_lambda.gif) 19 | 20 | #### Extracts strings 21 | ![6-example-extractLet_string](https://raw.githubusercontent.com/dmannock/vscode-fsharp-refactor/master/docs/6-example-extractLet_string.gif) 22 | 23 | ### Inline 24 | Inline binding 25 | 1. Move cursor to binding or a usage 26 | 2. Use command (ctrl+shift+I) 27 | 28 | ![2-example-inlineLet](https://raw.githubusercontent.com/dmannock/vscode-fsharp-refactor/master/docs/2-example-inlineLet.gif) 29 | 30 | Note: you can customise the hotkeys (file > preference > keyboard shortcuts) 31 | 32 | ## Requirements 33 | If you are here you probably have this covered. 34 | * Vscode 35 | * Ionide 36 | 37 | ## Roadmap 38 | who knows (see todos.txt for now) 39 | 40 | ## Building & Useful 41 | 42 | ### Full Build 43 | Runs the full build process including: 44 | * linting 45 | * unit tests 46 | * vscode tests 47 | * extension artifacts 48 | 49 | `` 50 | npm run build 51 | `` 52 | ### Test Watcher 53 | Runs unit tests when changes are detected. 54 | 55 | `` 56 | npm run unittest:watch 57 | `` 58 | -------------------------------------------------------------------------------- /docs/2-example-inlineLet.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dmannock/vscode-fsharp-refactor/c3075d780334262a376ed63bd7330bf52b56a224/docs/2-example-inlineLet.gif -------------------------------------------------------------------------------- /docs/4-example-extractLet_expression.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dmannock/vscode-fsharp-refactor/c3075d780334262a376ed63bd7330bf52b56a224/docs/4-example-extractLet_expression.gif -------------------------------------------------------------------------------- /docs/5-example-extractLet_lambda.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dmannock/vscode-fsharp-refactor/c3075d780334262a376ed63bd7330bf52b56a224/docs/5-example-extractLet_lambda.gif -------------------------------------------------------------------------------- /docs/6-example-extractLet_string.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dmannock/vscode-fsharp-refactor/c3075d780334262a376ed63bd7330bf52b56a224/docs/6-example-extractLet_string.gif -------------------------------------------------------------------------------- /example.fs: -------------------------------------------------------------------------------- 1 | 2 | //[*] example 1 //////////////////////////////////////// 3 | let test arg1 = 4 | let added = 1 + arg1 5 | let multi = added * 10 6 | multi / 2 7 | 8 | //[*] example 2 //////////////////////////////////////// 9 | let inlineTest arg1 = 10 | let inlineMe = 1 + arg1 11 | inlineMe * 2 / (3 - inlineMe) 12 | 13 | //[*] example 3 //////////////////////////////////////// 14 | let anotherInlineTest arg1 = 15 | let inlineMe = 1 + arg1 16 | let dontInline = 12345 17 | inlineMe * 2 / (3 - inlineMe) + dontInline 18 | 19 | //some use case examples from https://github.com/ionide/ionide-vscode-fsharp/issues/172 20 | 21 | //[*] example 4 //////////////////////////////////////// 22 | let extractLet chars = 23 | let noSpaces = chars |> Array.filter ((<>) ' ') 24 | noSpaces 25 | 26 | //[*] example 5 //////////////////////////////////////// 27 | let extractLambda o = 28 | let res = (o |> Array.fold (fun acc n -> (n |> Array.toList) @ acc ) []).Head 29 | res 30 | //should refactor to: 31 | // let collectSignatures acc n = 32 | // (n |> Array.toList) @ acc 33 | // let res = (o |> Array.fold collectSignatures []).Head 34 | 35 | //[x] example 6 //////////////////////////////////////// 36 | let stringToExtract = "/usr/bin/bash" 37 | 38 | //should refactor to: 39 | // let prefix = "/usr/bin/" 40 | // let stringToExtract = prefix + "bash" 41 | 42 | //[] example 7 //////////////////////////////////////// 43 | let bashPath = "/usr/bin/bash" 44 | let zshPath = "/usr/bin/zsh" 45 | 46 | //should refactor to: 47 | // let prefix = "/usr/bin/" 48 | // let bashPath = prefix + "bash" 49 | // let zshPath = prefix + "zsh" 50 | 51 | //[x] example 8 //////////////////////////////////////// 52 | let inlineTestSimilarName arg1 = 53 | let inlineMe = 1 + arg1 54 | let inlineMeWithSimilarName = inlineMe * 2 / (3 - inlineMe) 55 | inlineMeWithSimilarName 56 | 57 | //[x] example 9 //////////////////////////////////////// 58 | let inlineTestSimilarName2 arg1 = 59 | let inlineMe = 1 + arg1 60 | let inlineMeWithSimilarName = inlineMe * 2 / (3 - inlineMe) 61 | inlineMeWithSimilarName 62 | 63 | //[x] issue #1 //////////////////////////////////////// 64 | //Extracting a snippet with a string closing string lteral symbol at the end mangles the refactored let expression 65 | let commaJoin = Array.map string >> String.concat ", " 66 | 67 | //should refactor to: 68 | // let extracted = Array.map id >> String.concat ", " 69 | // let str = extracted 70 | 71 | //[x] issue #4 //////////////////////////////////////// 72 | // I cannot inline the expectedEvents let-binding, but if I remove the empty line between the definition and its usage, I can inline it. 73 | test "Y" { 74 | let expectedEvents = [] 75 | 76 | equal [] expectedEvents "" 77 | } 78 | 79 | //[x] example #10 //////////////////////////////////////// 80 | // Inlining a let binding on an outer scope with empty lines between 81 | let expectedEvents = [] 82 | 83 | let inlineScope initialEvents = 84 | initialEvents = expectedEvents 85 | 86 | 87 | //[x] issue #5 //////////////////////////////////////// 88 | // inline with type annotations 89 | let expectedEvents :int list = [] 90 | let checkCurrent evts = 91 | evts = expectedEvents 92 | 93 | //[x] issue #6 //////////////////////////////////////// 94 | // extracting let binding when selection contains a lambda and other expressions 95 | let getter() = [1;2] 96 | let extractLambda = 97 | let res = getter() |> List.map (fun a -> a * 2) |> List.sum 98 | res 99 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "vscode-fsharp-refactor", 3 | "version": "0.6.2", 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/highlight": { 17 | "version": "7.5.0", 18 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.5.0.tgz", 19 | "integrity": "sha512-7dV4eu9gBxoM0dAnj/BCFDW9LFU0zvTrkq0ugM7pnHEgguOEeOz1so2ZghEdzviYzQEED0r4EAgpsBChKy1TRQ==", 20 | "dev": true, 21 | "requires": { 22 | "chalk": "^2.0.0", 23 | "esutils": "^2.0.2", 24 | "js-tokens": "^4.0.0" 25 | } 26 | }, 27 | "@types/mocha": { 28 | "version": "5.2.7", 29 | "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-5.2.7.tgz", 30 | "integrity": "sha512-NYrtPht0wGzhwe9+/idPaBB+TqkY9AhTvOLMkThm0IoEfLaiVQZwBwyJ5puCkO3AUCWrmcoePjp2mbFocKy4SQ==", 31 | "dev": true 32 | }, 33 | "@types/node": { 34 | "version": "13.1.4", 35 | "resolved": "https://registry.npmjs.org/@types/node/-/node-13.1.4.tgz", 36 | "integrity": "sha512-Lue/mlp2egZJoHXZr4LndxDAd7i/7SQYhV0EjWfb/a4/OZ6tuVwMCVPiwkU5nsEipxEf7hmkSU7Em5VQ8P5NGA==", 37 | "dev": true 38 | }, 39 | "agent-base": { 40 | "version": "4.3.0", 41 | "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.3.0.tgz", 42 | "integrity": "sha512-salcGninV0nPrwpGNn4VTXBb1SOuXQBiqbrNXoeizJsHrsL6ERFM2Ne3JUSBWRE6aeNJI2ROP/WEEIDUiDe3cg==", 43 | "dev": true, 44 | "requires": { 45 | "es6-promisify": "^5.0.0" 46 | } 47 | }, 48 | "ajv": { 49 | "version": "6.10.2", 50 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.2.tgz", 51 | "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", 52 | "dev": true, 53 | "requires": { 54 | "fast-deep-equal": "^2.0.1", 55 | "fast-json-stable-stringify": "^2.0.0", 56 | "json-schema-traverse": "^0.4.1", 57 | "uri-js": "^4.2.2" 58 | } 59 | }, 60 | "ansi-colors": { 61 | "version": "3.2.3", 62 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.3.tgz", 63 | "integrity": "sha512-LEHHyuhlPY3TmuUYMh2oz89lTShfvgbmzaBcxve9t/9Wuy7Dwf4yoAKcND7KFT1HAQfqZ12qtc+DUrBMeKF9nw==", 64 | "dev": true 65 | }, 66 | "ansi-regex": { 67 | "version": "3.0.0", 68 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", 69 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", 70 | "dev": true 71 | }, 72 | "ansi-styles": { 73 | "version": "3.2.1", 74 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 75 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 76 | "dev": true, 77 | "requires": { 78 | "color-convert": "^1.9.0" 79 | } 80 | }, 81 | "anymatch": { 82 | "version": "3.1.1", 83 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", 84 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", 85 | "dev": true, 86 | "requires": { 87 | "normalize-path": "^3.0.0", 88 | "picomatch": "^2.0.4" 89 | } 90 | }, 91 | "arg": { 92 | "version": "4.1.2", 93 | "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.2.tgz", 94 | "integrity": "sha512-+ytCkGcBtHZ3V2r2Z06AncYO8jz46UEamcspGoU8lHcEbpn6J77QK0vdWvChsclg/tM5XIJC5tnjmPp7Eq6Obg==", 95 | "dev": true 96 | }, 97 | "argparse": { 98 | "version": "1.0.10", 99 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 100 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 101 | "dev": true, 102 | "requires": { 103 | "sprintf-js": "~1.0.2" 104 | } 105 | }, 106 | "asn1": { 107 | "version": "0.2.4", 108 | "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", 109 | "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", 110 | "dev": true, 111 | "requires": { 112 | "safer-buffer": "~2.1.0" 113 | } 114 | }, 115 | "assert-plus": { 116 | "version": "1.0.0", 117 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 118 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", 119 | "dev": true 120 | }, 121 | "asynckit": { 122 | "version": "0.4.0", 123 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 124 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", 125 | "dev": true 126 | }, 127 | "aws-sign2": { 128 | "version": "0.7.0", 129 | "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", 130 | "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=", 131 | "dev": true 132 | }, 133 | "aws4": { 134 | "version": "1.9.0", 135 | "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.9.0.tgz", 136 | "integrity": "sha512-Uvq6hVe90D0B2WEnUqtdgY1bATGz3mw33nH9Y+dmA+w5DHvUmBgkr5rM/KCHpCsiFNRUfokW/szpPPgMK2hm4A==", 137 | "dev": true 138 | }, 139 | "balanced-match": { 140 | "version": "1.0.0", 141 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 142 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 143 | "dev": true 144 | }, 145 | "bcrypt-pbkdf": { 146 | "version": "1.0.2", 147 | "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", 148 | "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", 149 | "dev": true, 150 | "requires": { 151 | "tweetnacl": "^0.14.3" 152 | } 153 | }, 154 | "binary-extensions": { 155 | "version": "2.0.0", 156 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz", 157 | "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==", 158 | "dev": true 159 | }, 160 | "brace-expansion": { 161 | "version": "1.1.11", 162 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 163 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 164 | "dev": true, 165 | "requires": { 166 | "balanced-match": "^1.0.0", 167 | "concat-map": "0.0.1" 168 | } 169 | }, 170 | "braces": { 171 | "version": "3.0.2", 172 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 173 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 174 | "dev": true, 175 | "requires": { 176 | "fill-range": "^7.0.1" 177 | } 178 | }, 179 | "browser-stdout": { 180 | "version": "1.3.1", 181 | "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", 182 | "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", 183 | "dev": true 184 | }, 185 | "buffer-from": { 186 | "version": "1.1.1", 187 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", 188 | "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", 189 | "dev": true 190 | }, 191 | "builtin-modules": { 192 | "version": "1.1.1", 193 | "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", 194 | "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", 195 | "dev": true 196 | }, 197 | "camelcase": { 198 | "version": "5.3.1", 199 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 200 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 201 | "dev": true 202 | }, 203 | "caseless": { 204 | "version": "0.12.0", 205 | "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", 206 | "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", 207 | "dev": true 208 | }, 209 | "chalk": { 210 | "version": "2.4.2", 211 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 212 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 213 | "dev": true, 214 | "requires": { 215 | "ansi-styles": "^3.2.1", 216 | "escape-string-regexp": "^1.0.5", 217 | "supports-color": "^5.3.0" 218 | } 219 | }, 220 | "chokidar": { 221 | "version": "3.3.0", 222 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.3.0.tgz", 223 | "integrity": "sha512-dGmKLDdT3Gdl7fBUe8XK+gAtGmzy5Fn0XkkWQuYxGIgWVPPse2CxFA5mtrlD0TOHaHjEUqkWNyP1XdHoJES/4A==", 224 | "dev": true, 225 | "requires": { 226 | "anymatch": "~3.1.1", 227 | "braces": "~3.0.2", 228 | "fsevents": "~2.1.1", 229 | "glob-parent": "~5.1.0", 230 | "is-binary-path": "~2.1.0", 231 | "is-glob": "~4.0.1", 232 | "normalize-path": "~3.0.0", 233 | "readdirp": "~3.2.0" 234 | } 235 | }, 236 | "cliui": { 237 | "version": "5.0.0", 238 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", 239 | "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", 240 | "dev": true, 241 | "requires": { 242 | "string-width": "^3.1.0", 243 | "strip-ansi": "^5.2.0", 244 | "wrap-ansi": "^5.1.0" 245 | }, 246 | "dependencies": { 247 | "ansi-regex": { 248 | "version": "4.1.0", 249 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 250 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 251 | "dev": true 252 | }, 253 | "string-width": { 254 | "version": "3.1.0", 255 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 256 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 257 | "dev": true, 258 | "requires": { 259 | "emoji-regex": "^7.0.1", 260 | "is-fullwidth-code-point": "^2.0.0", 261 | "strip-ansi": "^5.1.0" 262 | } 263 | }, 264 | "strip-ansi": { 265 | "version": "5.2.0", 266 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 267 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 268 | "dev": true, 269 | "requires": { 270 | "ansi-regex": "^4.1.0" 271 | } 272 | } 273 | } 274 | }, 275 | "color-convert": { 276 | "version": "1.9.1", 277 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.1.tgz", 278 | "integrity": "sha512-mjGanIiwQJskCC18rPR6OmrZ6fm2Lc7PeGFYwCmy5J34wC6F1PzdGL6xeMfmgicfYcNLGuVFA3WzXtIDCQSZxQ==", 279 | "dev": true, 280 | "requires": { 281 | "color-name": "^1.1.1" 282 | } 283 | }, 284 | "color-name": { 285 | "version": "1.1.3", 286 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 287 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 288 | "dev": true 289 | }, 290 | "combined-stream": { 291 | "version": "1.0.8", 292 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 293 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 294 | "dev": true, 295 | "requires": { 296 | "delayed-stream": "~1.0.0" 297 | } 298 | }, 299 | "commander": { 300 | "version": "2.15.1", 301 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.15.1.tgz", 302 | "integrity": "sha512-VlfT9F3V0v+jr4yxPc5gg9s62/fIVWsd2Bk2iD435um1NlGMYdVCq+MjcXnhYq2icNOizHr1kK+5TI6H0Hy0ag==", 303 | "dev": true 304 | }, 305 | "concat-map": { 306 | "version": "0.0.1", 307 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 308 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 309 | "dev": true 310 | }, 311 | "core-util-is": { 312 | "version": "1.0.2", 313 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 314 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", 315 | "dev": true 316 | }, 317 | "dashdash": { 318 | "version": "1.14.1", 319 | "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", 320 | "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", 321 | "dev": true, 322 | "requires": { 323 | "assert-plus": "^1.0.0" 324 | } 325 | }, 326 | "debug": { 327 | "version": "3.1.0", 328 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", 329 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", 330 | "dev": true, 331 | "requires": { 332 | "ms": "2.0.0" 333 | } 334 | }, 335 | "decamelize": { 336 | "version": "1.2.0", 337 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 338 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", 339 | "dev": true 340 | }, 341 | "define-properties": { 342 | "version": "1.1.3", 343 | "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", 344 | "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", 345 | "dev": true, 346 | "requires": { 347 | "object-keys": "^1.0.12" 348 | } 349 | }, 350 | "delayed-stream": { 351 | "version": "1.0.0", 352 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 353 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", 354 | "dev": true 355 | }, 356 | "diff": { 357 | "version": "3.5.0", 358 | "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", 359 | "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", 360 | "dev": true 361 | }, 362 | "ecc-jsbn": { 363 | "version": "0.1.2", 364 | "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", 365 | "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", 366 | "dev": true, 367 | "requires": { 368 | "jsbn": "~0.1.0", 369 | "safer-buffer": "^2.1.0" 370 | } 371 | }, 372 | "emoji-regex": { 373 | "version": "7.0.3", 374 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 375 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 376 | "dev": true 377 | }, 378 | "es-abstract": { 379 | "version": "1.17.0", 380 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.0.tgz", 381 | "integrity": "sha512-yYkE07YF+6SIBmg1MsJ9dlub5L48Ek7X0qz+c/CPCHS9EBXfESorzng4cJQjJW5/pB6vDF41u7F8vUhLVDqIug==", 382 | "dev": true, 383 | "requires": { 384 | "es-to-primitive": "^1.2.1", 385 | "function-bind": "^1.1.1", 386 | "has": "^1.0.3", 387 | "has-symbols": "^1.0.1", 388 | "is-callable": "^1.1.5", 389 | "is-regex": "^1.0.5", 390 | "object-inspect": "^1.7.0", 391 | "object-keys": "^1.1.1", 392 | "object.assign": "^4.1.0", 393 | "string.prototype.trimleft": "^2.1.1", 394 | "string.prototype.trimright": "^2.1.1" 395 | } 396 | }, 397 | "es-to-primitive": { 398 | "version": "1.2.1", 399 | "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", 400 | "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", 401 | "dev": true, 402 | "requires": { 403 | "is-callable": "^1.1.4", 404 | "is-date-object": "^1.0.1", 405 | "is-symbol": "^1.0.2" 406 | } 407 | }, 408 | "es6-promise": { 409 | "version": "4.2.8", 410 | "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", 411 | "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==", 412 | "dev": true 413 | }, 414 | "es6-promisify": { 415 | "version": "5.0.0", 416 | "resolved": "https://registry.npmjs.org/es6-promisify/-/es6-promisify-5.0.0.tgz", 417 | "integrity": "sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM=", 418 | "dev": true, 419 | "requires": { 420 | "es6-promise": "^4.0.3" 421 | } 422 | }, 423 | "escape-string-regexp": { 424 | "version": "1.0.5", 425 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 426 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 427 | "dev": true 428 | }, 429 | "esprima": { 430 | "version": "4.0.1", 431 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 432 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 433 | "dev": true 434 | }, 435 | "esutils": { 436 | "version": "2.0.3", 437 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 438 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 439 | "dev": true 440 | }, 441 | "extend": { 442 | "version": "3.0.2", 443 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", 444 | "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", 445 | "dev": true 446 | }, 447 | "extsprintf": { 448 | "version": "1.3.0", 449 | "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", 450 | "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", 451 | "dev": true 452 | }, 453 | "fast-deep-equal": { 454 | "version": "2.0.1", 455 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", 456 | "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=", 457 | "dev": true 458 | }, 459 | "fast-json-stable-stringify": { 460 | "version": "2.1.0", 461 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", 462 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", 463 | "dev": true 464 | }, 465 | "fill-range": { 466 | "version": "7.0.1", 467 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 468 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 469 | "dev": true, 470 | "requires": { 471 | "to-regex-range": "^5.0.1" 472 | } 473 | }, 474 | "find-up": { 475 | "version": "3.0.0", 476 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", 477 | "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", 478 | "dev": true, 479 | "requires": { 480 | "locate-path": "^3.0.0" 481 | } 482 | }, 483 | "flat": { 484 | "version": "4.1.0", 485 | "resolved": "https://registry.npmjs.org/flat/-/flat-4.1.0.tgz", 486 | "integrity": "sha512-Px/TiLIznH7gEDlPXcUD4KnBusa6kR6ayRUVcnEAbreRIuhkqow/mun59BuRXwoYk7ZQOLW1ZM05ilIvK38hFw==", 487 | "dev": true, 488 | "requires": { 489 | "is-buffer": "~2.0.3" 490 | } 491 | }, 492 | "forever-agent": { 493 | "version": "0.6.1", 494 | "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", 495 | "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", 496 | "dev": true 497 | }, 498 | "form-data": { 499 | "version": "2.3.3", 500 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", 501 | "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", 502 | "dev": true, 503 | "requires": { 504 | "asynckit": "^0.4.0", 505 | "combined-stream": "^1.0.6", 506 | "mime-types": "^2.1.12" 507 | } 508 | }, 509 | "fs.realpath": { 510 | "version": "1.0.0", 511 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 512 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 513 | "dev": true 514 | }, 515 | "fsevents": { 516 | "version": "2.1.2", 517 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.2.tgz", 518 | "integrity": "sha512-R4wDiBwZ0KzpgOWetKDug1FZcYhqYnUYKtfZYt4mD5SBz76q0KR4Q9o7GIPamsVPGmW3EYPPJ0dOOjvx32ldZA==", 519 | "dev": true, 520 | "optional": true 521 | }, 522 | "function-bind": { 523 | "version": "1.1.1", 524 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 525 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 526 | "dev": true 527 | }, 528 | "get-caller-file": { 529 | "version": "2.0.5", 530 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 531 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 532 | "dev": true 533 | }, 534 | "getpass": { 535 | "version": "0.1.7", 536 | "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", 537 | "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", 538 | "dev": true, 539 | "requires": { 540 | "assert-plus": "^1.0.0" 541 | } 542 | }, 543 | "glob": { 544 | "version": "7.1.2", 545 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", 546 | "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", 547 | "dev": true, 548 | "requires": { 549 | "fs.realpath": "^1.0.0", 550 | "inflight": "^1.0.4", 551 | "inherits": "2", 552 | "minimatch": "^3.0.4", 553 | "once": "^1.3.0", 554 | "path-is-absolute": "^1.0.0" 555 | } 556 | }, 557 | "glob-parent": { 558 | "version": "5.1.0", 559 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.0.tgz", 560 | "integrity": "sha512-qjtRgnIVmOfnKUE3NJAQEdk+lKrxfw8t5ke7SXtfMTHcjsBfOfWXCQfdb30zfDoZQ2IRSIiidmjtbHZPZ++Ihw==", 561 | "dev": true, 562 | "requires": { 563 | "is-glob": "^4.0.1" 564 | } 565 | }, 566 | "growl": { 567 | "version": "1.10.5", 568 | "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz", 569 | "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==", 570 | "dev": true 571 | }, 572 | "har-schema": { 573 | "version": "2.0.0", 574 | "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", 575 | "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", 576 | "dev": true 577 | }, 578 | "har-validator": { 579 | "version": "5.1.3", 580 | "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.3.tgz", 581 | "integrity": "sha512-sNvOCzEQNr/qrvJgc3UG/kD4QtlHycrzwS+6mfTrrSq97BvaYcPZZI1ZSqGSPR73Cxn4LKTD4PttRwfU7jWq5g==", 582 | "dev": true, 583 | "requires": { 584 | "ajv": "^6.5.5", 585 | "har-schema": "^2.0.0" 586 | } 587 | }, 588 | "has": { 589 | "version": "1.0.3", 590 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 591 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 592 | "dev": true, 593 | "requires": { 594 | "function-bind": "^1.1.1" 595 | } 596 | }, 597 | "has-flag": { 598 | "version": "3.0.0", 599 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 600 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 601 | "dev": true 602 | }, 603 | "has-symbols": { 604 | "version": "1.0.1", 605 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", 606 | "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", 607 | "dev": true 608 | }, 609 | "he": { 610 | "version": "1.2.0", 611 | "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", 612 | "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", 613 | "dev": true 614 | }, 615 | "http-proxy-agent": { 616 | "version": "2.1.0", 617 | "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-2.1.0.tgz", 618 | "integrity": "sha512-qwHbBLV7WviBl0rQsOzH6o5lwyOIvwp/BdFnvVxXORldu5TmjFfjzBcWUWS5kWAZhmv+JtiDhSuQCp4sBfbIgg==", 619 | "dev": true, 620 | "requires": { 621 | "agent-base": "4", 622 | "debug": "3.1.0" 623 | } 624 | }, 625 | "http-signature": { 626 | "version": "1.2.0", 627 | "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", 628 | "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", 629 | "dev": true, 630 | "requires": { 631 | "assert-plus": "^1.0.0", 632 | "jsprim": "^1.2.2", 633 | "sshpk": "^1.7.0" 634 | } 635 | }, 636 | "https-proxy-agent": { 637 | "version": "2.2.4", 638 | "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.4.tgz", 639 | "integrity": "sha512-OmvfoQ53WLjtA9HeYP9RNrWMJzzAz1JGaSFr1nijg0PVR1JaD/xbJq1mdEIIlxGpXp9eSe/O2LgU9DJmTPd0Eg==", 640 | "dev": true, 641 | "requires": { 642 | "agent-base": "^4.3.0", 643 | "debug": "^3.1.0" 644 | } 645 | }, 646 | "inflight": { 647 | "version": "1.0.6", 648 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 649 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 650 | "dev": true, 651 | "requires": { 652 | "once": "^1.3.0", 653 | "wrappy": "1" 654 | } 655 | }, 656 | "inherits": { 657 | "version": "2.0.3", 658 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 659 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", 660 | "dev": true 661 | }, 662 | "is-binary-path": { 663 | "version": "2.1.0", 664 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 665 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 666 | "dev": true, 667 | "requires": { 668 | "binary-extensions": "^2.0.0" 669 | } 670 | }, 671 | "is-buffer": { 672 | "version": "2.0.4", 673 | "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.4.tgz", 674 | "integrity": "sha512-Kq1rokWXOPXWuaMAqZiJW4XxsmD9zGx9q4aePabbn3qCRGedtH7Cm+zV8WETitMfu1wdh+Rvd6w5egwSngUX2A==", 675 | "dev": true 676 | }, 677 | "is-callable": { 678 | "version": "1.1.5", 679 | "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.5.tgz", 680 | "integrity": "sha512-ESKv5sMCJB2jnHTWZ3O5itG+O128Hsus4K4Qh1h2/cgn2vbgnLSVqfV46AeJA9D5EeeLa9w81KUXMtn34zhX+Q==", 681 | "dev": true 682 | }, 683 | "is-date-object": { 684 | "version": "1.0.2", 685 | "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", 686 | "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", 687 | "dev": true 688 | }, 689 | "is-extglob": { 690 | "version": "2.1.1", 691 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 692 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 693 | "dev": true 694 | }, 695 | "is-fullwidth-code-point": { 696 | "version": "2.0.0", 697 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 698 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 699 | "dev": true 700 | }, 701 | "is-glob": { 702 | "version": "4.0.1", 703 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 704 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 705 | "dev": true, 706 | "requires": { 707 | "is-extglob": "^2.1.1" 708 | } 709 | }, 710 | "is-number": { 711 | "version": "7.0.0", 712 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 713 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 714 | "dev": true 715 | }, 716 | "is-regex": { 717 | "version": "1.0.5", 718 | "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.5.tgz", 719 | "integrity": "sha512-vlKW17SNq44owv5AQR3Cq0bQPEb8+kF3UKZ2fiZNOWtztYE5i0CzCZxFDwO58qAOWtxdBRVO/V5Qin1wjCqFYQ==", 720 | "dev": true, 721 | "requires": { 722 | "has": "^1.0.3" 723 | } 724 | }, 725 | "is-symbol": { 726 | "version": "1.0.3", 727 | "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", 728 | "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", 729 | "dev": true, 730 | "requires": { 731 | "has-symbols": "^1.0.1" 732 | } 733 | }, 734 | "is-typedarray": { 735 | "version": "1.0.0", 736 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 737 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", 738 | "dev": true 739 | }, 740 | "isexe": { 741 | "version": "2.0.0", 742 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 743 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 744 | "dev": true 745 | }, 746 | "isstream": { 747 | "version": "0.1.2", 748 | "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", 749 | "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=", 750 | "dev": true 751 | }, 752 | "js-tokens": { 753 | "version": "4.0.0", 754 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 755 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 756 | "dev": true 757 | }, 758 | "js-yaml": { 759 | "version": "3.13.1", 760 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", 761 | "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", 762 | "dev": true, 763 | "requires": { 764 | "argparse": "^1.0.7", 765 | "esprima": "^4.0.0" 766 | } 767 | }, 768 | "jsbn": { 769 | "version": "0.1.1", 770 | "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", 771 | "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", 772 | "dev": true 773 | }, 774 | "json-schema": { 775 | "version": "0.2.3", 776 | "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", 777 | "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=", 778 | "dev": true 779 | }, 780 | "json-schema-traverse": { 781 | "version": "0.4.1", 782 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 783 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", 784 | "dev": true 785 | }, 786 | "json-stringify-safe": { 787 | "version": "5.0.1", 788 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", 789 | "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", 790 | "dev": true 791 | }, 792 | "jsprim": { 793 | "version": "1.4.1", 794 | "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", 795 | "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", 796 | "dev": true, 797 | "requires": { 798 | "assert-plus": "1.0.0", 799 | "extsprintf": "1.3.0", 800 | "json-schema": "0.2.3", 801 | "verror": "1.10.0" 802 | } 803 | }, 804 | "locate-path": { 805 | "version": "3.0.0", 806 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", 807 | "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", 808 | "dev": true, 809 | "requires": { 810 | "p-locate": "^3.0.0", 811 | "path-exists": "^3.0.0" 812 | } 813 | }, 814 | "lodash": { 815 | "version": "4.17.19", 816 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.19.tgz", 817 | "integrity": "sha512-JNvd8XER9GQX0v2qJgsaN/mzFCNA5BRe/j8JN9d+tWyGLSodKQHKFicdwNYzWwI3wjRnaKPsGj1XkBjx/F96DQ==", 818 | "dev": true 819 | }, 820 | "log-symbols": { 821 | "version": "2.2.0", 822 | "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", 823 | "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", 824 | "dev": true, 825 | "requires": { 826 | "chalk": "^2.0.1" 827 | } 828 | }, 829 | "make-error": { 830 | "version": "1.3.5", 831 | "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.5.tgz", 832 | "integrity": "sha512-c3sIjNUow0+8swNwVpqoH4YCShKNFkMaw6oH1mNS2haDZQqkeZFlHS3dhoeEbKKmJB4vXpJucU6oH75aDYeE9g==", 833 | "dev": true 834 | }, 835 | "mime-db": { 836 | "version": "1.43.0", 837 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.43.0.tgz", 838 | "integrity": "sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ==", 839 | "dev": true 840 | }, 841 | "mime-types": { 842 | "version": "2.1.26", 843 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.26.tgz", 844 | "integrity": "sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==", 845 | "dev": true, 846 | "requires": { 847 | "mime-db": "1.43.0" 848 | } 849 | }, 850 | "minimatch": { 851 | "version": "3.0.4", 852 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 853 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 854 | "dev": true, 855 | "requires": { 856 | "brace-expansion": "^1.1.7" 857 | } 858 | }, 859 | "minimist": { 860 | "version": "0.0.8", 861 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 862 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", 863 | "dev": true 864 | }, 865 | "mkdirp": { 866 | "version": "0.5.1", 867 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 868 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 869 | "dev": true, 870 | "requires": { 871 | "minimist": "0.0.8" 872 | } 873 | }, 874 | "mocha": { 875 | "version": "7.0.0", 876 | "resolved": "https://registry.npmjs.org/mocha/-/mocha-7.0.0.tgz", 877 | "integrity": "sha512-CirsOPbO3jU86YKjjMzFLcXIb5YiGLUrjrXFHoJ3e2z9vWiaZVCZQ2+gtRGMPWF+nFhN6AWwLM/juzAQ6KRkbA==", 878 | "dev": true, 879 | "requires": { 880 | "ansi-colors": "3.2.3", 881 | "browser-stdout": "1.3.1", 882 | "chokidar": "3.3.0", 883 | "debug": "3.2.6", 884 | "diff": "3.5.0", 885 | "escape-string-regexp": "1.0.5", 886 | "find-up": "3.0.0", 887 | "glob": "7.1.3", 888 | "growl": "1.10.5", 889 | "he": "1.2.0", 890 | "js-yaml": "3.13.1", 891 | "log-symbols": "2.2.0", 892 | "minimatch": "3.0.4", 893 | "mkdirp": "0.5.1", 894 | "ms": "2.1.1", 895 | "node-environment-flags": "1.0.6", 896 | "object.assign": "4.1.0", 897 | "strip-json-comments": "2.0.1", 898 | "supports-color": "6.0.0", 899 | "which": "1.3.1", 900 | "wide-align": "1.1.3", 901 | "yargs": "13.3.0", 902 | "yargs-parser": "13.1.1", 903 | "yargs-unparser": "1.6.0" 904 | }, 905 | "dependencies": { 906 | "debug": { 907 | "version": "3.2.6", 908 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 909 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 910 | "dev": true, 911 | "requires": { 912 | "ms": "^2.1.1" 913 | } 914 | }, 915 | "glob": { 916 | "version": "7.1.3", 917 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", 918 | "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", 919 | "dev": true, 920 | "requires": { 921 | "fs.realpath": "^1.0.0", 922 | "inflight": "^1.0.4", 923 | "inherits": "2", 924 | "minimatch": "^3.0.4", 925 | "once": "^1.3.0", 926 | "path-is-absolute": "^1.0.0" 927 | } 928 | }, 929 | "ms": { 930 | "version": "2.1.1", 931 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 932 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", 933 | "dev": true 934 | }, 935 | "supports-color": { 936 | "version": "6.0.0", 937 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.0.0.tgz", 938 | "integrity": "sha512-on9Kwidc1IUQo+bQdhi8+Tijpo0e1SS6RoGo2guUwn5vdaxw8RXOF9Vb2ws+ihWOmh4JnCJOvaziZWP1VABaLg==", 939 | "dev": true, 940 | "requires": { 941 | "has-flag": "^3.0.0" 942 | } 943 | } 944 | } 945 | }, 946 | "ms": { 947 | "version": "2.0.0", 948 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 949 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 950 | "dev": true 951 | }, 952 | "node-environment-flags": { 953 | "version": "1.0.6", 954 | "resolved": "https://registry.npmjs.org/node-environment-flags/-/node-environment-flags-1.0.6.tgz", 955 | "integrity": "sha512-5Evy2epuL+6TM0lCQGpFIj6KwiEsGh1SrHUhTbNX+sLbBtjidPZFAnVK9y5yU1+h//RitLbRHTIMyxQPtxMdHw==", 956 | "dev": true, 957 | "requires": { 958 | "object.getownpropertydescriptors": "^2.0.3", 959 | "semver": "^5.7.0" 960 | } 961 | }, 962 | "normalize-path": { 963 | "version": "3.0.0", 964 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 965 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 966 | "dev": true 967 | }, 968 | "oauth-sign": { 969 | "version": "0.9.0", 970 | "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", 971 | "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", 972 | "dev": true 973 | }, 974 | "object-inspect": { 975 | "version": "1.7.0", 976 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz", 977 | "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==", 978 | "dev": true 979 | }, 980 | "object-keys": { 981 | "version": "1.1.1", 982 | "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", 983 | "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", 984 | "dev": true 985 | }, 986 | "object.assign": { 987 | "version": "4.1.0", 988 | "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz", 989 | "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==", 990 | "dev": true, 991 | "requires": { 992 | "define-properties": "^1.1.2", 993 | "function-bind": "^1.1.1", 994 | "has-symbols": "^1.0.0", 995 | "object-keys": "^1.0.11" 996 | } 997 | }, 998 | "object.getownpropertydescriptors": { 999 | "version": "2.1.0", 1000 | "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz", 1001 | "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==", 1002 | "dev": true, 1003 | "requires": { 1004 | "define-properties": "^1.1.3", 1005 | "es-abstract": "^1.17.0-next.1" 1006 | } 1007 | }, 1008 | "once": { 1009 | "version": "1.4.0", 1010 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1011 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1012 | "dev": true, 1013 | "requires": { 1014 | "wrappy": "1" 1015 | } 1016 | }, 1017 | "p-limit": { 1018 | "version": "2.2.2", 1019 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.2.2.tgz", 1020 | "integrity": "sha512-WGR+xHecKTr7EbUEhyLSh5Dube9JtdiG78ufaeLxTgpudf/20KqyMioIUZJAezlTIi6evxuoUs9YXc11cU+yzQ==", 1021 | "dev": true, 1022 | "requires": { 1023 | "p-try": "^2.0.0" 1024 | } 1025 | }, 1026 | "p-locate": { 1027 | "version": "3.0.0", 1028 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", 1029 | "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", 1030 | "dev": true, 1031 | "requires": { 1032 | "p-limit": "^2.0.0" 1033 | } 1034 | }, 1035 | "p-try": { 1036 | "version": "2.2.0", 1037 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 1038 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 1039 | "dev": true 1040 | }, 1041 | "path-exists": { 1042 | "version": "3.0.0", 1043 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", 1044 | "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", 1045 | "dev": true 1046 | }, 1047 | "path-is-absolute": { 1048 | "version": "1.0.1", 1049 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1050 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 1051 | "dev": true 1052 | }, 1053 | "path-parse": { 1054 | "version": "1.0.6", 1055 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", 1056 | "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", 1057 | "dev": true 1058 | }, 1059 | "performance-now": { 1060 | "version": "2.1.0", 1061 | "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", 1062 | "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=", 1063 | "dev": true 1064 | }, 1065 | "picomatch": { 1066 | "version": "2.2.1", 1067 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.1.tgz", 1068 | "integrity": "sha512-ISBaA8xQNmwELC7eOjqFKMESB2VIqt4PPDD0nsS95b/9dZXvVKOlz9keMSnoGGKcOHXfTvDD6WMaRoSc9UuhRA==", 1069 | "dev": true 1070 | }, 1071 | "psl": { 1072 | "version": "1.7.0", 1073 | "resolved": "https://registry.npmjs.org/psl/-/psl-1.7.0.tgz", 1074 | "integrity": "sha512-5NsSEDv8zY70ScRnOTn7bK7eanl2MvFrOrS/R6x+dBt5g1ghnj9Zv90kO8GwT8gxcu2ANyFprnFYB85IogIJOQ==", 1075 | "dev": true 1076 | }, 1077 | "punycode": { 1078 | "version": "2.1.1", 1079 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 1080 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 1081 | "dev": true 1082 | }, 1083 | "qs": { 1084 | "version": "6.5.2", 1085 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", 1086 | "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==", 1087 | "dev": true 1088 | }, 1089 | "querystringify": { 1090 | "version": "2.1.1", 1091 | "resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.1.1.tgz", 1092 | "integrity": "sha512-w7fLxIRCRT7U8Qu53jQnJyPkYZIaR4n5151KMfcJlO/A9397Wxb1amJvROTK6TOnp7PfoAmg/qXiNHI+08jRfA==", 1093 | "dev": true 1094 | }, 1095 | "readdirp": { 1096 | "version": "3.2.0", 1097 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.2.0.tgz", 1098 | "integrity": "sha512-crk4Qu3pmXwgxdSgGhgA/eXiJAPQiX4GMOZZMXnqKxHX7TaoL+3gQVo/WeuAiogr07DpnfjIMpXXa+PAIvwPGQ==", 1099 | "dev": true, 1100 | "requires": { 1101 | "picomatch": "^2.0.4" 1102 | } 1103 | }, 1104 | "request": { 1105 | "version": "2.88.0", 1106 | "resolved": "https://registry.npmjs.org/request/-/request-2.88.0.tgz", 1107 | "integrity": "sha512-NAqBSrijGLZdM0WZNsInLJpkJokL72XYjUpnB0iwsRgxh7dB6COrHnTBNwN0E+lHDAJzu7kLAkDeY08z2/A0hg==", 1108 | "dev": true, 1109 | "requires": { 1110 | "aws-sign2": "~0.7.0", 1111 | "aws4": "^1.8.0", 1112 | "caseless": "~0.12.0", 1113 | "combined-stream": "~1.0.6", 1114 | "extend": "~3.0.2", 1115 | "forever-agent": "~0.6.1", 1116 | "form-data": "~2.3.2", 1117 | "har-validator": "~5.1.0", 1118 | "http-signature": "~1.2.0", 1119 | "is-typedarray": "~1.0.0", 1120 | "isstream": "~0.1.2", 1121 | "json-stringify-safe": "~5.0.1", 1122 | "mime-types": "~2.1.19", 1123 | "oauth-sign": "~0.9.0", 1124 | "performance-now": "^2.1.0", 1125 | "qs": "~6.5.2", 1126 | "safe-buffer": "^5.1.2", 1127 | "tough-cookie": "~2.4.3", 1128 | "tunnel-agent": "^0.6.0", 1129 | "uuid": "^3.3.2" 1130 | } 1131 | }, 1132 | "require-directory": { 1133 | "version": "2.1.1", 1134 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 1135 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", 1136 | "dev": true 1137 | }, 1138 | "require-main-filename": { 1139 | "version": "2.0.0", 1140 | "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", 1141 | "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", 1142 | "dev": true 1143 | }, 1144 | "requires-port": { 1145 | "version": "1.0.0", 1146 | "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", 1147 | "integrity": "sha1-kl0mAdOaxIXgkc8NpcbmlNw9yv8=", 1148 | "dev": true 1149 | }, 1150 | "resolve": { 1151 | "version": "1.14.2", 1152 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.14.2.tgz", 1153 | "integrity": "sha512-EjlOBLBO1kxsUxsKjLt7TAECyKW6fOh1VRkykQkKGzcBbjjPIxBqGh0jf7GJ3k/f5mxMqW3htMD3WdTUVtW8HQ==", 1154 | "dev": true, 1155 | "requires": { 1156 | "path-parse": "^1.0.6" 1157 | } 1158 | }, 1159 | "safe-buffer": { 1160 | "version": "5.2.0", 1161 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.0.tgz", 1162 | "integrity": "sha512-fZEwUGbVl7kouZs1jCdMLdt95hdIv0ZeHg6L7qPeciMZhZ+/gdesW4wgTARkrFWEpspjEATAzUGPG8N2jJiwbg==", 1163 | "dev": true 1164 | }, 1165 | "safer-buffer": { 1166 | "version": "2.1.2", 1167 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1168 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", 1169 | "dev": true 1170 | }, 1171 | "semver": { 1172 | "version": "5.7.1", 1173 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1174 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1175 | "dev": true 1176 | }, 1177 | "set-blocking": { 1178 | "version": "2.0.0", 1179 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", 1180 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", 1181 | "dev": true 1182 | }, 1183 | "source-map": { 1184 | "version": "0.6.1", 1185 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 1186 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 1187 | "dev": true 1188 | }, 1189 | "source-map-support": { 1190 | "version": "0.5.3", 1191 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.3.tgz", 1192 | "integrity": "sha512-eKkTgWYeBOQqFGXRfKabMFdnWepo51vWqEdoeikaEPFiJC7MCU5j2h4+6Q8npkZTeLGbSyecZvRxiSoWl3rh+w==", 1193 | "dev": true, 1194 | "requires": { 1195 | "source-map": "^0.6.0" 1196 | } 1197 | }, 1198 | "sprintf-js": { 1199 | "version": "1.0.3", 1200 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 1201 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 1202 | "dev": true 1203 | }, 1204 | "sshpk": { 1205 | "version": "1.16.1", 1206 | "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", 1207 | "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", 1208 | "dev": true, 1209 | "requires": { 1210 | "asn1": "~0.2.3", 1211 | "assert-plus": "^1.0.0", 1212 | "bcrypt-pbkdf": "^1.0.0", 1213 | "dashdash": "^1.12.0", 1214 | "ecc-jsbn": "~0.1.1", 1215 | "getpass": "^0.1.1", 1216 | "jsbn": "~0.1.0", 1217 | "safer-buffer": "^2.0.2", 1218 | "tweetnacl": "~0.14.0" 1219 | } 1220 | }, 1221 | "string-width": { 1222 | "version": "2.1.1", 1223 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", 1224 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", 1225 | "dev": true, 1226 | "requires": { 1227 | "is-fullwidth-code-point": "^2.0.0", 1228 | "strip-ansi": "^4.0.0" 1229 | } 1230 | }, 1231 | "string.prototype.trimleft": { 1232 | "version": "2.1.1", 1233 | "resolved": "https://registry.npmjs.org/string.prototype.trimleft/-/string.prototype.trimleft-2.1.1.tgz", 1234 | "integrity": "sha512-iu2AGd3PuP5Rp7x2kEZCrB2Nf41ehzh+goo8TV7z8/XDBbsvc6HQIlUl9RjkZ4oyrW1XM5UwlGl1oVEaDjg6Ag==", 1235 | "dev": true, 1236 | "requires": { 1237 | "define-properties": "^1.1.3", 1238 | "function-bind": "^1.1.1" 1239 | } 1240 | }, 1241 | "string.prototype.trimright": { 1242 | "version": "2.1.1", 1243 | "resolved": "https://registry.npmjs.org/string.prototype.trimright/-/string.prototype.trimright-2.1.1.tgz", 1244 | "integrity": "sha512-qFvWL3/+QIgZXVmJBfpHmxLB7xsUXz6HsUmP8+5dRaC3Q7oKUv9Vo6aMCRZC1smrtyECFsIT30PqBJ1gTjAs+g==", 1245 | "dev": true, 1246 | "requires": { 1247 | "define-properties": "^1.1.3", 1248 | "function-bind": "^1.1.1" 1249 | } 1250 | }, 1251 | "strip-ansi": { 1252 | "version": "4.0.0", 1253 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", 1254 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", 1255 | "dev": true, 1256 | "requires": { 1257 | "ansi-regex": "^3.0.0" 1258 | } 1259 | }, 1260 | "strip-json-comments": { 1261 | "version": "2.0.1", 1262 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1263 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1264 | "dev": true 1265 | }, 1266 | "supports-color": { 1267 | "version": "5.4.0", 1268 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.4.0.tgz", 1269 | "integrity": "sha512-zjaXglF5nnWpsq470jSv6P9DwPvgLkuapYmfDm3JWOm0vkNTVF2tI4UrN2r6jH1qM/uc/WtxYY1hYoA2dOKj5w==", 1270 | "dev": true, 1271 | "requires": { 1272 | "has-flag": "^3.0.0" 1273 | } 1274 | }, 1275 | "to-regex-range": { 1276 | "version": "5.0.1", 1277 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1278 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1279 | "dev": true, 1280 | "requires": { 1281 | "is-number": "^7.0.0" 1282 | } 1283 | }, 1284 | "tough-cookie": { 1285 | "version": "2.4.3", 1286 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.4.3.tgz", 1287 | "integrity": "sha512-Q5srk/4vDM54WJsJio3XNn6K2sCG+CQ8G5Wz6bZhRZoAe/+TxjWB/GlFAnYEbkYVlON9FMk/fE3h2RLpPXo4lQ==", 1288 | "dev": true, 1289 | "requires": { 1290 | "psl": "^1.1.24", 1291 | "punycode": "^1.4.1" 1292 | }, 1293 | "dependencies": { 1294 | "punycode": { 1295 | "version": "1.4.1", 1296 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", 1297 | "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=", 1298 | "dev": true 1299 | } 1300 | } 1301 | }, 1302 | "ts-node": { 1303 | "version": "8.5.4", 1304 | "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-8.5.4.tgz", 1305 | "integrity": "sha512-izbVCRV68EasEPQ8MSIGBNK9dc/4sYJJKYA+IarMQct1RtEot6Xp0bXuClsbUSnKpg50ho+aOAx8en5c+y4OFw==", 1306 | "dev": true, 1307 | "requires": { 1308 | "arg": "^4.1.0", 1309 | "diff": "^4.0.1", 1310 | "make-error": "^1.1.1", 1311 | "source-map-support": "^0.5.6", 1312 | "yn": "^3.0.0" 1313 | }, 1314 | "dependencies": { 1315 | "diff": { 1316 | "version": "4.0.1", 1317 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.1.tgz", 1318 | "integrity": "sha512-s2+XdvhPCOF01LRQBC8hf4vhbVmI2CGS5aZnxLJlT5FtdhPCDFq80q++zK2KlrVorVDdL5BOGZ/VfLrVtYNF+Q==", 1319 | "dev": true 1320 | }, 1321 | "source-map-support": { 1322 | "version": "0.5.16", 1323 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.16.tgz", 1324 | "integrity": "sha512-efyLRJDr68D9hBBNIPWFjhpFzURh+KJykQwvMyW5UiZzYwoF6l4YMMDIJJEyFWxWCqfyxLzz6tSfUFR+kXXsVQ==", 1325 | "dev": true, 1326 | "requires": { 1327 | "buffer-from": "^1.0.0", 1328 | "source-map": "^0.6.0" 1329 | } 1330 | } 1331 | } 1332 | }, 1333 | "tslib": { 1334 | "version": "1.10.0", 1335 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.10.0.tgz", 1336 | "integrity": "sha512-qOebF53frne81cf0S9B41ByenJ3/IuH8yJKngAX35CmiZySA0khhkovshKK+jGCaMnVomla7gVlIcc3EvKPbTQ==", 1337 | "dev": true 1338 | }, 1339 | "tslint": { 1340 | "version": "5.20.1", 1341 | "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.20.1.tgz", 1342 | "integrity": "sha512-EcMxhzCFt8k+/UP5r8waCf/lzmeSyVlqxqMEDQE7rWYiQky8KpIBz1JAoYXfROHrPZ1XXd43q8yQnULOLiBRQg==", 1343 | "dev": true, 1344 | "requires": { 1345 | "@babel/code-frame": "^7.0.0", 1346 | "builtin-modules": "^1.1.1", 1347 | "chalk": "^2.3.0", 1348 | "commander": "^2.12.1", 1349 | "diff": "^4.0.1", 1350 | "glob": "^7.1.1", 1351 | "js-yaml": "^3.13.1", 1352 | "minimatch": "^3.0.4", 1353 | "mkdirp": "^0.5.1", 1354 | "resolve": "^1.3.2", 1355 | "semver": "^5.3.0", 1356 | "tslib": "^1.8.0", 1357 | "tsutils": "^2.29.0" 1358 | }, 1359 | "dependencies": { 1360 | "diff": { 1361 | "version": "4.0.1", 1362 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.1.tgz", 1363 | "integrity": "sha512-s2+XdvhPCOF01LRQBC8hf4vhbVmI2CGS5aZnxLJlT5FtdhPCDFq80q++zK2KlrVorVDdL5BOGZ/VfLrVtYNF+Q==", 1364 | "dev": true 1365 | } 1366 | } 1367 | }, 1368 | "tsutils": { 1369 | "version": "2.29.0", 1370 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", 1371 | "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", 1372 | "dev": true, 1373 | "requires": { 1374 | "tslib": "^1.8.1" 1375 | } 1376 | }, 1377 | "tunnel-agent": { 1378 | "version": "0.6.0", 1379 | "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", 1380 | "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", 1381 | "dev": true, 1382 | "requires": { 1383 | "safe-buffer": "^5.0.1" 1384 | } 1385 | }, 1386 | "tweetnacl": { 1387 | "version": "0.14.5", 1388 | "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", 1389 | "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", 1390 | "dev": true 1391 | }, 1392 | "typescript": { 1393 | "version": "3.7.4", 1394 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.7.4.tgz", 1395 | "integrity": "sha512-A25xv5XCtarLwXpcDNZzCGvW2D1S3/bACratYBx2sax8PefsFhlYmkQicKHvpYflFS8if4zne5zT5kpJ7pzuvw==", 1396 | "dev": true 1397 | }, 1398 | "uri-js": { 1399 | "version": "4.2.2", 1400 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", 1401 | "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", 1402 | "dev": true, 1403 | "requires": { 1404 | "punycode": "^2.1.0" 1405 | } 1406 | }, 1407 | "url-parse": { 1408 | "version": "1.4.7", 1409 | "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.4.7.tgz", 1410 | "integrity": "sha512-d3uaVyzDB9tQoSXFvuSUNFibTd9zxd2bkVrDRvF5TmvWWQwqE4lgYJ5m+x1DbecWkw+LK4RNl2CU1hHuOKPVlg==", 1411 | "dev": true, 1412 | "requires": { 1413 | "querystringify": "^2.1.1", 1414 | "requires-port": "^1.0.0" 1415 | } 1416 | }, 1417 | "uuid": { 1418 | "version": "3.3.3", 1419 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.3.tgz", 1420 | "integrity": "sha512-pW0No1RGHgzlpHJO1nsVrHKpOEIxkGg1xB+v0ZmdNH5OAeAwzAVrCnI2/6Mtx+Uys6iaylxa+D3g4j63IKKjSQ==", 1421 | "dev": true 1422 | }, 1423 | "verror": { 1424 | "version": "1.10.0", 1425 | "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", 1426 | "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", 1427 | "dev": true, 1428 | "requires": { 1429 | "assert-plus": "^1.0.0", 1430 | "core-util-is": "1.0.2", 1431 | "extsprintf": "^1.2.0" 1432 | } 1433 | }, 1434 | "vscode": { 1435 | "version": "1.1.36", 1436 | "resolved": "https://registry.npmjs.org/vscode/-/vscode-1.1.36.tgz", 1437 | "integrity": "sha512-cGFh9jmGLcTapCpPCKvn8aG/j9zVQ+0x5hzYJq5h5YyUXVGa1iamOaB2M2PZXoumQPES4qeAP1FwkI0b6tL4bQ==", 1438 | "dev": true, 1439 | "requires": { 1440 | "glob": "^7.1.2", 1441 | "mocha": "^5.2.0", 1442 | "request": "^2.88.0", 1443 | "semver": "^5.4.1", 1444 | "source-map-support": "^0.5.0", 1445 | "url-parse": "^1.4.4", 1446 | "vscode-test": "^0.4.1" 1447 | }, 1448 | "dependencies": { 1449 | "he": { 1450 | "version": "1.1.1", 1451 | "resolved": "https://registry.npmjs.org/he/-/he-1.1.1.tgz", 1452 | "integrity": "sha1-k0EP0hsAlzUVH4howvJx80J+I/0=", 1453 | "dev": true 1454 | }, 1455 | "mocha": { 1456 | "version": "5.2.0", 1457 | "resolved": "https://registry.npmjs.org/mocha/-/mocha-5.2.0.tgz", 1458 | "integrity": "sha512-2IUgKDhc3J7Uug+FxMXuqIyYzH7gJjXECKe/w43IGgQHTSj3InJi+yAA7T24L9bQMRKiUEHxEX37G5JpVUGLcQ==", 1459 | "dev": true, 1460 | "requires": { 1461 | "browser-stdout": "1.3.1", 1462 | "commander": "2.15.1", 1463 | "debug": "3.1.0", 1464 | "diff": "3.5.0", 1465 | "escape-string-regexp": "1.0.5", 1466 | "glob": "7.1.2", 1467 | "growl": "1.10.5", 1468 | "he": "1.1.1", 1469 | "minimatch": "3.0.4", 1470 | "mkdirp": "0.5.1", 1471 | "supports-color": "5.4.0" 1472 | } 1473 | } 1474 | } 1475 | }, 1476 | "vscode-test": { 1477 | "version": "0.4.3", 1478 | "resolved": "https://registry.npmjs.org/vscode-test/-/vscode-test-0.4.3.tgz", 1479 | "integrity": "sha512-EkMGqBSefZH2MgW65nY05rdRSko15uvzq4VAPM5jVmwYuFQKE7eikKXNJDRxL+OITXHB6pI+a3XqqD32Y3KC5w==", 1480 | "dev": true, 1481 | "requires": { 1482 | "http-proxy-agent": "^2.1.0", 1483 | "https-proxy-agent": "^2.2.1" 1484 | } 1485 | }, 1486 | "which": { 1487 | "version": "1.3.1", 1488 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", 1489 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", 1490 | "dev": true, 1491 | "requires": { 1492 | "isexe": "^2.0.0" 1493 | } 1494 | }, 1495 | "which-module": { 1496 | "version": "2.0.0", 1497 | "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", 1498 | "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", 1499 | "dev": true 1500 | }, 1501 | "wide-align": { 1502 | "version": "1.1.3", 1503 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", 1504 | "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", 1505 | "dev": true, 1506 | "requires": { 1507 | "string-width": "^1.0.2 || 2" 1508 | } 1509 | }, 1510 | "wrap-ansi": { 1511 | "version": "5.1.0", 1512 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", 1513 | "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", 1514 | "dev": true, 1515 | "requires": { 1516 | "ansi-styles": "^3.2.0", 1517 | "string-width": "^3.0.0", 1518 | "strip-ansi": "^5.0.0" 1519 | }, 1520 | "dependencies": { 1521 | "ansi-regex": { 1522 | "version": "4.1.0", 1523 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 1524 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 1525 | "dev": true 1526 | }, 1527 | "string-width": { 1528 | "version": "3.1.0", 1529 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 1530 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 1531 | "dev": true, 1532 | "requires": { 1533 | "emoji-regex": "^7.0.1", 1534 | "is-fullwidth-code-point": "^2.0.0", 1535 | "strip-ansi": "^5.1.0" 1536 | } 1537 | }, 1538 | "strip-ansi": { 1539 | "version": "5.2.0", 1540 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1541 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1542 | "dev": true, 1543 | "requires": { 1544 | "ansi-regex": "^4.1.0" 1545 | } 1546 | } 1547 | } 1548 | }, 1549 | "wrappy": { 1550 | "version": "1.0.2", 1551 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1552 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1553 | "dev": true 1554 | }, 1555 | "y18n": { 1556 | "version": "4.0.0", 1557 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", 1558 | "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==", 1559 | "dev": true 1560 | }, 1561 | "yargs": { 1562 | "version": "13.3.0", 1563 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.0.tgz", 1564 | "integrity": "sha512-2eehun/8ALW8TLoIl7MVaRUrg+yCnenu8B4kBlRxj3GJGDKU1Og7sMXPNm1BYyM1DOJmTZ4YeN/Nwxv+8XJsUA==", 1565 | "dev": true, 1566 | "requires": { 1567 | "cliui": "^5.0.0", 1568 | "find-up": "^3.0.0", 1569 | "get-caller-file": "^2.0.1", 1570 | "require-directory": "^2.1.1", 1571 | "require-main-filename": "^2.0.0", 1572 | "set-blocking": "^2.0.0", 1573 | "string-width": "^3.0.0", 1574 | "which-module": "^2.0.0", 1575 | "y18n": "^4.0.0", 1576 | "yargs-parser": "^13.1.1" 1577 | }, 1578 | "dependencies": { 1579 | "ansi-regex": { 1580 | "version": "4.1.0", 1581 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 1582 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 1583 | "dev": true 1584 | }, 1585 | "string-width": { 1586 | "version": "3.1.0", 1587 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 1588 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 1589 | "dev": true, 1590 | "requires": { 1591 | "emoji-regex": "^7.0.1", 1592 | "is-fullwidth-code-point": "^2.0.0", 1593 | "strip-ansi": "^5.1.0" 1594 | } 1595 | }, 1596 | "strip-ansi": { 1597 | "version": "5.2.0", 1598 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1599 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1600 | "dev": true, 1601 | "requires": { 1602 | "ansi-regex": "^4.1.0" 1603 | } 1604 | } 1605 | } 1606 | }, 1607 | "yargs-parser": { 1608 | "version": "13.1.1", 1609 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.1.tgz", 1610 | "integrity": "sha512-oVAVsHz6uFrg3XQheFII8ESO2ssAf9luWuAd6Wexsu4F3OtIW0o8IribPXYrD4WC24LWtPrJlGy87y5udK+dxQ==", 1611 | "dev": true, 1612 | "requires": { 1613 | "camelcase": "^5.0.0", 1614 | "decamelize": "^1.2.0" 1615 | } 1616 | }, 1617 | "yargs-unparser": { 1618 | "version": "1.6.0", 1619 | "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-1.6.0.tgz", 1620 | "integrity": "sha512-W9tKgmSn0DpSatfri0nx52Joq5hVXgeLiqR/5G0sZNDoLZFOr/xjBUDcShCOGNsBnEMNo1KAMBkTej1Hm62HTw==", 1621 | "dev": true, 1622 | "requires": { 1623 | "flat": "^4.1.0", 1624 | "lodash": "^4.17.15", 1625 | "yargs": "^13.3.0" 1626 | } 1627 | }, 1628 | "yn": { 1629 | "version": "3.1.1", 1630 | "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", 1631 | "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", 1632 | "dev": true 1633 | } 1634 | } 1635 | } 1636 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "vscode-fsharp-refactor", 3 | "displayName": "vscode fsharp refactor", 4 | "description": "F# refactoring tools for vscode", 5 | "version": "0.6.2", 6 | "publisher": "danmannock", 7 | "author": "Dan Mannock", 8 | "engines": { 9 | "vscode": "^1.19.0" 10 | }, 11 | "categories": [ 12 | "Other" 13 | ], 14 | "main": "./out/extension", 15 | "activationEvents": [ 16 | "onCommand:fsharp-refactor.extractLet", 17 | "onCommand:fsharp-refactor.inlineLet" 18 | ], 19 | "contributes": { 20 | "commands": [ 21 | { 22 | "command": "fsharp-refactor.extractLet", 23 | "title": "Extract Let Binding", 24 | "description": "Extract expression to a let binding", 25 | "category": "F#Refactor" 26 | }, 27 | { 28 | "command": "fsharp-refactor.inlineLet", 29 | "title": "Inlines Binding", 30 | "description": "Inlines let binding to usages", 31 | "category": "F#Refactor" 32 | } 33 | ], 34 | "menus": { 35 | "commandPalette": [ 36 | { 37 | "command": "fsharp-refactor.extractLet", 38 | "when": "editorHasSelection" 39 | }, 40 | { 41 | "command": "fsharp-refactor.inlineLet", 42 | "when": "editorTextFocus" 43 | } 44 | ] 45 | }, 46 | "keybindings": [ 47 | { 48 | "command": "fsharp-refactor.extractLet", 49 | "key": "Ctrl+Shift+R", 50 | "mac": "cmd+Shift+R", 51 | "when": "editorHasSelection && editorLangId == 'fsharp'" 52 | }, 53 | { 54 | "command": "fsharp-refactor.inlineLet", 55 | "key": "Ctrl+Shift+I", 56 | "mac": "cmd+Shift+I", 57 | "when": "editorTextFocus && editorLangId == 'fsharp'" 58 | } 59 | ] 60 | }, 61 | "scripts": { 62 | "lint": "tslint -p ./", 63 | "compile": "tsc -p ./", 64 | "watch": "tsc -watch -p ./", 65 | "unittest": "mocha -r ts-node/register ./src/test-unit/**/*.test.ts", 66 | "unittest:watch": "mocha -r ts-node/register ./src/test-unit/**/*.test.ts -w --watch-extensions ts", 67 | "test": "npm run compile && node ./node_modules/vscode/bin/test", 68 | "build": "npm run lint && npm run unittest && npm run test", 69 | "vscode:prepublish": "npm run build", 70 | "postinstall": "node ./node_modules/vscode/bin/install" 71 | }, 72 | "devDependencies": { 73 | "@types/mocha": "^5.2.7", 74 | "@types/node": "^13.1.4", 75 | "mocha": "^7.0.0", 76 | "ts-node": "^8.5.4", 77 | "tslint": "^5.20.1", 78 | "typescript": "^3.7.4", 79 | "vscode": "^1.1.36" 80 | }, 81 | "license": "MIT", 82 | "repository": { 83 | "type": "git", 84 | "url": "https://github.com/dmannock/vscode-fsharp-refactor.git" 85 | }, 86 | "keywords": [ 87 | "fSharp", 88 | "vscode", 89 | "vscode-extension", 90 | "tooling", 91 | "refactoring" 92 | ] 93 | } 94 | -------------------------------------------------------------------------------- /src/extension-commands.ts: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | import * as vscode from "vscode"; 3 | import { 4 | getExtractedString, 5 | IMatchedBindingLine, 6 | isLambdaSelection, 7 | isStringSelection, 8 | lambdaBindingFromSelection, 9 | matchBindingLine, 10 | } from "./general-funcs"; 11 | 12 | import { 13 | getBindingDeclarationAbove, 14 | getExpandedSelection, 15 | getIndentation, 16 | getSelectionDetails, 17 | getWordInstancesBelow, 18 | ISelectionDetails, 19 | isSelectionValid, 20 | } from "./vscode-funcs"; 21 | 22 | const initialBindingName = "extracted"; 23 | 24 | export async function extractLet(editor: vscode.TextEditor): Promise { 25 | if (editor.selections.length > 1) { 26 | vscode.window.showWarningMessage("Multiple selection are not supported"); 27 | return false; 28 | } 29 | const document = editor.document; 30 | const selectionDetails = getSelectionDetails(editor.selections, document); 31 | if (!isSelectionValid(selectionDetails)) { 32 | return false; 33 | } 34 | const indentation = getIndentation(document, selectionDetails.line); 35 | const textLine = document.lineAt(selectionDetails.line).text; 36 | let extractedBinding; 37 | if (isLambdaSelection(selectionDetails.text)) { 38 | const { args, body } = lambdaBindingFromSelection(selectionDetails.text); 39 | extractedBinding = `${indentation}let ${initialBindingName} ${args.join(" ")} = ${body}\r\n`; 40 | } else if (isStringSelection(selectionDetails.text, textLine)) { 41 | const { start, end } = selectionDetails.selection; 42 | const { extractedText, newLine } = getExtractedString(textLine, start.character, end.character); 43 | extractedBinding = `${indentation}let ${initialBindingName} = ${extractedText}\r\n`; 44 | return editor.edit((eb) => { 45 | eb.replace(new vscode.Range( 46 | new vscode.Position(selectionDetails.line, 0), 47 | new vscode.Position(selectionDetails.line, textLine.length) 48 | ), newLine); 49 | eb.insert(new vscode.Position(selectionDetails.line, 0), extractedBinding); 50 | }); 51 | } else { 52 | extractedBinding = `${indentation}let ${initialBindingName} = ${selectionDetails.text}\r\n`; 53 | } 54 | return editor.edit((eb) => { 55 | eb.replace(selectionDetails.selection, initialBindingName); 56 | eb.insert(new vscode.Position(selectionDetails.line, 0), extractedBinding); 57 | }); 58 | } 59 | 60 | export async function inlineLet(editor: vscode.TextEditor): Promise { 61 | if (editor.selections.length > 1) { 62 | vscode.window.showWarningMessage("Multiple selection are not supported"); 63 | return false; 64 | } 65 | const document = editor.document; 66 | const selectionDetails = getExpandedSelection(editor.selections, document); 67 | if (!isSelectionValid(selectionDetails)) { 68 | return false; 69 | } 70 | const currentLine = document.lineAt(selectionDetails.line); 71 | const currentLineRegexMatch = matchBindingLine(selectionDetails.text)(currentLine.text); 72 | if (currentLineRegexMatch) { 73 | return await inlineAllOccurrences(editor, currentLineRegexMatch, currentLine); 74 | } else { 75 | const bindingDeclaration = getBindingDeclarationAbove(document, selectionDetails.text, 76 | selectionDetails.line, currentLine.firstNonWhitespaceCharacterIndex); 77 | if (bindingDeclaration) { 78 | return await inlineAllOccurrences(editor, bindingDeclaration.matchedLineRegexMatch, 79 | bindingDeclaration.matchedLine); 80 | } 81 | } 82 | } 83 | 84 | async function inlineAllOccurrences(editor: vscode.TextEditor, 85 | matchedBindingLine: IMatchedBindingLine, 86 | currentLine: vscode.TextLine 87 | ): Promise { 88 | const document = editor.document; 89 | const { bindingName, expression, requiresParens } = matchedBindingLine; 90 | const occurrencesToReplace = getWordInstancesBelow(document, bindingName, 91 | currentLine.lineNumber, currentLine.firstNonWhitespaceCharacterIndex); 92 | if (occurrencesToReplace.length === 0) { 93 | vscode.window.showWarningMessage("No occurrences were found to replace."); 94 | return false; 95 | } 96 | return editor.edit((eb) => { 97 | eb.delete(currentLine.rangeIncludingLineBreak); 98 | for (const range of occurrencesToReplace) { 99 | eb.replace(range, requiresParens ? `(${expression})` : expression); 100 | } 101 | }); 102 | } 103 | -------------------------------------------------------------------------------- /src/extension.ts: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | import * as vscode from "vscode"; 3 | import { 4 | extractLet, 5 | inlineLet 6 | } from "./extension-commands"; 7 | 8 | export function activate(context: vscode.ExtensionContext) { 9 | registerWithAutoDisposal("fsharp-refactor.extractLet", (editor) => 10 | extractLet(editor) 11 | .then(() => vscode.commands.executeCommand("editor.action.rename")) 12 | ); 13 | 14 | registerWithAutoDisposal("fsharp-refactor.inlineLet", inlineLet); 15 | 16 | function registerWithAutoDisposal(command, handler) { 17 | context.subscriptions.push(vscode.commands.registerTextEditorCommand(command, handler)); 18 | } 19 | 20 | /* tslint:disable:no-console */ 21 | console.log("F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#"); 22 | console.log("F# Fsharp Refactor extension active :) F#"); 23 | console.log("F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#F#"); 24 | /* tslint:enable:no-console */ 25 | } 26 | -------------------------------------------------------------------------------- /src/general-funcs.ts: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | 3 | const lambdaRegexPattern = `^\\(fun([^-]+)->(.+)\\)$`; 4 | const createStringRegexPattern = 5 | (selectedText) => `${selectedText.startsWith(`"`) 6 | ? "" 7 | : `\\"`}(.*)${selectedText}(.*)${selectedText.endsWith(`"`) 8 | ? "" 9 | : `\\"`}`; 10 | 11 | export interface IMatchedBindingLine { 12 | indentation: string; 13 | binding: string; 14 | bindingName: string; 15 | expression: string; 16 | typeAnnotation: string; 17 | requiresParens: boolean; 18 | } 19 | 20 | export function ensureParenthesesWrapping(text) { 21 | const bracketsCount = text 22 | .split("") 23 | .reduce((acc, cur) => 24 | cur === "(" 25 | ? Object.assign(acc, { open: acc.open + 1 }) 26 | : cur === ")" 27 | ? Object.assign(acc, { close: acc.close + 1 }) 28 | : acc, 29 | { open: 0, close: 0}); 30 | return bracketsCount.open > bracketsCount.close 31 | ? text + ")" 32 | : bracketsCount.open < bracketsCount.close 33 | ? "(" + text 34 | : text; 35 | } 36 | 37 | export function isLambdaSelection(selectedText: string) { 38 | return new RegExp(lambdaRegexPattern).test(selectedText); 39 | } 40 | 41 | export function isStringSelection(selectedText: string, wholeLine: string) { 42 | return new RegExp(createStringRegexPattern(selectedText)).test(wholeLine); 43 | } 44 | 45 | export function lambdaBindingFromSelection(text: string) { 46 | const [, rawArgs, rawBody] = new RegExp(lambdaRegexPattern).exec(text); 47 | const args = rawArgs.split(" ").filter((x) => x); 48 | return { 49 | args, 50 | body: ensureParenthesesWrapping(rawBody).trim(), 51 | }; 52 | } 53 | 54 | export const matchBindingLine = (bindingNameToMatch = "\\S+") => (text): IMatchedBindingLine => { 55 | const pattern = `^(\\s*)(let)\\s+\\b(${bindingNameToMatch})\\b\\s*(:([\\s\\S]+))?\\s*=\\s*([\\s\\S]+)$`; 56 | const matched = new RegExp(pattern).exec(text); 57 | if (!matched) { 58 | return null; 59 | } 60 | const [, indentation, binding, bindingName, , typeAnnotation, expression] = matched; 61 | const requiresParens = expression.trim().includes(" ") || expression.includes("(", 1); 62 | return { 63 | binding, 64 | bindingName, 65 | expression, 66 | indentation, 67 | requiresParens, 68 | typeAnnotation: typeAnnotation && typeAnnotation.trim(), 69 | }; 70 | }; 71 | 72 | export function wordIndexesInText(text: string, toFind: string): number[] { 73 | const found = []; 74 | const regex = new RegExp(`\\b${toFind}\\b`, "g"); 75 | let match; 76 | do { 77 | match = regex.exec(text); 78 | if (match == null) { 79 | return found; 80 | } 81 | found.push(match.index); 82 | } while (true); 83 | } 84 | 85 | export function getExtractedString(textLine: string, selectionStartPos: number, selectionEndPos: number) { 86 | const initialBindingName = "extracted"; 87 | const quoteChars = `"`; 88 | const selectedText = textLine.substring(selectionStartPos, selectionEndPos); 89 | 90 | const { hasStartQuote, hasEndQuote, containsWholeQuotedString } = stringQuoteDetails(selectedText, quoteChars); 91 | 92 | const textBeforeSelection = textLine.substring(0, selectionStartPos); 93 | const mid = textLine.substring(selectionStartPos, selectionStartPos); 94 | const textAfterSelection = textLine.substring(selectionEndPos, Infinity); 95 | 96 | const extractedText = `${!hasStartQuote && !containsWholeQuotedString 97 | ? `"` : ""}${selectedText}${!hasEndQuote ? `"` : ""}`; 98 | 99 | let newLine = ""; 100 | // issue #1 - added to resolve that and only that 101 | if (!hasStartQuote && hasEndQuote && containsWholeQuotedString) { 102 | newLine = `${textBeforeSelection}${initialBindingName}`; 103 | } else { 104 | newLine = `${textBeforeSelection}` 105 | + `${hasStartQuote && !hasEndQuote 106 | ? `${initialBindingName} + ${quoteChars}${textAfterSelection}` : ""}` 107 | + `${hasEndQuote && !hasStartQuote 108 | ? `${mid}${quoteChars} + ${initialBindingName}` : ""}` 109 | + `${!hasStartQuote && !hasEndQuote 110 | ? `${quoteChars} + ${initialBindingName} + "${textAfterSelection}` : "" }`; 111 | } 112 | return { 113 | extractedText, 114 | newLine, 115 | }; 116 | } 117 | 118 | export function stringQuoteDetails(text: string, quoteChar: string = `"`) { 119 | const hasStartQuote = text[0] === quoteChar; 120 | const hasEndQuote = text[text.length - 1] === quoteChar; 121 | const containsWholeQuotedString = text.split("").filter((c) => c === quoteChar).length === 2; 122 | return { hasStartQuote, hasEndQuote, containsWholeQuotedString }; 123 | } 124 | -------------------------------------------------------------------------------- /src/test-unit/general-funcs.test.ts: -------------------------------------------------------------------------------- 1 | import * as assert from "assert"; 2 | import { 3 | ensureParenthesesWrapping, 4 | getExtractedString, 5 | matchBindingLine, 6 | stringQuoteDetails, 7 | wordIndexesInText 8 | } from "../general-funcs"; 9 | 10 | describe("General Funcs: ensureParenthesesWrapping", () => { 11 | 12 | it("doesn't wrap text without parentheses", () => { 13 | const actual = ensureParenthesesWrapping("wrap me"); 14 | assert.equal(actual, "wrap me"); 15 | }); 16 | 17 | it("wraps missing start parentheses", () => { 18 | const actual = ensureParenthesesWrapping("wrap me)"); 19 | assert.equal(actual, "(wrap me)"); 20 | }); 21 | 22 | it("wraps missing end parentheses", () => { 23 | const actual = ensureParenthesesWrapping("(wrap me"); 24 | assert.equal(actual, "(wrap me)"); 25 | }); 26 | 27 | it("wraps multiple opening with missing end parentheses", () => { 28 | const actual = ensureParenthesesWrapping("(complete(wrap me)missing"); 29 | assert.equal(actual, "(complete(wrap me)missing)"); 30 | }); 31 | 32 | }); 33 | 34 | describe("General Funcs: wordIndexesInText", () => { 35 | 36 | it("with non-existent text finds no occurrences", () => { 37 | const actual = wordIndexesInText("ab cd ef be", "findme"); 38 | assert.deepEqual(actual, []); 39 | }); 40 | 41 | it("finds multiple occurrences", () => { 42 | const actual = wordIndexesInText("ab findme cd ef findme be", "findme"); 43 | assert.deepEqual(actual, [3, 16]); 44 | }); 45 | 46 | }); 47 | 48 | describe("General Funcs: getExtractedString", () => { 49 | 50 | it("extracts start of quoted string with double quote selected", () => { 51 | const wholeLine = `let stringToExtract = "/usr/bin/bash"`; 52 | // select "/usr/bin/ 53 | const { extractedText, newLine } = getExtractedString(wholeLine, 22, 32); 54 | 55 | const expectedExtractedString = `"/usr/bin/"`; 56 | const expectedWholeLine = `let stringToExtract = extracted + "bash"`; 57 | 58 | assert.equal(extractedText, expectedExtractedString); 59 | assert.equal(newLine, expectedWholeLine); 60 | }); 61 | 62 | it("extracts end of quoted string with double quote selected", () => { 63 | const wholeLine = `let stringToExtract = "/usr/bin/bash"`; 64 | // select bash" 65 | const { extractedText, newLine } = getExtractedString(wholeLine, 32, 37); 66 | 67 | const expectedExtractedString = `"bash"`; 68 | const expectedWholeLine = `let stringToExtract = "/usr/bin/" + extracted`; 69 | 70 | assert.equal(extractedText, expectedExtractedString); 71 | assert.equal(newLine, expectedWholeLine); 72 | }); 73 | 74 | it("extracts middle of quoted string", () => { 75 | const wholeLine = `let stringToExtract = "/usr/bin/bash"`; 76 | // select bash 77 | const { extractedText, newLine } = getExtractedString(wholeLine, 27, 31); 78 | 79 | const expectedExtractedString = `"/bin"`; 80 | const expectedWholeLine = `let stringToExtract = "/usr" + extracted + "/bash"`; 81 | 82 | assert.equal(extractedText, expectedExtractedString); 83 | assert.equal(newLine, expectedWholeLine); 84 | }); 85 | 86 | it("extracts selection with quoted string at the end", () => { 87 | const wholeLine = `let str = types |> Array.map fsSig |> String.concat ", "`; 88 | // select bash 89 | const { extractedText, newLine } = getExtractedString(wholeLine, 10, 57); 90 | 91 | const expectedExtractedString = `types |> Array.map fsSig |> String.concat ", "`; 92 | const expectedWholeLine = `let str = extracted`; 93 | 94 | assert.equal(extractedText, expectedExtractedString); 95 | assert.equal(newLine, expectedWholeLine); 96 | }); 97 | 98 | }); 99 | 100 | describe("General Funcs: selectedStringPositions", () => { 101 | 102 | it("start and end quotes", () => { 103 | const selected = `"/usr/bin/bash"`; 104 | assert.deepEqual(stringQuoteDetails(selected), 105 | { 106 | containsWholeQuotedString: true, 107 | hasEndQuote: true, 108 | hasStartQuote: true, 109 | }); 110 | }); 111 | 112 | it("no start but with end quotes", () => { 113 | const selected = `/usr/bin/bash"`; 114 | assert.deepEqual(stringQuoteDetails(selected), 115 | { 116 | containsWholeQuotedString: false, 117 | hasEndQuote: true, 118 | hasStartQuote: false, 119 | }); 120 | }); 121 | 122 | it("start but no end quotes", () => { 123 | const selected = `"/usr/bin/bash`; 124 | assert.deepEqual(stringQuoteDetails(selected), 125 | { 126 | containsWholeQuotedString: false, 127 | hasEndQuote: false, 128 | hasStartQuote: true, 129 | }); 130 | }); 131 | 132 | it("no start quotes but ends with a complete string literal", () => { 133 | const selected = `types |> Array.map fsSig |> String.concat ", "`; 134 | assert.deepEqual(stringQuoteDetails(selected), 135 | { 136 | containsWholeQuotedString: true, 137 | hasEndQuote: true, 138 | hasStartQuote: false, 139 | }); 140 | }); 141 | 142 | }); 143 | 144 | describe("General Funcs: matchBindingLine", () => { 145 | 146 | it("binding returns parts for binding name", () => { 147 | const line = ` let inlineMe = 1 + arg1`; 148 | assert.deepEqual(matchBindingLine("inlineMe")(line), 149 | { 150 | binding: "let", 151 | bindingName: "inlineMe", 152 | expression: "1 + arg1", 153 | indentation: " ", 154 | requiresParens: true, 155 | typeAnnotation: null, 156 | }); 157 | }); 158 | 159 | it("binding returns parts for binding name with type annotation", () => { 160 | const line = `let expectedEvents :int list = []`; 161 | assert.deepEqual(matchBindingLine("expectedEvents")(line), 162 | { 163 | binding: "let", 164 | bindingName: "expectedEvents", 165 | expression: "[]", 166 | indentation: "", 167 | requiresParens: false, 168 | typeAnnotation: "int list", 169 | }); 170 | }); 171 | 172 | }); 173 | -------------------------------------------------------------------------------- /src/test/0-extension.test.ts: -------------------------------------------------------------------------------- 1 | import * as assert from "assert"; 2 | import * as vscode from "vscode"; 3 | 4 | const extensionId = "danmannock.vscode-fsharp-refactor"; 5 | 6 | // file prefix to run these first 7 | suite("Extension Tests", () => { 8 | 9 | test("should have loaded extension", () => { 10 | assert.ok(vscode.extensions.getExtension(extensionId)); 11 | }); 12 | 13 | test("should have activated extension", (done) => { 14 | const extension = vscode.extensions.getExtension(extensionId); 15 | if (!extension.isActive) { 16 | extension.activate().then( 17 | () => done(), 18 | () => done("Activation failed.")); 19 | return; 20 | } 21 | done(); 22 | }); 23 | 24 | }); 25 | -------------------------------------------------------------------------------- /src/test/1-extension-command-extractLet.test.ts: -------------------------------------------------------------------------------- 1 | import { extractLet } from "../extension-commands"; 2 | import { 3 | createSelection, 4 | runComparisonTest, 5 | } from "./utils"; 6 | 7 | suite("Extension 'extractLet' Command Tests", () => { 8 | 9 | // add comparison tests to array 10 | /* tslint:disable:object-literal-sort-keys */ 11 | [{ 12 | description: "should extract let binding (example 1)", 13 | content: `let test arg1 = 14 | let added = 1 + arg1 15 | let multi = added * 10 * arg1 16 | multi / 2`, 17 | expectedContent: `let test arg1 = 18 | let added = 1 + arg1 19 | let extracted = 10 * arg1 20 | let multi = added * extracted 21 | multi / 2`, 22 | // select '10 * arg1' 23 | selection: createSelection(2, 28, 2, 37), 24 | action: extractLet, 25 | }, 26 | { 27 | description: "should extract unary function to let binding (example 4)", 28 | content: `let extractLet chars = 29 | let noSpaces = chars |> Array.filter ((<>) ' ') 30 | noSpaces`, 31 | expectedContent: `let extractLet chars = 32 | let extracted = ((<>) ' ') 33 | let noSpaces = chars |> Array.filter extracted 34 | noSpaces`, 35 | // select i((<>) ' ') on line 1 36 | selection: createSelection(1, 45, 1, 55), 37 | action: extractLet, 38 | }, 39 | { 40 | description: "should extract parameterised function to let binding (example 5)", 41 | content: `let extractLambda o = 42 | let res = (o |> Array.fold (fun acc n -> (n |> Array.toList) @ acc ) []).Head 43 | res`, 44 | expectedContent: `let extractLambda o = 45 | let extracted acc n = (n |> Array.toList) @ acc 46 | let res = (o |> Array.fold extracted []).Head 47 | res`, 48 | // select (fun acc n -> (n |> Array.toList) @ acc ) on line 1 49 | selection: createSelection(1, 35, 1, 76), 50 | action: extractLet, 51 | }, 52 | { 53 | description: "should extract let binding when selection contains a lambda and other expressions (issue 6)", 54 | content: 55 | `let getter() = [1;2] 56 | let extractLambda = 57 | let res = getter() |> List.map (fun a -> a * 2) |> List.sum 58 | res`, 59 | expectedContent: 60 | `let getter() = [1;2] 61 | let extractLambda = 62 | let extracted = getter() |> List.map (fun a -> a * 2) 63 | let res = extracted |> List.sum 64 | res`, 65 | // select getter() |> List.map (fun a -> a * 2) on line 2 66 | selection: createSelection(2, 14, 2, 51), 67 | action: extractLet, 68 | }] 69 | .forEach(runComparisonTest); 70 | 71 | }); 72 | -------------------------------------------------------------------------------- /src/test/2-extension-command-inlineLet.test.ts: -------------------------------------------------------------------------------- 1 | import { inlineLet } from "../extension-commands"; 2 | import { 3 | createSelection, 4 | runComparisonTest, 5 | } from "./utils"; 6 | 7 | suite("Extension 'inlineLet' Command Tests", () => { 8 | 9 | // add comparison tests to array 10 | /* tslint:disable:object-literal-sort-keys */ 11 | [{ 12 | description: "should inline let binding from declaration (example 2)", 13 | content: `let inlineTest arg1 = 14 | let inlineMe = 1 + arg1 15 | inlineMe * 2 / (3 - inlineMe)`, 16 | expectedContent: `let inlineTest arg1 = 17 | (1 + arg1) * 2 / (3 - (1 + arg1))`, 18 | // select inlineMe binding on line 1 19 | selection: createSelection(1, 12, 1, 12), 20 | action: inlineLet, 21 | }, 22 | { 23 | description: "should inline let binding from usage (still example 2)", 24 | content: `let inlineTest arg1 = 25 | let inlineMe = 1 + arg1 26 | inlineMe * 2 / (3 - inlineMe)`, 27 | expectedContent: `let inlineTest arg1 = 28 | (1 + arg1) * 2 / (3 - (1 + arg1))`, 29 | // select last inlineMe on line 2 30 | selection: createSelection(2, 30, 2, 30), 31 | action: inlineLet, 32 | }, 33 | { 34 | description: "should inline let binding from usage with binding at beginning of context (example 3)", 35 | content: `let inlineTest arg1 = 36 | let inlineMe = 1 + arg1 37 | let dontInline = 12345 38 | inlineMe * 2 / (3 - inlineMe) + dontInline`, 39 | expectedContent: `let inlineTest arg1 = 40 | let dontInline = 12345 41 | (1 + arg1) * 2 / (3 - (1 + arg1)) + dontInline`, 42 | // select inlineMe binding on line 3 43 | selection: createSelection(3, 30, 3, 30), 44 | action: inlineLet, 45 | }, 46 | { 47 | description: "should inline let binding from declaration leaving similar binding name (example 8)", 48 | content: `let inlineTestSimilarName arg1 = 49 | let inlineMe = 1 + arg1 50 | let inlineMeWithSimilarName = inlineMe * 2 / (3 - inlineMe) 51 | inlineMeWithSimilarName`, 52 | expectedContent: `let inlineTestSimilarName arg1 = 53 | let inlineMeWithSimilarName = (1 + arg1) * 2 / (3 - (1 + arg1)) 54 | inlineMeWithSimilarName`, 55 | // select inlineMe binding on line 1 56 | selection: createSelection(1, 12, 1, 12), 57 | action: inlineLet, 58 | }, 59 | { 60 | description: "should inline let binding from usage at end of context leaving similar binding name (example 9)", 61 | content: `let inlineTestSimilarName2 arg1 = 62 | let inlineMe = 1 + arg1 63 | let inlineMeWithSimilarName = inlineMe * 2 / (3 - inlineMe) 64 | inlineMeWithSimilarName`, 65 | expectedContent: `let inlineTestSimilarName2 arg1 = 66 | let inlineMeWithSimilarName = (1 + arg1) * 2 / (3 - (1 + arg1)) 67 | inlineMeWithSimilarName`, 68 | // select inlineMe binding on line 2 69 | selection: createSelection(2, 59, 2, 67), 70 | action: inlineLet, 71 | }, 72 | { 73 | description: "should inline let binding from binding with empty line below (issue #4)", 74 | content: 75 | `test "Y" { 76 | let expectedEvents = [] 77 | 78 | equal [] expectedEvents "" 79 | }`, 80 | expectedContent: 81 | `test "Y" { 82 | 83 | equal [] [] "" 84 | }`, 85 | // select expectedEvents binding on line 1 86 | selection: createSelection(1, 10, 1, 10), 87 | action: inlineLet, 88 | }, 89 | { 90 | description: 91 | "should inline let binding from binding with empty line below & usage in another scope (example 10)", 92 | content: 93 | `let expectedEvents = [] 94 | 95 | let inlineScope initialEvents = 96 | 97 | initialEvents = expectedEvents`, 98 | expectedContent: 99 | ` 100 | let inlineScope initialEvents = 101 | 102 | initialEvents = []`, 103 | // select expectedEvents binding on line 0 104 | selection: createSelection(0, 6, 0, 6), 105 | action: inlineLet, 106 | }, 107 | { 108 | description: 109 | "should inline let binding with type annotations (issue #5)", 110 | content: 111 | `test "Y" { 112 | let expectedEvents: int list = [] 113 | 114 | equal [] expectedEvents "" 115 | }`, 116 | expectedContent: 117 | `test "Y" { 118 | 119 | equal [] [] "" 120 | }`, 121 | // select expectedEvents binding on line 0 122 | selection: createSelection(1, 8, 1, 8), 123 | action: inlineLet, 124 | }, 125 | { 126 | description: 127 | "should inline let binding from usage with type annotations (found after issue #5)", 128 | content: 129 | `test "Y" { 130 | let expectedEvents: int list = [] 131 | 132 | equal [] expectedEvents "" 133 | }`, 134 | expectedContent: 135 | `test "Y" { 136 | 137 | equal [] [] "" 138 | }`, 139 | // select expectedEvents binding on line 3 140 | selection: createSelection(3, 14, 3, 14), 141 | action: inlineLet, 142 | }] 143 | .forEach(runComparisonTest); 144 | 145 | }); 146 | -------------------------------------------------------------------------------- /src/test/index.ts: -------------------------------------------------------------------------------- 1 | // 2 | // PLEASE DO NOT MODIFY / DELETE UNLESS YOU KNOW WHAT YOU ARE DOING 3 | // 4 | // This file is providing the test runner to use when running extension tests. 5 | // By default the test runner in use is Mocha based. 6 | // 7 | // You can provide your own test runner if you want to override it by exporting 8 | // a function run(testRoot: string, clb: (error:Error) => void) that the extension 9 | // host can call to run the tests. The test runner is expected to use console.log 10 | // to report the results back to the caller. When the tests are finished, return 11 | // a possible error to the callback or null if none. 12 | 13 | import * as testRunner from "vscode/lib/testrunner"; 14 | 15 | // You can directly control Mocha options by uncommenting the following lines 16 | // See https://github.com/mochajs/mocha/wiki/Using-mocha-programmatically#set-options for more info 17 | testRunner.configure({ 18 | ui: "tdd", // the TDD UI is being used in extension.test.ts (suite, test, etc.) 19 | useColors: true, // colored output from test results 20 | }); 21 | 22 | module.exports = testRunner; 23 | -------------------------------------------------------------------------------- /src/test/utils.ts: -------------------------------------------------------------------------------- 1 | import * as assert from "assert"; 2 | import * as fs from "fs"; 3 | import * as os from "os"; 4 | import * as path from "path"; 5 | import * as vscode from "vscode"; 6 | 7 | export const preTestSetup = async (fileContent: string) => { 8 | const file = await createTestFile(fileContent); 9 | const document = await vscode.workspace.openTextDocument(file); 10 | await vscode.window.showTextDocument(document); 11 | return vscode.window.activeTextEditor; 12 | }; 13 | 14 | export function createSelection(startLine, startPos, endLine, endPos) { 15 | return new vscode.Selection(new vscode.Position(startLine, startPos), new vscode.Position(endLine, endPos)); 16 | } 17 | 18 | export function runComparisonTest({ 19 | description, 20 | content, 21 | expectedContent, 22 | selection, 23 | action, 24 | }) { 25 | test(description, async () => { 26 | const editor = await preTestSetup(content); 27 | editor.selection = selection; 28 | await action(editor); 29 | const actualText = await getAllText(editor.document); 30 | const normalisedActualText = actualText.replace("\r\n", "\n"); 31 | const normalisedExpectedContent = expectedContent.replace("\r\n", "\n"); 32 | assert.equal(normalisedActualText, normalisedExpectedContent); 33 | }); 34 | } 35 | 36 | const getAllText = async (document: vscode.TextDocument) => document.getText(new vscode.Range( 37 | new vscode.Position(0, 0), 38 | new vscode.Position(document.lineCount, Infinity)) 39 | ); 40 | 41 | const createTestFile = async (content: string) => new Promise((resolve, reject) => { 42 | const tmpFile = path.join(os.tmpdir(), randomName()); 43 | return fs.writeFile( 44 | tmpFile, 45 | content, 46 | (err) => err 47 | ? reject(err) 48 | : resolve(vscode.Uri.file(tmpFile)) 49 | ); 50 | }); 51 | 52 | const randomName = () => Math.random() 53 | .toString(36) 54 | .replace(/[^a-z]+/g, "") 55 | .substr(0, 10); 56 | -------------------------------------------------------------------------------- /src/vscode-funcs.ts: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | import * as vscode from "vscode"; 3 | import { 4 | matchBindingLine, 5 | wordIndexesInText 6 | } from "./general-funcs"; 7 | 8 | export interface ISelectionDetails { 9 | line: number; 10 | range: vscode.Range; 11 | selection: vscode.Selection; 12 | text: string; 13 | } 14 | 15 | export function getBindingDeclarationAbove(doc: vscode.TextDocument, bindingName: string, 16 | startingLine: number, indentationCharCount: number) { 17 | let currentLine: vscode.TextLine; 18 | const matchBindingLineForName = matchBindingLine(bindingName); 19 | for (let i = startingLine - 1; i >= 0; i--) { 20 | currentLine = doc.lineAt(i); 21 | const matched = matchBindingLineForName(currentLine.text); 22 | if (matched) { 23 | return { 24 | matchedLine: currentLine, 25 | matchedLineRegexMatch: matched, 26 | }; 27 | } 28 | } 29 | return null; 30 | } 31 | 32 | export function getExpandedSelection(sel: vscode.Selection[], doc: vscode.TextDocument): ISelectionDetails { 33 | const selection = sel[0]; 34 | const wordRange = doc.getWordRangeAtPosition(new vscode.Position(sel[0].start.line, sel[0].start.character)); 35 | const text = doc.getText(wordRange); 36 | return { 37 | line: wordRange.start.line, 38 | range: wordRange, 39 | selection, 40 | text, 41 | }; 42 | } 43 | 44 | export function getIndentation(doc: vscode.TextDocument, line: number): string { 45 | const matched = doc.lineAt(line).text.match(/^\s+/); 46 | return matched ? matched[0] : ""; 47 | } 48 | 49 | export function getSelectionDetails(sel: vscode.Selection[], doc: vscode.TextDocument): ISelectionDetails { 50 | const selection = sel[0]; 51 | const wordRange = new vscode.Range(selection.start, selection.end); 52 | const text = doc.getText(wordRange); 53 | return { 54 | line: wordRange.start.line, 55 | range: wordRange, 56 | selection, 57 | text, 58 | }; 59 | } 60 | 61 | export function getWordInstancesBelow(doc: vscode.TextDocument, word: string, 62 | startingLine: number, indentationCharCount: number): vscode.Position[] { 63 | const positions = []; 64 | for (let i = startingLine + 1; i < doc.lineCount; i++) { 65 | const currentLine = doc.lineAt(i); 66 | if (currentLine.firstNonWhitespaceCharacterIndex < indentationCharCount && !currentLine.isEmptyOrWhitespace) { 67 | break; 68 | } 69 | wordIndexesInText(currentLine.text, word) 70 | .map((wordIndex) => doc.getWordRangeAtPosition(new vscode.Position(i, wordIndex))) 71 | .forEach((position) => positions.push(position)); 72 | } 73 | return positions; 74 | } 75 | 76 | export function isSelectionValid(selectionDetails: ISelectionDetails) { 77 | if (!selectionDetails.range.isSingleLine) { 78 | vscode.window.showWarningMessage("Multiple line selections are not supported"); 79 | return false; 80 | } 81 | if (selectionDetails.range.isEmpty) { 82 | vscode.window.showInformationMessage("Select the expression to extract"); 83 | return false; 84 | } 85 | return true; 86 | } 87 | -------------------------------------------------------------------------------- /todos.txt: -------------------------------------------------------------------------------- 1 | //TODO: 2 | //refactor - simplify, extract, more fp 3 | //additional safeguards 4 | //add to ctrl + .? 5 | //find more suitable hotkeys 6 | //context aware - use lang service? 7 | //extension settings 8 | //features (from actual use cases): 9 | // inline 10 | // args 11 | // extract funcs 12 | // convert similar types -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "module": "commonjs", 4 | "target": "es6", 5 | "outDir": "out", 6 | "lib": [ 7 | "es6" 8 | ], 9 | "sourceMap": true, 10 | "rootDir": "src" 11 | }, 12 | "exclude": [ 13 | "node_modules", 14 | ".vscode-test" 15 | ] 16 | } -------------------------------------------------------------------------------- /tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "defaultSeverity": "error", 3 | "extends": [ 4 | "tslint:recommended" 5 | ], 6 | "jsRules": {}, 7 | "rules": { 8 | "trailing-comma": [ 9 | true, 10 | { 11 | "multiline": { 12 | "objects": "always", 13 | "arrays": "always", 14 | "functions": "never", 15 | "typeLiterals": "ignore" 16 | }, 17 | "esSpecCompliant": true 18 | } 19 | ] 20 | }, 21 | "rulesDirectory": [] 22 | } --------------------------------------------------------------------------------