├── .eslintrc.json ├── .gitignore ├── .vscode ├── extensions.json ├── launch.json ├── settings.json └── tasks.json ├── .vscodeignore ├── CHANGELOG.md ├── LICENSE ├── README.md ├── TODO ├── azure-pipelines.yml ├── image ├── command-palette.png ├── output-pane-cues.png ├── output-pane.png └── sonicpi-vscode.mp4 ├── package-lock.json ├── package.json ├── snippets └── sonic-pi-snippets.json ├── src ├── config.ts ├── extension.ts ├── main.ts ├── oscsender.ts └── test │ ├── runTest.ts │ └── suite │ ├── extension.test.ts │ └── index.ts ├── tsconfig.json └── vsc-extension-quickstart.md /.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "root": true, 3 | "parser": "@typescript-eslint/parser", 4 | "parserOptions": { 5 | "ecmaVersion": 6, 6 | "sourceType": "module" 7 | }, 8 | "plugins": [ 9 | "@typescript-eslint" 10 | ], 11 | "rules": { 12 | "@typescript-eslint/class-name-casing": "warn", 13 | "@typescript-eslint/semi": "warn", 14 | "curly": "warn", 15 | "eqeqeq": "warn", 16 | "no-throw-literal": "warn", 17 | "indent": ["error", 4, {"ignoreComments": true}], 18 | "no-mixed-spaces-and-tabs": "error", 19 | "no-trailing-spaces": ["error", {"skipBlankLines": true}], 20 | "no-unused-vars": "warn", 21 | "semi": "off" 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | out 2 | node_modules 3 | .vscode-test/ 4 | *.vsix 5 | -------------------------------------------------------------------------------- /.vscode/extensions.json: -------------------------------------------------------------------------------- 1 | { 2 | // See http://go.microsoft.com/fwlink/?LinkId=827846 3 | // for the documentation about the extensions.json format 4 | "recommendations": [ 5 | "dbaeumer.vscode-eslint" 6 | ] 7 | } 8 | -------------------------------------------------------------------------------- /.vscode/launch.json: -------------------------------------------------------------------------------- 1 | // A launch configuration that compiles the extension and then opens it inside a new window 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | { 6 | "version": "0.2.0", 7 | "configurations": [ 8 | { 9 | "name": "Run Extension", 10 | "type": "extensionHost", 11 | "request": "launch", 12 | "runtimeExecutable": "${execPath}", 13 | "args": [ 14 | "--extensionDevelopmentPath=${workspaceFolder}" 15 | ], 16 | "outFiles": [ 17 | "${workspaceFolder}/out/**/*.js" 18 | ], 19 | "preLaunchTask": "${defaultBuildTask}" 20 | }, 21 | { 22 | "name": "Extension Tests", 23 | "type": "extensionHost", 24 | "request": "launch", 25 | "runtimeExecutable": "${execPath}", 26 | "args": [ 27 | "--extensionDevelopmentPath=${workspaceFolder}", 28 | "--extensionTestsPath=${workspaceFolder}/out/test/suite/index" 29 | ], 30 | "outFiles": [ 31 | "${workspaceFolder}/out/test/**/*.js" 32 | ], 33 | "preLaunchTask": "${defaultBuildTask}" 34 | } 35 | ] 36 | } 37 | -------------------------------------------------------------------------------- /.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 | // Turn off tsc task auto detection since we have the necessary tasks as npm scripts 10 | "typescript.tsc.autoDetect": "off" 11 | } -------------------------------------------------------------------------------- /.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 | } 21 | -------------------------------------------------------------------------------- /.vscodeignore: -------------------------------------------------------------------------------- 1 | .vscode/** 2 | .vscode-test/** 3 | out/test/** 4 | src/** 5 | .gitignore 6 | vsc-extension-quickstart.md 7 | **/tsconfig.json 8 | **/.eslintrc.json 9 | **/*.map 10 | **/*.ts 11 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Change Log 2 | 3 | Just some basic functionality for now. 4 | 5 | ## [Unreleased] 6 | 7 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | // MIT License 2 | 3 | // Copyright (c) 2020 Luis Lloret 4 | 5 | // Permission is hereby granted, free of charge, to any person obtaining a copy 6 | // of this software and associated documentation files (the "Software"), to deal 7 | // in the Software without restriction, including without limitation the rights 8 | // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | // copies of the Software, and to permit persons to whom the Software is 10 | // furnished to do so, subject to the following conditions: 11 | 12 | // The above copyright notice and this permission notice shall be included in all 13 | // copies or substantial portions of the Software. 14 | 15 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | // SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | [![Build Status](https://dev.azure.com/luislloret/sonic-pi-vscode-editor/_apis/build/status/llloret.sonic-pi-vscode-editor?branchName=master)](https://dev.azure.com/luislloret/sonic-pi-vscode-editor/_build/latest?definitionId=1&branchName=master) 2 | 3 | # sonicpieditor README 4 | 5 | This is an extension to work with Sonic Pi within vscode. It will launch Sonic Pi's backend when you open 6 | a ruby file. 7 | 8 | At the moment this is run as an extension in development, so see the Requirements section below for instructions 9 | on how to run it. 10 | 11 | **Please feel free to contribute with your Pull Requests. Any help is welcome!** 12 | 13 | Also, if you like this project or are interested in its progress, it would be great if you star it in github to help spread the word! Thank you! 14 | 15 | ## Features 16 | 17 | This is just starting, but we already have enough features to have some fun! 18 | - Configurable Sonic Pi server launch options. Now you can choose between: 19 | - always: launches the server when vscode starts 20 | - ruby: launches the server when there is a ruby file visible in vscode (this is the default) 21 | - custom: launches the server when there is a file with your given custom extension visible in vscode 22 | - never: do not launch the server automatically (use the Sonic Pi: Start Server command) 23 | - Configurable Sonic Pi root path, in case the default does not work for you 24 | - Can run code pressing Alt-R (or Cmd-R on Mac, just like in Sonic Pi's editor) or with command palette "Sonic Pi: Run" (see [Screenshot](image/command-palette.png)) 25 | - Can stop running audio with Alt-S (or Cmd-S on Mac) or "Sonic Pi: Stop" 26 | - Can run the selected code with Alt-T (or Cmd-T on Mac). If there is no code selected, it will offer to run the whole file instead (and persist the choice) 27 | - Shows logs and cues in the output panel (see [logs](image/output-pane.png) and [cues](image/output-pane-cues.png)) 28 | - Some snippets like live_loop, effects, synths, samples, and possibly more coming soon. See the snippets directory for the full list and contribute new ones if you feel like it! 29 | - fx -> instantiate effect with autocomplete list chooser 30 | - fx x (where x is a letter) -> effect instantiation 31 | - us -> instantiate synth with autocomplete list chooser 32 | - us x (where x is a letter) -> specific synth instantiation 33 | - sa -> instantiate sample with autocomplete list chooser 34 | - Highlight errors as reported by the Sonic Pi server 35 | - And of course, you have syntax highlighting, autoformatting, all the goodies that you usually have with vscode! 36 | 37 | * See a very short video of Robin Newman's arrangement of "Pase El Agua" launched from this extension, showing 38 | the thing working, logs, etc: [Video](image/sonicpi-vscode.mp4) 39 | 40 | 41 | (You can find Robin's original work here: https://in-thread.sonic-pi.net/t/three-more-pieces-for-sonic-pi/2434). 42 | 43 | ## Requirements 44 | 45 | The extension runs in development mode. Follow these steps: 46 | - Go to the extension directory (where this file is located) 47 | - run "npm install", to install the necessary node dependencies 48 | - run "code .", to open the extension directory in vscode 49 | - press F5 to run the extension 50 | - (optional) see the Sonic Settings in vscode and configure how you want to start the server, by default it will launch when there is a ruby file visible in the editor 51 | 52 | If you run into problems, let me know, and I'll do my best to help you set this up. 53 | 54 | **Make sure you configure the Sonic Pi root path in the configuration if the default setting does not work for you** 55 | Open Settings -> Extensions -> Sonic Pi -> Sonic Pi Root Directory 56 | 57 | 58 | ## Known Issues 59 | 60 | I have not tested this in Linux yet. Works nicely in Windows and Mac. 61 | 62 | 63 | ## Open questions 64 | 65 | -------------------------------------------------------------------------------- /TODO: -------------------------------------------------------------------------------- 1 | A list of things that would be nice to have 2 | - Better diagnostics in case the server does not start 3 | - Sonic Pi help 4 | - Colorize output panel (log and cues), if possible 5 | - Syntax highlight synths and FX 6 | - I18n 7 | 8 | -------------------------------------------------------------------------------- /azure-pipelines.yml: -------------------------------------------------------------------------------- 1 | # Node.js 2 | # Build a general Node.js project with npm. 3 | # Add steps that analyze code, save build artifacts, deploy, and more: 4 | # https://docs.microsoft.com/azure/devops/pipelines/languages/javascript 5 | 6 | # For now trigger on everything 7 | #trigger: 8 | #- master 9 | 10 | strategy: 11 | matrix: 12 | linux: 13 | imageName: 'ubuntu-18.04' 14 | mac: 15 | imageName: 'macos-10.14' 16 | windows: 17 | imageName: 'windows-2019' 18 | 19 | pool: 20 | vmImage: $(imageName) 21 | 22 | 23 | steps: 24 | - task: NodeTool@0 25 | inputs: 26 | versionSpec: '10.x' 27 | displayName: 'Install Node.js' 28 | 29 | - script: | 30 | npm install 31 | npm run compile 32 | npm run lint 33 | displayName: 'npm install and build' -------------------------------------------------------------------------------- /image/command-palette.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/llloret/sonic-pi-vscode-editor/10d343abb3632a56c34d5bb2a84752efae7bc7bd/image/command-palette.png -------------------------------------------------------------------------------- /image/output-pane-cues.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/llloret/sonic-pi-vscode-editor/10d343abb3632a56c34d5bb2a84752efae7bc7bd/image/output-pane-cues.png -------------------------------------------------------------------------------- /image/output-pane.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/llloret/sonic-pi-vscode-editor/10d343abb3632a56c34d5bb2a84752efae7bc7bd/image/output-pane.png -------------------------------------------------------------------------------- /image/sonicpi-vscode.mp4: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/llloret/sonic-pi-vscode-editor/10d343abb3632a56c34d5bb2a84752efae7bc7bd/image/sonicpi-vscode.mp4 -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "sonicpieditor", 3 | "version": "0.0.1", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@babel/code-frame": { 8 | "version": "7.8.3", 9 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.8.3.tgz", 10 | "integrity": "sha512-a9gxpmdXtZEInkCSHUJDLHZVBgb1QS0jhss4cPP93EW7s+uC5bikET2twEF3KV+7rDblJcmNvTR7VJejqd2C2g==", 11 | "dev": true, 12 | "requires": { 13 | "@babel/highlight": "^7.8.3" 14 | } 15 | }, 16 | "@babel/helper-validator-identifier": { 17 | "version": "7.9.5", 18 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.9.5.tgz", 19 | "integrity": "sha512-/8arLKUFq882w4tWGj9JYzRpAlZgiWUJ+dtteNTDqrRBz9Iguck9Rn3ykuBDoUwh2TO4tSAJlrxDUOXWklJe4g==", 20 | "dev": true 21 | }, 22 | "@babel/highlight": { 23 | "version": "7.9.0", 24 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.9.0.tgz", 25 | "integrity": "sha512-lJZPilxX7Op3Nv/2cvFdnlepPXDxi29wxteT57Q965oc5R9v86ztx0jfxVrTcBk8C2kcPkkDa2Z4T3ZsPPVWsQ==", 26 | "dev": true, 27 | "requires": { 28 | "@babel/helper-validator-identifier": "^7.9.0", 29 | "chalk": "^2.0.0", 30 | "js-tokens": "^4.0.0" 31 | } 32 | }, 33 | "@types/color-name": { 34 | "version": "1.1.1", 35 | "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz", 36 | "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==", 37 | "dev": true 38 | }, 39 | "@types/eslint-visitor-keys": { 40 | "version": "1.0.0", 41 | "resolved": "https://registry.npmjs.org/@types/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz", 42 | "integrity": "sha512-OCutwjDZ4aFS6PB1UZ988C4YgwlBHJd6wCeQqaLdmadZ/7e+w79+hbMUFC1QXDNCmdyoRfAFdm0RypzwR+Qpag==", 43 | "dev": true 44 | }, 45 | "@types/events": { 46 | "version": "3.0.0", 47 | "resolved": "https://registry.npmjs.org/@types/events/-/events-3.0.0.tgz", 48 | "integrity": "sha512-EaObqwIvayI5a8dCzhFrjKzVwKLxjoG9T6Ppd5CEo07LRKfQ8Yokw54r5+Wq7FaBQ+yXRvQAYPrHwya1/UFt9g==", 49 | "dev": true 50 | }, 51 | "@types/glob": { 52 | "version": "7.1.1", 53 | "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.1.1.tgz", 54 | "integrity": "sha512-1Bh06cbWJUHMC97acuD6UMG29nMt0Aqz1vF3guLfG+kHHJhy3AyohZFFxYk2f7Q1SQIrNwvncxAE0N/9s70F2w==", 55 | "dev": true, 56 | "requires": { 57 | "@types/events": "*", 58 | "@types/minimatch": "*", 59 | "@types/node": "*" 60 | } 61 | }, 62 | "@types/json-schema": { 63 | "version": "7.0.4", 64 | "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.4.tgz", 65 | "integrity": "sha512-8+KAKzEvSUdeo+kmqnKrqgeE+LcA0tjYWFY7RPProVYwnqDjukzO+3b6dLD56rYX5TdWejnEOLJYOIeh4CXKuA==", 66 | "dev": true 67 | }, 68 | "@types/minimatch": { 69 | "version": "3.0.3", 70 | "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz", 71 | "integrity": "sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA==", 72 | "dev": true 73 | }, 74 | "@types/mocha": { 75 | "version": "7.0.2", 76 | "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-7.0.2.tgz", 77 | "integrity": "sha512-ZvO2tAcjmMi8V/5Z3JsyofMe3hasRcaw88cto5etSVMwVQfeivGAlEYmaQgceUSVYFofVjT+ioHsATjdWcFt1w==", 78 | "dev": true 79 | }, 80 | "@types/node": { 81 | "version": "13.13.9", 82 | "resolved": "https://registry.npmjs.org/@types/node/-/node-13.13.9.tgz", 83 | "integrity": "sha512-EPZBIGed5gNnfWCiwEIwTE2Jdg4813odnG8iNPMQGrqVxrI+wL68SPtPeCX+ZxGBaA6pKAVc6jaKgP/Q0QzfdQ==", 84 | "dev": true 85 | }, 86 | "@types/vscode": { 87 | "version": "1.45.1", 88 | "resolved": "https://registry.npmjs.org/@types/vscode/-/vscode-1.45.1.tgz", 89 | "integrity": "sha512-0NO9qrrEJBO8FsqHCrFMgR2suKnwCsKBWvRSb2OzH5gs4i3QO5AhEMQYrSzDbU/wLPt7N617/rN9lPY213gmwg==", 90 | "dev": true 91 | }, 92 | "@typescript-eslint/eslint-plugin": { 93 | "version": "2.34.0", 94 | "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-2.34.0.tgz", 95 | "integrity": "sha512-4zY3Z88rEE99+CNvTbXSyovv2z9PNOVffTWD2W8QF5s2prBQtwN2zadqERcrHpcR7O/+KMI3fcTAmUUhK/iQcQ==", 96 | "dev": true, 97 | "requires": { 98 | "@typescript-eslint/experimental-utils": "2.34.0", 99 | "functional-red-black-tree": "^1.0.1", 100 | "regexpp": "^3.0.0", 101 | "tsutils": "^3.17.1" 102 | } 103 | }, 104 | "@typescript-eslint/experimental-utils": { 105 | "version": "2.34.0", 106 | "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-2.34.0.tgz", 107 | "integrity": "sha512-eS6FTkq+wuMJ+sgtuNTtcqavWXqsflWcfBnlYhg/nS4aZ1leewkXGbvBhaapn1q6qf4M71bsR1tez5JTRMuqwA==", 108 | "dev": true, 109 | "requires": { 110 | "@types/json-schema": "^7.0.3", 111 | "@typescript-eslint/typescript-estree": "2.34.0", 112 | "eslint-scope": "^5.0.0", 113 | "eslint-utils": "^2.0.0" 114 | } 115 | }, 116 | "@typescript-eslint/parser": { 117 | "version": "2.34.0", 118 | "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-2.34.0.tgz", 119 | "integrity": "sha512-03ilO0ucSD0EPTw2X4PntSIRFtDPWjrVq7C3/Z3VQHRC7+13YB55rcJI3Jt+YgeHbjUdJPcPa7b23rXCBokuyA==", 120 | "dev": true, 121 | "requires": { 122 | "@types/eslint-visitor-keys": "^1.0.0", 123 | "@typescript-eslint/experimental-utils": "2.34.0", 124 | "@typescript-eslint/typescript-estree": "2.34.0", 125 | "eslint-visitor-keys": "^1.1.0" 126 | } 127 | }, 128 | "@typescript-eslint/typescript-estree": { 129 | "version": "2.34.0", 130 | "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-2.34.0.tgz", 131 | "integrity": "sha512-OMAr+nJWKdlVM9LOqCqh3pQQPwxHAN7Du8DR6dmwCrAmxtiXQnhHJ6tBNtf+cggqfo51SG/FCwnKhXCIM7hnVg==", 132 | "dev": true, 133 | "requires": { 134 | "debug": "^4.1.1", 135 | "eslint-visitor-keys": "^1.1.0", 136 | "glob": "^7.1.6", 137 | "is-glob": "^4.0.1", 138 | "lodash": "^4.17.15", 139 | "semver": "^7.3.2", 140 | "tsutils": "^3.17.1" 141 | } 142 | }, 143 | "acorn": { 144 | "version": "7.2.0", 145 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.2.0.tgz", 146 | "integrity": "sha512-apwXVmYVpQ34m/i71vrApRrRKCWQnZZF1+npOD0WV5xZFfwWOmKGQ2RWlfdy9vWITsenisM8M0Qeq8agcFHNiQ==", 147 | "dev": true 148 | }, 149 | "acorn-jsx": { 150 | "version": "5.2.0", 151 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.2.0.tgz", 152 | "integrity": "sha512-HiUX/+K2YpkpJ+SzBffkM/AQ2YE03S0U1kjTLVpoJdhZMOWy8qvXVN9JdLqv2QsaQ6MPYQIuNmwD8zOiYUofLQ==", 153 | "dev": true 154 | }, 155 | "agent-base": { 156 | "version": "4.3.0", 157 | "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.3.0.tgz", 158 | "integrity": "sha512-salcGninV0nPrwpGNn4VTXBb1SOuXQBiqbrNXoeizJsHrsL6ERFM2Ne3JUSBWRE6aeNJI2ROP/WEEIDUiDe3cg==", 159 | "dev": true, 160 | "requires": { 161 | "es6-promisify": "^5.0.0" 162 | } 163 | }, 164 | "ajv": { 165 | "version": "6.12.2", 166 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.2.tgz", 167 | "integrity": "sha512-k+V+hzjm5q/Mr8ef/1Y9goCmlsK4I6Sm74teeyGvFk1XrOsbsKLjEdrvny42CZ+a8sXbk8KWpY/bDwS+FLL2UQ==", 168 | "dev": true, 169 | "requires": { 170 | "fast-deep-equal": "^3.1.1", 171 | "fast-json-stable-stringify": "^2.0.0", 172 | "json-schema-traverse": "^0.4.1", 173 | "uri-js": "^4.2.2" 174 | } 175 | }, 176 | "ansi-colors": { 177 | "version": "3.2.3", 178 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.3.tgz", 179 | "integrity": "sha512-LEHHyuhlPY3TmuUYMh2oz89lTShfvgbmzaBcxve9t/9Wuy7Dwf4yoAKcND7KFT1HAQfqZ12qtc+DUrBMeKF9nw==", 180 | "dev": true 181 | }, 182 | "ansi-escapes": { 183 | "version": "4.3.1", 184 | "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.1.tgz", 185 | "integrity": "sha512-JWF7ocqNrp8u9oqpgV+wH5ftbt+cfvv+PTjOvKLT3AdYly/LmORARfEVT1iyjwN+4MqE5UmVKoAdIBqeoCHgLA==", 186 | "dev": true, 187 | "requires": { 188 | "type-fest": "^0.11.0" 189 | }, 190 | "dependencies": { 191 | "type-fest": { 192 | "version": "0.11.0", 193 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.11.0.tgz", 194 | "integrity": "sha512-OdjXJxnCN1AvyLSzeKIgXTXxV+99ZuXl3Hpo9XpJAv9MBcHrrJOQ5kV7ypXOuQie+AmWG25hLbiKdwYTifzcfQ==", 195 | "dev": true 196 | } 197 | } 198 | }, 199 | "ansi-regex": { 200 | "version": "5.0.0", 201 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", 202 | "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", 203 | "dev": true 204 | }, 205 | "ansi-styles": { 206 | "version": "3.2.1", 207 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 208 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 209 | "dev": true, 210 | "requires": { 211 | "color-convert": "^1.9.0" 212 | } 213 | }, 214 | "anymatch": { 215 | "version": "3.1.1", 216 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", 217 | "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", 218 | "dev": true, 219 | "requires": { 220 | "normalize-path": "^3.0.0", 221 | "picomatch": "^2.0.4" 222 | } 223 | }, 224 | "argparse": { 225 | "version": "1.0.10", 226 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 227 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 228 | "dev": true, 229 | "requires": { 230 | "sprintf-js": "~1.0.2" 231 | } 232 | }, 233 | "astral-regex": { 234 | "version": "1.0.0", 235 | "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", 236 | "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", 237 | "dev": true 238 | }, 239 | "async-limiter": { 240 | "version": "1.0.1", 241 | "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", 242 | "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==" 243 | }, 244 | "balanced-match": { 245 | "version": "1.0.0", 246 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 247 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 248 | "dev": true 249 | }, 250 | "binary-extensions": { 251 | "version": "2.0.0", 252 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz", 253 | "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==", 254 | "dev": true 255 | }, 256 | "brace-expansion": { 257 | "version": "1.1.11", 258 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 259 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 260 | "dev": true, 261 | "requires": { 262 | "balanced-match": "^1.0.0", 263 | "concat-map": "0.0.1" 264 | } 265 | }, 266 | "braces": { 267 | "version": "3.0.2", 268 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 269 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 270 | "dev": true, 271 | "requires": { 272 | "fill-range": "^7.0.1" 273 | } 274 | }, 275 | "browser-stdout": { 276 | "version": "1.3.1", 277 | "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", 278 | "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", 279 | "dev": true 280 | }, 281 | "callsites": { 282 | "version": "3.1.0", 283 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", 284 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", 285 | "dev": true 286 | }, 287 | "camelcase": { 288 | "version": "5.3.1", 289 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 290 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 291 | "dev": true 292 | }, 293 | "chalk": { 294 | "version": "2.4.2", 295 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 296 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 297 | "dev": true, 298 | "requires": { 299 | "ansi-styles": "^3.2.1", 300 | "escape-string-regexp": "^1.0.5", 301 | "supports-color": "^5.3.0" 302 | } 303 | }, 304 | "chardet": { 305 | "version": "0.7.0", 306 | "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", 307 | "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", 308 | "dev": true 309 | }, 310 | "chokidar": { 311 | "version": "3.3.0", 312 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.3.0.tgz", 313 | "integrity": "sha512-dGmKLDdT3Gdl7fBUe8XK+gAtGmzy5Fn0XkkWQuYxGIgWVPPse2CxFA5mtrlD0TOHaHjEUqkWNyP1XdHoJES/4A==", 314 | "dev": true, 315 | "requires": { 316 | "anymatch": "~3.1.1", 317 | "braces": "~3.0.2", 318 | "fsevents": "~2.1.1", 319 | "glob-parent": "~5.1.0", 320 | "is-binary-path": "~2.1.0", 321 | "is-glob": "~4.0.1", 322 | "normalize-path": "~3.0.0", 323 | "readdirp": "~3.2.0" 324 | } 325 | }, 326 | "cli-cursor": { 327 | "version": "3.1.0", 328 | "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", 329 | "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", 330 | "dev": true, 331 | "requires": { 332 | "restore-cursor": "^3.1.0" 333 | } 334 | }, 335 | "cli-width": { 336 | "version": "2.2.1", 337 | "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz", 338 | "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", 339 | "dev": true 340 | }, 341 | "cliui": { 342 | "version": "5.0.0", 343 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", 344 | "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", 345 | "dev": true, 346 | "requires": { 347 | "string-width": "^3.1.0", 348 | "strip-ansi": "^5.2.0", 349 | "wrap-ansi": "^5.1.0" 350 | }, 351 | "dependencies": { 352 | "emoji-regex": { 353 | "version": "7.0.3", 354 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 355 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 356 | "dev": true 357 | }, 358 | "is-fullwidth-code-point": { 359 | "version": "2.0.0", 360 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 361 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 362 | "dev": true 363 | }, 364 | "string-width": { 365 | "version": "3.1.0", 366 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 367 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 368 | "dev": true, 369 | "requires": { 370 | "emoji-regex": "^7.0.1", 371 | "is-fullwidth-code-point": "^2.0.0", 372 | "strip-ansi": "^5.1.0" 373 | } 374 | } 375 | } 376 | }, 377 | "color-convert": { 378 | "version": "1.9.3", 379 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 380 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 381 | "dev": true, 382 | "requires": { 383 | "color-name": "1.1.3" 384 | } 385 | }, 386 | "color-name": { 387 | "version": "1.1.3", 388 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 389 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 390 | "dev": true 391 | }, 392 | "concat-map": { 393 | "version": "0.0.1", 394 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 395 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 396 | "dev": true 397 | }, 398 | "cross-spawn": { 399 | "version": "6.0.5", 400 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", 401 | "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", 402 | "dev": true, 403 | "requires": { 404 | "nice-try": "^1.0.4", 405 | "path-key": "^2.0.1", 406 | "semver": "^5.5.0", 407 | "shebang-command": "^1.2.0", 408 | "which": "^1.2.9" 409 | }, 410 | "dependencies": { 411 | "semver": { 412 | "version": "5.7.1", 413 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 414 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 415 | "dev": true 416 | } 417 | } 418 | }, 419 | "debug": { 420 | "version": "4.1.1", 421 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", 422 | "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", 423 | "dev": true, 424 | "requires": { 425 | "ms": "^2.1.1" 426 | } 427 | }, 428 | "decamelize": { 429 | "version": "1.2.0", 430 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 431 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", 432 | "dev": true 433 | }, 434 | "deep-is": { 435 | "version": "0.1.3", 436 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", 437 | "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", 438 | "dev": true 439 | }, 440 | "define-properties": { 441 | "version": "1.1.3", 442 | "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", 443 | "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", 444 | "dev": true, 445 | "requires": { 446 | "object-keys": "^1.0.12" 447 | } 448 | }, 449 | "diff": { 450 | "version": "3.5.0", 451 | "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", 452 | "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", 453 | "dev": true 454 | }, 455 | "doctrine": { 456 | "version": "3.0.0", 457 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", 458 | "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", 459 | "dev": true, 460 | "requires": { 461 | "esutils": "^2.0.2" 462 | } 463 | }, 464 | "emoji-regex": { 465 | "version": "8.0.0", 466 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 467 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 468 | "dev": true 469 | }, 470 | "es-abstract": { 471 | "version": "1.17.5", 472 | "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.5.tgz", 473 | "integrity": "sha512-BR9auzDbySxOcfog0tLECW8l28eRGpDpU3Dm3Hp4q/N+VtLTmyj4EUN088XZWQDW/hzj6sYRDXeOFsaAODKvpg==", 474 | "dev": true, 475 | "requires": { 476 | "es-to-primitive": "^1.2.1", 477 | "function-bind": "^1.1.1", 478 | "has": "^1.0.3", 479 | "has-symbols": "^1.0.1", 480 | "is-callable": "^1.1.5", 481 | "is-regex": "^1.0.5", 482 | "object-inspect": "^1.7.0", 483 | "object-keys": "^1.1.1", 484 | "object.assign": "^4.1.0", 485 | "string.prototype.trimleft": "^2.1.1", 486 | "string.prototype.trimright": "^2.1.1" 487 | } 488 | }, 489 | "es-to-primitive": { 490 | "version": "1.2.1", 491 | "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", 492 | "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", 493 | "dev": true, 494 | "requires": { 495 | "is-callable": "^1.1.4", 496 | "is-date-object": "^1.0.1", 497 | "is-symbol": "^1.0.2" 498 | } 499 | }, 500 | "es6-promise": { 501 | "version": "4.2.8", 502 | "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", 503 | "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==", 504 | "dev": true 505 | }, 506 | "es6-promisify": { 507 | "version": "5.0.0", 508 | "resolved": "https://registry.npmjs.org/es6-promisify/-/es6-promisify-5.0.0.tgz", 509 | "integrity": "sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM=", 510 | "dev": true, 511 | "requires": { 512 | "es6-promise": "^4.0.3" 513 | } 514 | }, 515 | "escape-string-regexp": { 516 | "version": "1.0.5", 517 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 518 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 519 | "dev": true 520 | }, 521 | "eslint": { 522 | "version": "6.8.0", 523 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-6.8.0.tgz", 524 | "integrity": "sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==", 525 | "dev": true, 526 | "requires": { 527 | "@babel/code-frame": "^7.0.0", 528 | "ajv": "^6.10.0", 529 | "chalk": "^2.1.0", 530 | "cross-spawn": "^6.0.5", 531 | "debug": "^4.0.1", 532 | "doctrine": "^3.0.0", 533 | "eslint-scope": "^5.0.0", 534 | "eslint-utils": "^1.4.3", 535 | "eslint-visitor-keys": "^1.1.0", 536 | "espree": "^6.1.2", 537 | "esquery": "^1.0.1", 538 | "esutils": "^2.0.2", 539 | "file-entry-cache": "^5.0.1", 540 | "functional-red-black-tree": "^1.0.1", 541 | "glob-parent": "^5.0.0", 542 | "globals": "^12.1.0", 543 | "ignore": "^4.0.6", 544 | "import-fresh": "^3.0.0", 545 | "imurmurhash": "^0.1.4", 546 | "inquirer": "^7.0.0", 547 | "is-glob": "^4.0.0", 548 | "js-yaml": "^3.13.1", 549 | "json-stable-stringify-without-jsonify": "^1.0.1", 550 | "levn": "^0.3.0", 551 | "lodash": "^4.17.14", 552 | "minimatch": "^3.0.4", 553 | "mkdirp": "^0.5.1", 554 | "natural-compare": "^1.4.0", 555 | "optionator": "^0.8.3", 556 | "progress": "^2.0.0", 557 | "regexpp": "^2.0.1", 558 | "semver": "^6.1.2", 559 | "strip-ansi": "^5.2.0", 560 | "strip-json-comments": "^3.0.1", 561 | "table": "^5.2.3", 562 | "text-table": "^0.2.0", 563 | "v8-compile-cache": "^2.0.3" 564 | }, 565 | "dependencies": { 566 | "eslint-utils": { 567 | "version": "1.4.3", 568 | "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz", 569 | "integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==", 570 | "dev": true, 571 | "requires": { 572 | "eslint-visitor-keys": "^1.1.0" 573 | } 574 | }, 575 | "regexpp": { 576 | "version": "2.0.1", 577 | "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz", 578 | "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", 579 | "dev": true 580 | }, 581 | "semver": { 582 | "version": "6.3.0", 583 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 584 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 585 | "dev": true 586 | } 587 | } 588 | }, 589 | "eslint-scope": { 590 | "version": "5.0.0", 591 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.0.0.tgz", 592 | "integrity": "sha512-oYrhJW7S0bxAFDvWqzvMPRm6pcgcnWc4QnofCAqRTRfQC0JcwenzGglTtsLyIuuWFfkqDG9vz67cnttSd53djw==", 593 | "dev": true, 594 | "requires": { 595 | "esrecurse": "^4.1.0", 596 | "estraverse": "^4.1.1" 597 | } 598 | }, 599 | "eslint-utils": { 600 | "version": "2.0.0", 601 | "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.0.0.tgz", 602 | "integrity": "sha512-0HCPuJv+7Wv1bACm8y5/ECVfYdfsAm9xmVb7saeFlxjPYALefjhbYoCkBjPdPzGH8wWyTpAez82Fh3VKYEZ8OA==", 603 | "dev": true, 604 | "requires": { 605 | "eslint-visitor-keys": "^1.1.0" 606 | } 607 | }, 608 | "eslint-visitor-keys": { 609 | "version": "1.1.0", 610 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.1.0.tgz", 611 | "integrity": "sha512-8y9YjtM1JBJU/A9Kc+SbaOV4y29sSWckBwMHa+FGtVj5gN/sbnKDf6xJUl+8g7FAij9LVaP8C24DUiH/f/2Z9A==", 612 | "dev": true 613 | }, 614 | "espree": { 615 | "version": "6.2.1", 616 | "resolved": "https://registry.npmjs.org/espree/-/espree-6.2.1.tgz", 617 | "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", 618 | "dev": true, 619 | "requires": { 620 | "acorn": "^7.1.1", 621 | "acorn-jsx": "^5.2.0", 622 | "eslint-visitor-keys": "^1.1.0" 623 | } 624 | }, 625 | "esprima": { 626 | "version": "4.0.1", 627 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 628 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 629 | "dev": true 630 | }, 631 | "esquery": { 632 | "version": "1.3.1", 633 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.3.1.tgz", 634 | "integrity": "sha512-olpvt9QG0vniUBZspVRN6lwB7hOZoTRtT+jzR+tS4ffYx2mzbw+z0XCOk44aaLYKApNX5nMm+E+P6o25ip/DHQ==", 635 | "dev": true, 636 | "requires": { 637 | "estraverse": "^5.1.0" 638 | }, 639 | "dependencies": { 640 | "estraverse": { 641 | "version": "5.1.0", 642 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.1.0.tgz", 643 | "integrity": "sha512-FyohXK+R0vE+y1nHLoBM7ZTyqRpqAlhdZHCWIWEviFLiGB8b04H6bQs8G+XTthacvT8VuwvteiP7RJSxMs8UEw==", 644 | "dev": true 645 | } 646 | } 647 | }, 648 | "esrecurse": { 649 | "version": "4.2.1", 650 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.1.tgz", 651 | "integrity": "sha512-64RBB++fIOAXPw3P9cy89qfMlvZEXZkqqJkjqqXIvzP5ezRZjW+lPWjw35UX/3EhUPFYbg5ER4JYgDw4007/DQ==", 652 | "dev": true, 653 | "requires": { 654 | "estraverse": "^4.1.0" 655 | } 656 | }, 657 | "estraverse": { 658 | "version": "4.3.0", 659 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", 660 | "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", 661 | "dev": true 662 | }, 663 | "esutils": { 664 | "version": "2.0.3", 665 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 666 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 667 | "dev": true 668 | }, 669 | "external-editor": { 670 | "version": "3.1.0", 671 | "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", 672 | "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", 673 | "dev": true, 674 | "requires": { 675 | "chardet": "^0.7.0", 676 | "iconv-lite": "^0.4.24", 677 | "tmp": "^0.0.33" 678 | } 679 | }, 680 | "fast-deep-equal": { 681 | "version": "3.1.1", 682 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.1.tgz", 683 | "integrity": "sha512-8UEa58QDLauDNfpbrX55Q9jrGHThw2ZMdOky5Gl1CDtVeJDPVrG4Jxx1N8jw2gkWaff5UUuX1KJd+9zGe2B+ZA==", 684 | "dev": true 685 | }, 686 | "fast-json-stable-stringify": { 687 | "version": "2.1.0", 688 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", 689 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", 690 | "dev": true 691 | }, 692 | "fast-levenshtein": { 693 | "version": "2.0.6", 694 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 695 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", 696 | "dev": true 697 | }, 698 | "figures": { 699 | "version": "3.2.0", 700 | "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", 701 | "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", 702 | "dev": true, 703 | "requires": { 704 | "escape-string-regexp": "^1.0.5" 705 | } 706 | }, 707 | "file-entry-cache": { 708 | "version": "5.0.1", 709 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", 710 | "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", 711 | "dev": true, 712 | "requires": { 713 | "flat-cache": "^2.0.1" 714 | } 715 | }, 716 | "fill-range": { 717 | "version": "7.0.1", 718 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 719 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 720 | "dev": true, 721 | "requires": { 722 | "to-regex-range": "^5.0.1" 723 | } 724 | }, 725 | "find-up": { 726 | "version": "3.0.0", 727 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", 728 | "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", 729 | "dev": true, 730 | "requires": { 731 | "locate-path": "^3.0.0" 732 | } 733 | }, 734 | "flat": { 735 | "version": "4.1.0", 736 | "resolved": "https://registry.npmjs.org/flat/-/flat-4.1.0.tgz", 737 | "integrity": "sha512-Px/TiLIznH7gEDlPXcUD4KnBusa6kR6ayRUVcnEAbreRIuhkqow/mun59BuRXwoYk7ZQOLW1ZM05ilIvK38hFw==", 738 | "dev": true, 739 | "requires": { 740 | "is-buffer": "~2.0.3" 741 | } 742 | }, 743 | "flat-cache": { 744 | "version": "2.0.1", 745 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", 746 | "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", 747 | "dev": true, 748 | "requires": { 749 | "flatted": "^2.0.0", 750 | "rimraf": "2.6.3", 751 | "write": "1.0.3" 752 | } 753 | }, 754 | "flatted": { 755 | "version": "2.0.2", 756 | "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", 757 | "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", 758 | "dev": true 759 | }, 760 | "fs.realpath": { 761 | "version": "1.0.0", 762 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 763 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 764 | "dev": true 765 | }, 766 | "fsevents": { 767 | "version": "2.1.3", 768 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", 769 | "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", 770 | "dev": true, 771 | "optional": true 772 | }, 773 | "function-bind": { 774 | "version": "1.1.1", 775 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 776 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 777 | "dev": true 778 | }, 779 | "functional-red-black-tree": { 780 | "version": "1.0.1", 781 | "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", 782 | "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", 783 | "dev": true 784 | }, 785 | "get-caller-file": { 786 | "version": "2.0.5", 787 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 788 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 789 | "dev": true 790 | }, 791 | "glob": { 792 | "version": "7.1.6", 793 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", 794 | "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", 795 | "dev": true, 796 | "requires": { 797 | "fs.realpath": "^1.0.0", 798 | "inflight": "^1.0.4", 799 | "inherits": "2", 800 | "minimatch": "^3.0.4", 801 | "once": "^1.3.0", 802 | "path-is-absolute": "^1.0.0" 803 | } 804 | }, 805 | "glob-parent": { 806 | "version": "5.1.2", 807 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 808 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 809 | "dev": true, 810 | "requires": { 811 | "is-glob": "^4.0.1" 812 | } 813 | }, 814 | "globals": { 815 | "version": "12.4.0", 816 | "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", 817 | "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", 818 | "dev": true, 819 | "requires": { 820 | "type-fest": "^0.8.1" 821 | } 822 | }, 823 | "growl": { 824 | "version": "1.10.5", 825 | "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz", 826 | "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==", 827 | "dev": true 828 | }, 829 | "has": { 830 | "version": "1.0.3", 831 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 832 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 833 | "dev": true, 834 | "requires": { 835 | "function-bind": "^1.1.1" 836 | } 837 | }, 838 | "has-flag": { 839 | "version": "3.0.0", 840 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 841 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 842 | "dev": true 843 | }, 844 | "has-symbols": { 845 | "version": "1.0.1", 846 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", 847 | "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", 848 | "dev": true 849 | }, 850 | "he": { 851 | "version": "1.2.0", 852 | "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", 853 | "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", 854 | "dev": true 855 | }, 856 | "http-proxy-agent": { 857 | "version": "2.1.0", 858 | "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-2.1.0.tgz", 859 | "integrity": "sha512-qwHbBLV7WviBl0rQsOzH6o5lwyOIvwp/BdFnvVxXORldu5TmjFfjzBcWUWS5kWAZhmv+JtiDhSuQCp4sBfbIgg==", 860 | "dev": true, 861 | "requires": { 862 | "agent-base": "4", 863 | "debug": "3.1.0" 864 | }, 865 | "dependencies": { 866 | "debug": { 867 | "version": "3.1.0", 868 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", 869 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", 870 | "dev": true, 871 | "requires": { 872 | "ms": "2.0.0" 873 | } 874 | }, 875 | "ms": { 876 | "version": "2.0.0", 877 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 878 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 879 | "dev": true 880 | } 881 | } 882 | }, 883 | "https-proxy-agent": { 884 | "version": "2.2.4", 885 | "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.4.tgz", 886 | "integrity": "sha512-OmvfoQ53WLjtA9HeYP9RNrWMJzzAz1JGaSFr1nijg0PVR1JaD/xbJq1mdEIIlxGpXp9eSe/O2LgU9DJmTPd0Eg==", 887 | "dev": true, 888 | "requires": { 889 | "agent-base": "^4.3.0", 890 | "debug": "^3.1.0" 891 | }, 892 | "dependencies": { 893 | "debug": { 894 | "version": "3.2.6", 895 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 896 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 897 | "dev": true, 898 | "requires": { 899 | "ms": "^2.1.1" 900 | } 901 | } 902 | } 903 | }, 904 | "iconv-lite": { 905 | "version": "0.4.24", 906 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 907 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 908 | "dev": true, 909 | "requires": { 910 | "safer-buffer": ">= 2.1.2 < 3" 911 | } 912 | }, 913 | "ignore": { 914 | "version": "4.0.6", 915 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", 916 | "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", 917 | "dev": true 918 | }, 919 | "import-fresh": { 920 | "version": "3.2.1", 921 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.2.1.tgz", 922 | "integrity": "sha512-6e1q1cnWP2RXD9/keSkxHScg508CdXqXWgWBaETNhyuBFz+kUZlKboh+ISK+bU++DmbHimVBrOz/zzPe0sZ3sQ==", 923 | "dev": true, 924 | "requires": { 925 | "parent-module": "^1.0.0", 926 | "resolve-from": "^4.0.0" 927 | } 928 | }, 929 | "imurmurhash": { 930 | "version": "0.1.4", 931 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 932 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 933 | "dev": true 934 | }, 935 | "inflight": { 936 | "version": "1.0.6", 937 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 938 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 939 | "dev": true, 940 | "requires": { 941 | "once": "^1.3.0", 942 | "wrappy": "1" 943 | } 944 | }, 945 | "inherits": { 946 | "version": "2.0.4", 947 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 948 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 949 | "dev": true 950 | }, 951 | "inquirer": { 952 | "version": "7.1.0", 953 | "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.1.0.tgz", 954 | "integrity": "sha512-5fJMWEmikSYu0nv/flMc475MhGbB7TSPd/2IpFV4I4rMklboCH2rQjYY5kKiYGHqUF9gvaambupcJFFG9dvReg==", 955 | "dev": true, 956 | "requires": { 957 | "ansi-escapes": "^4.2.1", 958 | "chalk": "^3.0.0", 959 | "cli-cursor": "^3.1.0", 960 | "cli-width": "^2.0.0", 961 | "external-editor": "^3.0.3", 962 | "figures": "^3.0.0", 963 | "lodash": "^4.17.15", 964 | "mute-stream": "0.0.8", 965 | "run-async": "^2.4.0", 966 | "rxjs": "^6.5.3", 967 | "string-width": "^4.1.0", 968 | "strip-ansi": "^6.0.0", 969 | "through": "^2.3.6" 970 | }, 971 | "dependencies": { 972 | "ansi-styles": { 973 | "version": "4.2.1", 974 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz", 975 | "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==", 976 | "dev": true, 977 | "requires": { 978 | "@types/color-name": "^1.1.1", 979 | "color-convert": "^2.0.1" 980 | } 981 | }, 982 | "chalk": { 983 | "version": "3.0.0", 984 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", 985 | "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", 986 | "dev": true, 987 | "requires": { 988 | "ansi-styles": "^4.1.0", 989 | "supports-color": "^7.1.0" 990 | } 991 | }, 992 | "color-convert": { 993 | "version": "2.0.1", 994 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 995 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 996 | "dev": true, 997 | "requires": { 998 | "color-name": "~1.1.4" 999 | } 1000 | }, 1001 | "color-name": { 1002 | "version": "1.1.4", 1003 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 1004 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 1005 | "dev": true 1006 | }, 1007 | "has-flag": { 1008 | "version": "4.0.0", 1009 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1010 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 1011 | "dev": true 1012 | }, 1013 | "strip-ansi": { 1014 | "version": "6.0.0", 1015 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 1016 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 1017 | "dev": true, 1018 | "requires": { 1019 | "ansi-regex": "^5.0.0" 1020 | } 1021 | }, 1022 | "supports-color": { 1023 | "version": "7.1.0", 1024 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz", 1025 | "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==", 1026 | "dev": true, 1027 | "requires": { 1028 | "has-flag": "^4.0.0" 1029 | } 1030 | } 1031 | } 1032 | }, 1033 | "is-binary-path": { 1034 | "version": "2.1.0", 1035 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 1036 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 1037 | "dev": true, 1038 | "requires": { 1039 | "binary-extensions": "^2.0.0" 1040 | } 1041 | }, 1042 | "is-buffer": { 1043 | "version": "2.0.4", 1044 | "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.4.tgz", 1045 | "integrity": "sha512-Kq1rokWXOPXWuaMAqZiJW4XxsmD9zGx9q4aePabbn3qCRGedtH7Cm+zV8WETitMfu1wdh+Rvd6w5egwSngUX2A==", 1046 | "dev": true 1047 | }, 1048 | "is-callable": { 1049 | "version": "1.1.5", 1050 | "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.5.tgz", 1051 | "integrity": "sha512-ESKv5sMCJB2jnHTWZ3O5itG+O128Hsus4K4Qh1h2/cgn2vbgnLSVqfV46AeJA9D5EeeLa9w81KUXMtn34zhX+Q==", 1052 | "dev": true 1053 | }, 1054 | "is-date-object": { 1055 | "version": "1.0.2", 1056 | "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", 1057 | "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", 1058 | "dev": true 1059 | }, 1060 | "is-extglob": { 1061 | "version": "2.1.1", 1062 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1063 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", 1064 | "dev": true 1065 | }, 1066 | "is-fullwidth-code-point": { 1067 | "version": "3.0.0", 1068 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1069 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1070 | "dev": true 1071 | }, 1072 | "is-glob": { 1073 | "version": "4.0.1", 1074 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", 1075 | "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", 1076 | "dev": true, 1077 | "requires": { 1078 | "is-extglob": "^2.1.1" 1079 | } 1080 | }, 1081 | "is-number": { 1082 | "version": "7.0.0", 1083 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 1084 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 1085 | "dev": true 1086 | }, 1087 | "is-regex": { 1088 | "version": "1.0.5", 1089 | "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.5.tgz", 1090 | "integrity": "sha512-vlKW17SNq44owv5AQR3Cq0bQPEb8+kF3UKZ2fiZNOWtztYE5i0CzCZxFDwO58qAOWtxdBRVO/V5Qin1wjCqFYQ==", 1091 | "dev": true, 1092 | "requires": { 1093 | "has": "^1.0.3" 1094 | } 1095 | }, 1096 | "is-symbol": { 1097 | "version": "1.0.3", 1098 | "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", 1099 | "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", 1100 | "dev": true, 1101 | "requires": { 1102 | "has-symbols": "^1.0.1" 1103 | } 1104 | }, 1105 | "isexe": { 1106 | "version": "2.0.0", 1107 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 1108 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 1109 | "dev": true 1110 | }, 1111 | "isomorphic-ws": { 1112 | "version": "4.0.1", 1113 | "resolved": "https://registry.npmjs.org/isomorphic-ws/-/isomorphic-ws-4.0.1.tgz", 1114 | "integrity": "sha512-BhBvN2MBpWTaSHdWRb/bwdZJ1WaehQ2L1KngkCkfLUGF0mAWAT1sQUQacEmQ0jXkFw/czDXPNQSL5u2/Krsz1w==" 1115 | }, 1116 | "js-tokens": { 1117 | "version": "4.0.0", 1118 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 1119 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 1120 | "dev": true 1121 | }, 1122 | "js-yaml": { 1123 | "version": "3.13.1", 1124 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", 1125 | "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", 1126 | "dev": true, 1127 | "requires": { 1128 | "argparse": "^1.0.7", 1129 | "esprima": "^4.0.0" 1130 | } 1131 | }, 1132 | "json-schema-traverse": { 1133 | "version": "0.4.1", 1134 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 1135 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", 1136 | "dev": true 1137 | }, 1138 | "json-stable-stringify-without-jsonify": { 1139 | "version": "1.0.1", 1140 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", 1141 | "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", 1142 | "dev": true 1143 | }, 1144 | "levn": { 1145 | "version": "0.3.0", 1146 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", 1147 | "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", 1148 | "dev": true, 1149 | "requires": { 1150 | "prelude-ls": "~1.1.2", 1151 | "type-check": "~0.3.2" 1152 | } 1153 | }, 1154 | "locate-path": { 1155 | "version": "3.0.0", 1156 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", 1157 | "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", 1158 | "dev": true, 1159 | "requires": { 1160 | "p-locate": "^3.0.0", 1161 | "path-exists": "^3.0.0" 1162 | } 1163 | }, 1164 | "lodash": { 1165 | "version": "4.17.21", 1166 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 1167 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 1168 | "dev": true 1169 | }, 1170 | "log-symbols": { 1171 | "version": "3.0.0", 1172 | "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-3.0.0.tgz", 1173 | "integrity": "sha512-dSkNGuI7iG3mfvDzUuYZyvk5dD9ocYCYzNU6CYDE6+Xqd+gwme6Z00NS3dUh8mq/73HaEtT7m6W+yUPtU6BZnQ==", 1174 | "dev": true, 1175 | "requires": { 1176 | "chalk": "^2.4.2" 1177 | } 1178 | }, 1179 | "mimic-fn": { 1180 | "version": "2.1.0", 1181 | "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", 1182 | "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", 1183 | "dev": true 1184 | }, 1185 | "minimatch": { 1186 | "version": "3.0.4", 1187 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1188 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1189 | "dev": true, 1190 | "requires": { 1191 | "brace-expansion": "^1.1.7" 1192 | } 1193 | }, 1194 | "minimist": { 1195 | "version": "1.2.5", 1196 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 1197 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", 1198 | "dev": true 1199 | }, 1200 | "mkdirp": { 1201 | "version": "0.5.5", 1202 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", 1203 | "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", 1204 | "dev": true, 1205 | "requires": { 1206 | "minimist": "^1.2.5" 1207 | } 1208 | }, 1209 | "mocha": { 1210 | "version": "7.1.2", 1211 | "resolved": "https://registry.npmjs.org/mocha/-/mocha-7.1.2.tgz", 1212 | "integrity": "sha512-o96kdRKMKI3E8U0bjnfqW4QMk12MwZ4mhdBTf+B5a1q9+aq2HRnj+3ZdJu0B/ZhJeK78MgYuv6L8d/rA5AeBJA==", 1213 | "dev": true, 1214 | "requires": { 1215 | "ansi-colors": "3.2.3", 1216 | "browser-stdout": "1.3.1", 1217 | "chokidar": "3.3.0", 1218 | "debug": "3.2.6", 1219 | "diff": "3.5.0", 1220 | "escape-string-regexp": "1.0.5", 1221 | "find-up": "3.0.0", 1222 | "glob": "7.1.3", 1223 | "growl": "1.10.5", 1224 | "he": "1.2.0", 1225 | "js-yaml": "3.13.1", 1226 | "log-symbols": "3.0.0", 1227 | "minimatch": "3.0.4", 1228 | "mkdirp": "0.5.5", 1229 | "ms": "2.1.1", 1230 | "node-environment-flags": "1.0.6", 1231 | "object.assign": "4.1.0", 1232 | "strip-json-comments": "2.0.1", 1233 | "supports-color": "6.0.0", 1234 | "which": "1.3.1", 1235 | "wide-align": "1.1.3", 1236 | "yargs": "13.3.2", 1237 | "yargs-parser": "13.1.2", 1238 | "yargs-unparser": "1.6.0" 1239 | }, 1240 | "dependencies": { 1241 | "debug": { 1242 | "version": "3.2.6", 1243 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 1244 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 1245 | "dev": true, 1246 | "requires": { 1247 | "ms": "^2.1.1" 1248 | } 1249 | }, 1250 | "glob": { 1251 | "version": "7.1.3", 1252 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", 1253 | "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", 1254 | "dev": true, 1255 | "requires": { 1256 | "fs.realpath": "^1.0.0", 1257 | "inflight": "^1.0.4", 1258 | "inherits": "2", 1259 | "minimatch": "^3.0.4", 1260 | "once": "^1.3.0", 1261 | "path-is-absolute": "^1.0.0" 1262 | } 1263 | }, 1264 | "ms": { 1265 | "version": "2.1.1", 1266 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 1267 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", 1268 | "dev": true 1269 | }, 1270 | "strip-json-comments": { 1271 | "version": "2.0.1", 1272 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 1273 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 1274 | "dev": true 1275 | }, 1276 | "supports-color": { 1277 | "version": "6.0.0", 1278 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.0.0.tgz", 1279 | "integrity": "sha512-on9Kwidc1IUQo+bQdhi8+Tijpo0e1SS6RoGo2guUwn5vdaxw8RXOF9Vb2ws+ihWOmh4JnCJOvaziZWP1VABaLg==", 1280 | "dev": true, 1281 | "requires": { 1282 | "has-flag": "^3.0.0" 1283 | } 1284 | } 1285 | } 1286 | }, 1287 | "ms": { 1288 | "version": "2.1.2", 1289 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1290 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1291 | "dev": true 1292 | }, 1293 | "mute-stream": { 1294 | "version": "0.0.8", 1295 | "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", 1296 | "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", 1297 | "dev": true 1298 | }, 1299 | "natural-compare": { 1300 | "version": "1.4.0", 1301 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 1302 | "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", 1303 | "dev": true 1304 | }, 1305 | "nice-try": { 1306 | "version": "1.0.5", 1307 | "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", 1308 | "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", 1309 | "dev": true 1310 | }, 1311 | "node-environment-flags": { 1312 | "version": "1.0.6", 1313 | "resolved": "https://registry.npmjs.org/node-environment-flags/-/node-environment-flags-1.0.6.tgz", 1314 | "integrity": "sha512-5Evy2epuL+6TM0lCQGpFIj6KwiEsGh1SrHUhTbNX+sLbBtjidPZFAnVK9y5yU1+h//RitLbRHTIMyxQPtxMdHw==", 1315 | "dev": true, 1316 | "requires": { 1317 | "object.getownpropertydescriptors": "^2.0.3", 1318 | "semver": "^5.7.0" 1319 | }, 1320 | "dependencies": { 1321 | "semver": { 1322 | "version": "5.7.1", 1323 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1324 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1325 | "dev": true 1326 | } 1327 | } 1328 | }, 1329 | "normalize-path": { 1330 | "version": "3.0.0", 1331 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1332 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1333 | "dev": true 1334 | }, 1335 | "object-inspect": { 1336 | "version": "1.7.0", 1337 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz", 1338 | "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==", 1339 | "dev": true 1340 | }, 1341 | "object-keys": { 1342 | "version": "1.1.1", 1343 | "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", 1344 | "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", 1345 | "dev": true 1346 | }, 1347 | "object.assign": { 1348 | "version": "4.1.0", 1349 | "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz", 1350 | "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==", 1351 | "dev": true, 1352 | "requires": { 1353 | "define-properties": "^1.1.2", 1354 | "function-bind": "^1.1.1", 1355 | "has-symbols": "^1.0.0", 1356 | "object-keys": "^1.0.11" 1357 | } 1358 | }, 1359 | "object.getownpropertydescriptors": { 1360 | "version": "2.1.0", 1361 | "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz", 1362 | "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==", 1363 | "dev": true, 1364 | "requires": { 1365 | "define-properties": "^1.1.3", 1366 | "es-abstract": "^1.17.0-next.1" 1367 | } 1368 | }, 1369 | "once": { 1370 | "version": "1.4.0", 1371 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1372 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1373 | "dev": true, 1374 | "requires": { 1375 | "wrappy": "1" 1376 | } 1377 | }, 1378 | "onetime": { 1379 | "version": "5.1.0", 1380 | "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.0.tgz", 1381 | "integrity": "sha512-5NcSkPHhwTVFIQN+TUqXoS5+dlElHXdpAWu9I0HP20YOtIi+aZ0Ct82jdlILDxjLEAWwvm+qj1m6aEtsDVmm6Q==", 1382 | "dev": true, 1383 | "requires": { 1384 | "mimic-fn": "^2.1.0" 1385 | } 1386 | }, 1387 | "optionator": { 1388 | "version": "0.8.3", 1389 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", 1390 | "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", 1391 | "dev": true, 1392 | "requires": { 1393 | "deep-is": "~0.1.3", 1394 | "fast-levenshtein": "~2.0.6", 1395 | "levn": "~0.3.0", 1396 | "prelude-ls": "~1.1.2", 1397 | "type-check": "~0.3.2", 1398 | "word-wrap": "~1.2.3" 1399 | } 1400 | }, 1401 | "os-tmpdir": { 1402 | "version": "1.0.2", 1403 | "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", 1404 | "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", 1405 | "dev": true 1406 | }, 1407 | "osc-js": { 1408 | "version": "2.1.0", 1409 | "resolved": "https://registry.npmjs.org/osc-js/-/osc-js-2.1.0.tgz", 1410 | "integrity": "sha512-EzlmJnYiVBcj1KNiS9zCTEBFUaGulNhwffJlay2pf5lASuU24vQZ24KmlM0lVGyu+xVf1ibrx06Z4ervIMBmHQ==", 1411 | "requires": { 1412 | "isomorphic-ws": "4.0.1", 1413 | "ws": "7.2.0" 1414 | } 1415 | }, 1416 | "p-limit": { 1417 | "version": "2.3.0", 1418 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 1419 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 1420 | "dev": true, 1421 | "requires": { 1422 | "p-try": "^2.0.0" 1423 | } 1424 | }, 1425 | "p-locate": { 1426 | "version": "3.0.0", 1427 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", 1428 | "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", 1429 | "dev": true, 1430 | "requires": { 1431 | "p-limit": "^2.0.0" 1432 | } 1433 | }, 1434 | "p-try": { 1435 | "version": "2.2.0", 1436 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 1437 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 1438 | "dev": true 1439 | }, 1440 | "parent-module": { 1441 | "version": "1.0.1", 1442 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", 1443 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", 1444 | "dev": true, 1445 | "requires": { 1446 | "callsites": "^3.0.0" 1447 | } 1448 | }, 1449 | "path-exists": { 1450 | "version": "3.0.0", 1451 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", 1452 | "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", 1453 | "dev": true 1454 | }, 1455 | "path-is-absolute": { 1456 | "version": "1.0.1", 1457 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1458 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 1459 | "dev": true 1460 | }, 1461 | "path-key": { 1462 | "version": "2.0.1", 1463 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", 1464 | "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", 1465 | "dev": true 1466 | }, 1467 | "picomatch": { 1468 | "version": "2.2.2", 1469 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", 1470 | "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", 1471 | "dev": true 1472 | }, 1473 | "prelude-ls": { 1474 | "version": "1.1.2", 1475 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", 1476 | "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", 1477 | "dev": true 1478 | }, 1479 | "progress": { 1480 | "version": "2.0.3", 1481 | "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", 1482 | "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", 1483 | "dev": true 1484 | }, 1485 | "punycode": { 1486 | "version": "2.1.1", 1487 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 1488 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 1489 | "dev": true 1490 | }, 1491 | "readdirp": { 1492 | "version": "3.2.0", 1493 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.2.0.tgz", 1494 | "integrity": "sha512-crk4Qu3pmXwgxdSgGhgA/eXiJAPQiX4GMOZZMXnqKxHX7TaoL+3gQVo/WeuAiogr07DpnfjIMpXXa+PAIvwPGQ==", 1495 | "dev": true, 1496 | "requires": { 1497 | "picomatch": "^2.0.4" 1498 | } 1499 | }, 1500 | "regexpp": { 1501 | "version": "3.1.0", 1502 | "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz", 1503 | "integrity": "sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q==", 1504 | "dev": true 1505 | }, 1506 | "require-directory": { 1507 | "version": "2.1.1", 1508 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 1509 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", 1510 | "dev": true 1511 | }, 1512 | "require-main-filename": { 1513 | "version": "2.0.0", 1514 | "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", 1515 | "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", 1516 | "dev": true 1517 | }, 1518 | "resolve-from": { 1519 | "version": "4.0.0", 1520 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", 1521 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", 1522 | "dev": true 1523 | }, 1524 | "restore-cursor": { 1525 | "version": "3.1.0", 1526 | "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", 1527 | "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", 1528 | "dev": true, 1529 | "requires": { 1530 | "onetime": "^5.1.0", 1531 | "signal-exit": "^3.0.2" 1532 | } 1533 | }, 1534 | "rimraf": { 1535 | "version": "2.6.3", 1536 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", 1537 | "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", 1538 | "dev": true, 1539 | "requires": { 1540 | "glob": "^7.1.3" 1541 | } 1542 | }, 1543 | "run-async": { 1544 | "version": "2.4.1", 1545 | "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.1.tgz", 1546 | "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", 1547 | "dev": true 1548 | }, 1549 | "rxjs": { 1550 | "version": "6.5.5", 1551 | "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.5.tgz", 1552 | "integrity": "sha512-WfQI+1gohdf0Dai/Bbmk5L5ItH5tYqm3ki2c5GdWhKjalzjg93N3avFjVStyZZz+A2Em+ZxKH5bNghw9UeylGQ==", 1553 | "dev": true, 1554 | "requires": { 1555 | "tslib": "^1.9.0" 1556 | } 1557 | }, 1558 | "safer-buffer": { 1559 | "version": "2.1.2", 1560 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1561 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", 1562 | "dev": true 1563 | }, 1564 | "semver": { 1565 | "version": "7.3.2", 1566 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", 1567 | "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", 1568 | "dev": true 1569 | }, 1570 | "set-blocking": { 1571 | "version": "2.0.0", 1572 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", 1573 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", 1574 | "dev": true 1575 | }, 1576 | "shebang-command": { 1577 | "version": "1.2.0", 1578 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", 1579 | "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", 1580 | "dev": true, 1581 | "requires": { 1582 | "shebang-regex": "^1.0.0" 1583 | } 1584 | }, 1585 | "shebang-regex": { 1586 | "version": "1.0.0", 1587 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", 1588 | "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", 1589 | "dev": true 1590 | }, 1591 | "signal-exit": { 1592 | "version": "3.0.3", 1593 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 1594 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", 1595 | "dev": true 1596 | }, 1597 | "slice-ansi": { 1598 | "version": "2.1.0", 1599 | "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz", 1600 | "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", 1601 | "dev": true, 1602 | "requires": { 1603 | "ansi-styles": "^3.2.0", 1604 | "astral-regex": "^1.0.0", 1605 | "is-fullwidth-code-point": "^2.0.0" 1606 | }, 1607 | "dependencies": { 1608 | "is-fullwidth-code-point": { 1609 | "version": "2.0.0", 1610 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 1611 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 1612 | "dev": true 1613 | } 1614 | } 1615 | }, 1616 | "sprintf-js": { 1617 | "version": "1.0.3", 1618 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 1619 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 1620 | "dev": true 1621 | }, 1622 | "string-width": { 1623 | "version": "4.2.0", 1624 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", 1625 | "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", 1626 | "dev": true, 1627 | "requires": { 1628 | "emoji-regex": "^8.0.0", 1629 | "is-fullwidth-code-point": "^3.0.0", 1630 | "strip-ansi": "^6.0.0" 1631 | }, 1632 | "dependencies": { 1633 | "strip-ansi": { 1634 | "version": "6.0.0", 1635 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", 1636 | "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", 1637 | "dev": true, 1638 | "requires": { 1639 | "ansi-regex": "^5.0.0" 1640 | } 1641 | } 1642 | } 1643 | }, 1644 | "string.prototype.trimend": { 1645 | "version": "1.0.1", 1646 | "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz", 1647 | "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==", 1648 | "dev": true, 1649 | "requires": { 1650 | "define-properties": "^1.1.3", 1651 | "es-abstract": "^1.17.5" 1652 | } 1653 | }, 1654 | "string.prototype.trimleft": { 1655 | "version": "2.1.2", 1656 | "resolved": "https://registry.npmjs.org/string.prototype.trimleft/-/string.prototype.trimleft-2.1.2.tgz", 1657 | "integrity": "sha512-gCA0tza1JBvqr3bfAIFJGqfdRTyPae82+KTnm3coDXkZN9wnuW3HjGgN386D7hfv5CHQYCI022/rJPVlqXyHSw==", 1658 | "dev": true, 1659 | "requires": { 1660 | "define-properties": "^1.1.3", 1661 | "es-abstract": "^1.17.5", 1662 | "string.prototype.trimstart": "^1.0.0" 1663 | } 1664 | }, 1665 | "string.prototype.trimright": { 1666 | "version": "2.1.2", 1667 | "resolved": "https://registry.npmjs.org/string.prototype.trimright/-/string.prototype.trimright-2.1.2.tgz", 1668 | "integrity": "sha512-ZNRQ7sY3KroTaYjRS6EbNiiHrOkjihL9aQE/8gfQ4DtAC/aEBRHFJa44OmoWxGGqXuJlfKkZW4WcXErGr+9ZFg==", 1669 | "dev": true, 1670 | "requires": { 1671 | "define-properties": "^1.1.3", 1672 | "es-abstract": "^1.17.5", 1673 | "string.prototype.trimend": "^1.0.0" 1674 | } 1675 | }, 1676 | "string.prototype.trimstart": { 1677 | "version": "1.0.1", 1678 | "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz", 1679 | "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==", 1680 | "dev": true, 1681 | "requires": { 1682 | "define-properties": "^1.1.3", 1683 | "es-abstract": "^1.17.5" 1684 | } 1685 | }, 1686 | "strip-ansi": { 1687 | "version": "5.2.0", 1688 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", 1689 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", 1690 | "dev": true, 1691 | "requires": { 1692 | "ansi-regex": "^4.1.0" 1693 | }, 1694 | "dependencies": { 1695 | "ansi-regex": { 1696 | "version": "4.1.0", 1697 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", 1698 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", 1699 | "dev": true 1700 | } 1701 | } 1702 | }, 1703 | "strip-json-comments": { 1704 | "version": "3.1.0", 1705 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.0.tgz", 1706 | "integrity": "sha512-e6/d0eBu7gHtdCqFt0xJr642LdToM5/cN4Qb9DbHjVx1CP5RyeM+zH7pbecEmDv/lBqb0QH+6Uqq75rxFPkM0w==", 1707 | "dev": true 1708 | }, 1709 | "supports-color": { 1710 | "version": "5.5.0", 1711 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1712 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1713 | "dev": true, 1714 | "requires": { 1715 | "has-flag": "^3.0.0" 1716 | } 1717 | }, 1718 | "table": { 1719 | "version": "5.4.6", 1720 | "resolved": "https://registry.npmjs.org/table/-/table-5.4.6.tgz", 1721 | "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", 1722 | "dev": true, 1723 | "requires": { 1724 | "ajv": "^6.10.2", 1725 | "lodash": "^4.17.14", 1726 | "slice-ansi": "^2.1.0", 1727 | "string-width": "^3.0.0" 1728 | }, 1729 | "dependencies": { 1730 | "emoji-regex": { 1731 | "version": "7.0.3", 1732 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 1733 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 1734 | "dev": true 1735 | }, 1736 | "is-fullwidth-code-point": { 1737 | "version": "2.0.0", 1738 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 1739 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 1740 | "dev": true 1741 | }, 1742 | "string-width": { 1743 | "version": "3.1.0", 1744 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 1745 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 1746 | "dev": true, 1747 | "requires": { 1748 | "emoji-regex": "^7.0.1", 1749 | "is-fullwidth-code-point": "^2.0.0", 1750 | "strip-ansi": "^5.1.0" 1751 | } 1752 | } 1753 | } 1754 | }, 1755 | "text-table": { 1756 | "version": "0.2.0", 1757 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", 1758 | "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", 1759 | "dev": true 1760 | }, 1761 | "through": { 1762 | "version": "2.3.8", 1763 | "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", 1764 | "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", 1765 | "dev": true 1766 | }, 1767 | "tmp": { 1768 | "version": "0.0.33", 1769 | "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", 1770 | "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", 1771 | "dev": true, 1772 | "requires": { 1773 | "os-tmpdir": "~1.0.2" 1774 | } 1775 | }, 1776 | "to-regex-range": { 1777 | "version": "5.0.1", 1778 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1779 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1780 | "dev": true, 1781 | "requires": { 1782 | "is-number": "^7.0.0" 1783 | } 1784 | }, 1785 | "tslib": { 1786 | "version": "1.13.0", 1787 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.13.0.tgz", 1788 | "integrity": "sha512-i/6DQjL8Xf3be4K/E6Wgpekn5Qasl1usyw++dAA35Ue5orEn65VIxOA+YvNNl9HV3qv70T7CNwjODHZrLwvd1Q==", 1789 | "dev": true 1790 | }, 1791 | "tsutils": { 1792 | "version": "3.17.1", 1793 | "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.17.1.tgz", 1794 | "integrity": "sha512-kzeQ5B8H3w60nFY2g8cJIuH7JDpsALXySGtwGJ0p2LSjLgay3NdIpqq5SoOBe46bKDW2iq25irHCr8wjomUS2g==", 1795 | "dev": true, 1796 | "requires": { 1797 | "tslib": "^1.8.1" 1798 | } 1799 | }, 1800 | "type-check": { 1801 | "version": "0.3.2", 1802 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", 1803 | "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", 1804 | "dev": true, 1805 | "requires": { 1806 | "prelude-ls": "~1.1.2" 1807 | } 1808 | }, 1809 | "type-fest": { 1810 | "version": "0.8.1", 1811 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 1812 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 1813 | "dev": true 1814 | }, 1815 | "typescript": { 1816 | "version": "3.9.3", 1817 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.3.tgz", 1818 | "integrity": "sha512-D/wqnB2xzNFIcoBG9FG8cXRDjiqSTbG2wd8DMZeQyJlP1vfTkIxH4GKveWaEBYySKIg+USu+E+EDIR47SqnaMQ==", 1819 | "dev": true 1820 | }, 1821 | "uri-js": { 1822 | "version": "4.2.2", 1823 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", 1824 | "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", 1825 | "dev": true, 1826 | "requires": { 1827 | "punycode": "^2.1.0" 1828 | } 1829 | }, 1830 | "utf8": { 1831 | "version": "3.0.0", 1832 | "resolved": "https://registry.npmjs.org/utf8/-/utf8-3.0.0.tgz", 1833 | "integrity": "sha512-E8VjFIQ/TyQgp+TZfS6l8yp/xWppSAHzidGiRrqe4bK4XP9pTRyKFgGJpO3SN7zdX4DeomTrwaseCHovfpFcqQ==" 1834 | }, 1835 | "uuid": { 1836 | "version": "8.1.0", 1837 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.1.0.tgz", 1838 | "integrity": "sha512-CI18flHDznR0lq54xBycOVmphdCYnQLKn8abKn7PXUiKUGdEd+/l9LWNJmugXel4hXq7S+RMNl34ecyC9TntWg==" 1839 | }, 1840 | "v8-compile-cache": { 1841 | "version": "2.1.0", 1842 | "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.0.tgz", 1843 | "integrity": "sha512-usZBT3PW+LOjM25wbqIlZwPeJV+3OSz3M1k1Ws8snlW39dZyYL9lOGC5FgPVHfk0jKmjiDV8Z0mIbVQPiwFs7g==", 1844 | "dev": true 1845 | }, 1846 | "vscode-test": { 1847 | "version": "1.3.0", 1848 | "resolved": "https://registry.npmjs.org/vscode-test/-/vscode-test-1.3.0.tgz", 1849 | "integrity": "sha512-LddukcBiSU2FVTDr3c1D8lwkiOvwlJdDL2hqVbn6gIz+rpTqUCkMZSKYm94Y1v0WXlHSDQBsXyY+tchWQgGVsw==", 1850 | "dev": true, 1851 | "requires": { 1852 | "http-proxy-agent": "^2.1.0", 1853 | "https-proxy-agent": "^2.2.4", 1854 | "rimraf": "^2.6.3" 1855 | } 1856 | }, 1857 | "which": { 1858 | "version": "1.3.1", 1859 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", 1860 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", 1861 | "dev": true, 1862 | "requires": { 1863 | "isexe": "^2.0.0" 1864 | } 1865 | }, 1866 | "which-module": { 1867 | "version": "2.0.0", 1868 | "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", 1869 | "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", 1870 | "dev": true 1871 | }, 1872 | "wide-align": { 1873 | "version": "1.1.3", 1874 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", 1875 | "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", 1876 | "dev": true, 1877 | "requires": { 1878 | "string-width": "^1.0.2 || 2" 1879 | }, 1880 | "dependencies": { 1881 | "ansi-regex": { 1882 | "version": "3.0.0", 1883 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", 1884 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", 1885 | "dev": true 1886 | }, 1887 | "is-fullwidth-code-point": { 1888 | "version": "2.0.0", 1889 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 1890 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 1891 | "dev": true 1892 | }, 1893 | "string-width": { 1894 | "version": "2.1.1", 1895 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", 1896 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", 1897 | "dev": true, 1898 | "requires": { 1899 | "is-fullwidth-code-point": "^2.0.0", 1900 | "strip-ansi": "^4.0.0" 1901 | } 1902 | }, 1903 | "strip-ansi": { 1904 | "version": "4.0.0", 1905 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", 1906 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", 1907 | "dev": true, 1908 | "requires": { 1909 | "ansi-regex": "^3.0.0" 1910 | } 1911 | } 1912 | } 1913 | }, 1914 | "word-wrap": { 1915 | "version": "1.2.3", 1916 | "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", 1917 | "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", 1918 | "dev": true 1919 | }, 1920 | "wrap-ansi": { 1921 | "version": "5.1.0", 1922 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", 1923 | "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", 1924 | "dev": true, 1925 | "requires": { 1926 | "ansi-styles": "^3.2.0", 1927 | "string-width": "^3.0.0", 1928 | "strip-ansi": "^5.0.0" 1929 | }, 1930 | "dependencies": { 1931 | "emoji-regex": { 1932 | "version": "7.0.3", 1933 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 1934 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 1935 | "dev": true 1936 | }, 1937 | "is-fullwidth-code-point": { 1938 | "version": "2.0.0", 1939 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 1940 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 1941 | "dev": true 1942 | }, 1943 | "string-width": { 1944 | "version": "3.1.0", 1945 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 1946 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 1947 | "dev": true, 1948 | "requires": { 1949 | "emoji-regex": "^7.0.1", 1950 | "is-fullwidth-code-point": "^2.0.0", 1951 | "strip-ansi": "^5.1.0" 1952 | } 1953 | } 1954 | } 1955 | }, 1956 | "wrappy": { 1957 | "version": "1.0.2", 1958 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1959 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 1960 | "dev": true 1961 | }, 1962 | "write": { 1963 | "version": "1.0.3", 1964 | "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", 1965 | "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", 1966 | "dev": true, 1967 | "requires": { 1968 | "mkdirp": "^0.5.1" 1969 | } 1970 | }, 1971 | "ws": { 1972 | "version": "7.2.0", 1973 | "resolved": "https://registry.npmjs.org/ws/-/ws-7.2.0.tgz", 1974 | "integrity": "sha512-+SqNqFbwTm/0DC18KYzIsMTnEWpLwJsiasW/O17la4iDRRIO9uaHbvKiAS3AHgTiuuWerK/brj4O6MYZkei9xg==", 1975 | "requires": { 1976 | "async-limiter": "^1.0.0" 1977 | } 1978 | }, 1979 | "y18n": { 1980 | "version": "4.0.1", 1981 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.1.tgz", 1982 | "integrity": "sha512-wNcy4NvjMYL8gogWWYAO7ZFWFfHcbdbE57tZO8e4cbpj8tfUcwrwqSl3ad8HxpYWCdXcJUCeKKZS62Av1affwQ==", 1983 | "dev": true 1984 | }, 1985 | "yargs": { 1986 | "version": "13.3.2", 1987 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", 1988 | "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", 1989 | "dev": true, 1990 | "requires": { 1991 | "cliui": "^5.0.0", 1992 | "find-up": "^3.0.0", 1993 | "get-caller-file": "^2.0.1", 1994 | "require-directory": "^2.1.1", 1995 | "require-main-filename": "^2.0.0", 1996 | "set-blocking": "^2.0.0", 1997 | "string-width": "^3.0.0", 1998 | "which-module": "^2.0.0", 1999 | "y18n": "^4.0.0", 2000 | "yargs-parser": "^13.1.2" 2001 | }, 2002 | "dependencies": { 2003 | "emoji-regex": { 2004 | "version": "7.0.3", 2005 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", 2006 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", 2007 | "dev": true 2008 | }, 2009 | "is-fullwidth-code-point": { 2010 | "version": "2.0.0", 2011 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 2012 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 2013 | "dev": true 2014 | }, 2015 | "string-width": { 2016 | "version": "3.1.0", 2017 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", 2018 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", 2019 | "dev": true, 2020 | "requires": { 2021 | "emoji-regex": "^7.0.1", 2022 | "is-fullwidth-code-point": "^2.0.0", 2023 | "strip-ansi": "^5.1.0" 2024 | } 2025 | } 2026 | } 2027 | }, 2028 | "yargs-parser": { 2029 | "version": "13.1.2", 2030 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz", 2031 | "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", 2032 | "dev": true, 2033 | "requires": { 2034 | "camelcase": "^5.0.0", 2035 | "decamelize": "^1.2.0" 2036 | } 2037 | }, 2038 | "yargs-unparser": { 2039 | "version": "1.6.0", 2040 | "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-1.6.0.tgz", 2041 | "integrity": "sha512-W9tKgmSn0DpSatfri0nx52Joq5hVXgeLiqR/5G0sZNDoLZFOr/xjBUDcShCOGNsBnEMNo1KAMBkTej1Hm62HTw==", 2042 | "dev": true, 2043 | "requires": { 2044 | "flat": "^4.1.0", 2045 | "lodash": "^4.17.15", 2046 | "yargs": "^13.3.0" 2047 | } 2048 | } 2049 | } 2050 | } 2051 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "sonicpieditor", 3 | "publisher": "Luis Lloret", 4 | "displayName": "SonicPiEditor", 5 | "description": "An editor for Sonic Pi", 6 | "version": "0.0.1", 7 | "engines": { 8 | "vscode": "^1.45.0" 9 | }, 10 | "categories": [ 11 | "Other" 12 | ], 13 | "activationEvents": [ 14 | "*" 15 | ], 16 | "main": "./out/extension.js", 17 | "contributes": { 18 | "commands": [ 19 | { 20 | "command": "sonicpieditor.startserver", 21 | "title": "Start Server", 22 | "category": "Sonic Pi" 23 | }, 24 | { 25 | "command": "sonicpieditor.run", 26 | "title": "Run", 27 | "category": "Sonic Pi" 28 | }, 29 | { 30 | "command": "sonicpieditor.runselected", 31 | "title": "Run selected text", 32 | "category": "Sonic Pi" 33 | }, 34 | { 35 | "command": "sonicpieditor.stop", 36 | "title": "Stop", 37 | "category": "Sonic Pi" 38 | }, 39 | { 40 | "command": "sonicpieditor.togglerecording", 41 | "title": "Toggle recording", 42 | "category": "Sonic Pi" 43 | } 44 | ], 45 | "menus": { 46 | "editor/title": [ 47 | { 48 | "command": "sonicpieditor.togglerecording" 49 | } 50 | ] 51 | }, 52 | "keybindings": [ 53 | { 54 | "command": "sonicpieditor.run", 55 | "key": "alt+r", 56 | "mac": "cmd+r", 57 | "when": "editorTextFocus" 58 | }, 59 | { 60 | "command": "sonicpieditor.runselected", 61 | "key": "alt+t", 62 | "mac": "cmd+t", 63 | "when": "editorTextFocus" 64 | }, 65 | { 66 | "command": "sonicpieditor.stop", 67 | "key": "alt+s", 68 | "mac": "cmd+s" 69 | }, 70 | { 71 | "command": "sonicpieditor.togglerecording", 72 | "key": "shift+alt+r", 73 | "mac": "shift+cmd+r" 74 | } 75 | ], 76 | "snippets": [ 77 | { 78 | "language": "ruby", 79 | "path": "./snippets/sonic-pi-snippets.json" 80 | } 81 | ], 82 | "configuration": { 83 | "title": "Sonic Pi", 84 | "properties": { 85 | "sonicpieditor.sonicPiRootDirectory": { 86 | "type": "string", 87 | "description": "Set Sonic Pi installation directory.\nOnly use this if the default does not work for you." 88 | }, 89 | "sonicpieditor.launchSonicPiServerAutomatically": { 90 | "type": "string", 91 | "enum": [ 92 | "ruby", 93 | "start", 94 | "never", 95 | "custom" 96 | ], 97 | "enumDescriptions": [ 98 | "Start the server when there is a ruby file open in the editor", 99 | "Start the server when starting vscode", 100 | "Do not start the server automatically (must launch it manually through the Sonic Pi: Start Server command)", 101 | "Start the server when opening a custom file extension - enter in the Launch Sonic Pi Server Custom Extension setting" 102 | ], 103 | "default": "ruby", 104 | "description": "Launch Sonic Pi Server automatically when..." 105 | }, 106 | "sonicpieditor.launchSonicPiServerCustomExtension": { 107 | "type": "string", 108 | "description": "If the Launch server automatically setting is set to custom, what extension triggers the launch?" 109 | }, 110 | "sonicpieditor.runFileWhenRunSelectedIsEmpty": { 111 | "type": "string", 112 | "enum": [ 113 | "always", 114 | "never" 115 | ], 116 | "description": "If there is no selection when Run selected is triggered, run the whole file" 117 | }, 118 | "sonicpieditor.invertStereo": { 119 | "type": "boolean", 120 | "default": false, 121 | "description": "Invert stereo" 122 | }, 123 | "sonicpieditor.forceMono": { 124 | "type": "boolean", 125 | "default": false, 126 | "description": "Force mono output" 127 | }, 128 | "sonicpieditor.safeMode": { 129 | "type": "boolean", 130 | "default": true, 131 | "description": "Synth argument checking functions.\nIf disabled, certain synth opt values may\ncreate unexpectedly loud or uncomfortable sounds." 132 | }, 133 | "sonicpieditor.logAutoscroll": { 134 | "type": "boolean", 135 | "default": true, 136 | "description": "Autoscroll the log" 137 | }, 138 | "sonicpieditor.logClearOnRun": { 139 | "type": "boolean", 140 | "default": true, 141 | "description": "Clear the log when executing a run" 142 | }, 143 | "sonicpieditor.flashBackgroundColor": { 144 | "type": "string", 145 | "default": "rgba(255,20,147,1.0)", 146 | "description": "Background flash color when running code" 147 | }, 148 | "sonicpieditor.flashTextColor": { 149 | "type": "string", 150 | "default": "rgba(255,255,255,1.0)", 151 | "description": "Text flash color when running code" 152 | } 153 | } 154 | } 155 | }, 156 | "scripts": { 157 | "vscode:prepublish": "npm run compile", 158 | "compile": "tsc -p ./", 159 | "lint": "eslint src --ext ts", 160 | "watch": "tsc -watch -p ./", 161 | "pretest": "npm run compile && npm run lint", 162 | "test": "node ./out/test/runTest.js" 163 | }, 164 | "devDependencies": { 165 | "@types/glob": "^7.1.1", 166 | "@types/mocha": "^7.0.2", 167 | "@types/node": "^13.13.9", 168 | "@types/vscode": "^1.45.0", 169 | "@typescript-eslint/eslint-plugin": "^2.30.0", 170 | "@typescript-eslint/parser": "^2.30.0", 171 | "eslint": "^6.8.0", 172 | "glob": "^7.1.6", 173 | "mocha": "^7.1.2", 174 | "typescript": "^3.8.3", 175 | "vscode-test": "^1.3.0" 176 | }, 177 | "dependencies": { 178 | "osc-js": "2.1.0", 179 | "utf8": "^3.0.0", 180 | "uuid": "8.1.0" 181 | } 182 | } 183 | -------------------------------------------------------------------------------- /snippets/sonic-pi-snippets.json: -------------------------------------------------------------------------------- 1 | { 2 | "live_loop": { 3 | "scope": "ruby", 4 | "prefix": ["live_loop", "ll"], 5 | "body": "live_loop :${1:foo} do\n${2:what}\n${3:sleep 1}\nend" 6 | }, 7 | 8 | "synth": { 9 | "scope": "ruby", 10 | "prefix": ["us", "use_synth"], 11 | "body": "use_synth :${1|beep,blade,bnoise,chipbass,chiplead,chipnoise,cnoise,dark_ambience,dpulse,dsaw,dtri,dull_bell,fm,gnoise,growl,hollow,hoover,mod_beep,mod_dsaw,mod_fm,mod_pulse,mod_saw,mod_sine,mod_tri,noise,piano,pluck,pnoise,pretty_bell,prophet,pulse,saw,sine,sound_in,sound_in_stereo,square,subpulse,supersaw,tb303,tech_saws,tri,zawa|}" 12 | }, 13 | 14 | "synth beep": { 15 | "scope": "ruby", 16 | "prefix": ["us b"], 17 | "body": "use_synth :beep" 18 | }, 19 | "synth chipbass": { 20 | "scope": "ruby", 21 | "prefix": ["us c"], 22 | "body": "use_synth :chipbass" 23 | }, 24 | "synth dark_ambience": { 25 | "scope": "ruby", 26 | "prefix": ["us d"], 27 | "body": "use_synth :dark_ambience" 28 | }, 29 | "synth fm": { 30 | "scope": "ruby", 31 | "prefix": ["us f"], 32 | "body": "use_synth :fm" 33 | }, 34 | "synth growl": { 35 | "scope": "ruby", 36 | "prefix": ["us g"], 37 | "body": "use_synth :growl" 38 | }, 39 | "synth hoover": { 40 | "scope": "ruby", 41 | "prefix": ["us h"], 42 | "body": "use_synth :beep" 43 | }, 44 | "synth noise": { 45 | "scope": "ruby", 46 | "prefix": ["us n"], 47 | "body": "use_synth :noise" 48 | }, 49 | "synth piano": { 50 | "scope": "ruby", 51 | "prefix": ["us p"], 52 | "body": "use_synth :piano" 53 | }, 54 | "synth saw": { 55 | "scope": "ruby", 56 | "prefix": ["us s"], 57 | "body": "use_synth :saw" 58 | }, 59 | "synth tb303": { 60 | "scope": "ruby", 61 | "prefix": ["us t"], 62 | "body": "use_synth :tb303" 63 | }, 64 | "synth zawa": { 65 | "scope": "ruby", 66 | "prefix": ["us z"], 67 | "body": "use_synth :zawa" 68 | }, 69 | "fx": { 70 | "scope": "ruby", 71 | "prefix": ["fx", "with_fx"], 72 | "body": "with_fx :${1|autotuner,band_eq,bitcrusher,bpf,compressor,distortion,echo,eq,flanger,gverb,hpf,ixi_techno,krush,level,lpf,mono,nbpf,nhpf,nlpf,normaliser,nrbpf,nrhpf,nrlpf,octaver,pan,panslicer,ping_pong,pitch_shift,rbpf,record,reverb,rhpf,ring_mod,rlpf,slicer,sound_out,sound_out_stereo,tanh,tremolo,vowel,whammy,wobble|} do\n\t${2:play 50\n\tsleep 1}\nend" 73 | }, 74 | "fx autotuner": { 75 | "scope": "ruby", 76 | "prefix": ["fx a"], 77 | "body": "with_fx :autotuner do\n\t$1\nend" 78 | }, 79 | "fx bitcrusher": { 80 | "scope": "ruby", 81 | "prefix": ["fx b"], 82 | "body": "with_fx :bitcrusher do\n\t$1\nend" 83 | }, 84 | "fx compressor": { 85 | "scope": "ruby", 86 | "prefix": ["fx c"], 87 | "body": "with_fx :compressor do\n\t$1\nend" 88 | }, 89 | "fx distortion": { 90 | "scope": "ruby", 91 | "prefix": ["fx d"], 92 | "body": "with_fx :distortion do\n\t$1\nend" 93 | }, 94 | "fx flanger": { 95 | "scope": "ruby", 96 | "prefix": ["fx f"], 97 | "body": "with_fx :flanger do\n\t$1\nend" 98 | }, 99 | "fx echo": { 100 | "scope": "ruby", 101 | "prefix": ["fx e"], 102 | "body": "with_fx :echo do\n\t$1\nend" 103 | }, 104 | "fx krush": { 105 | "scope": "ruby", 106 | "prefix": ["fx k"], 107 | "body": "with_fx :krush do\n\t$1\nend" 108 | }, 109 | "fx ping_pong": { 110 | "scope": "ruby", 111 | "prefix": ["fx p"], 112 | "body": "with_fx :ping_pong do\n\t$1\nend" 113 | }, 114 | "fx reverb": { 115 | "scope": "ruby", 116 | "prefix": ["fx r"], 117 | "body": "with_fx :reverb do\n\t$1\nend" 118 | }, 119 | "fx ringmod": { 120 | "scope": "ruby", 121 | "prefix": ["fx rm"], 122 | "body": "with_fx :ring_mod do\n\t$1\nend" 123 | }, 124 | "fx slicer": { 125 | "scope": "ruby", 126 | "prefix": ["fx s"], 127 | "body": "with_fx :slicer do\n\t$1\nend" 128 | }, 129 | "fx tremolo": { 130 | "scope": "ruby", 131 | "prefix": ["fx t"], 132 | "body": "with_fx :tremolo do\n\t$1\nend" 133 | }, 134 | "fx wobble": { 135 | "scope": "ruby", 136 | "prefix": ["fx w"], 137 | "body": "with_fx :wobble do\n\t$1\nend" 138 | }, 139 | "sa": { 140 | "scope": "ruby", 141 | "prefix": ["sa", "sample"], 142 | "body": "sample :${1|ambi_soft_buzz,ambi_swoosh,ambi_drone,ambi_glass_hum,ambi_glass_rub,ambi_haunted_hum,ambi_piano,ambi_lunar_land,ambi_dark_woosh,ambi_choir,ambi_sauna,bd_ada,bd_pure,bd_808,bd_zum,bd_gas,bd_sone,bd_haus,bd_zome,bd_boom,bd_klub,bd_fat,bd_tek,bd_mehackit,bass_hit_c,bass_hard_c,bass_thick_c,bass_drop_c,bass_woodsy_c,bass_voxy_c,bass_voxy_hit_c,bass_dnb_f,drum_heavy_kick,drum_tom_mid_soft,drum_tom_mid_hard,drum_tom_lo_soft,drum_tom_lo_hard,drum_tom_hi_soft,drum_tom_hi_hard,drum_splash_soft,drum_splash_hard,drum_snare_soft,drum_snare_hard,drum_cymbal_soft,drum_cymbal_hard,drum_cymbal_open,drum_cymbal_closed,drum_cymbal_pedal,drum_bass_soft,drum_bass_hard,drum_cowbell,drum_roll,elec_triangle,elec_snare,elec_lo_snare,elec_hi_snare,elec_mid_snare,elec_cymbal,elec_soft_kick,elec_filt_snare,elec_fuzz_tom,elec_chime,elec_bong,elec_twang,elec_wood,elec_pop,elec_beep,elec_blip,elec_blip2,elec_ping,elec_bell,elec_flip,elec_tick,elec_hollow_kick,elec_twip,elec_plip,elec_blup,glitch_bass_g,glitch_perc1,glitch_perc2,glitch_perc3,glitch_perc4,glitch_perc5,glitch_robot1,glitch_robot2,mehackit_phone1,mehackit_phone2,mehackit_phone3,mehackit_phone4,mehackit_robot1,mehackit_robot2,mehackit_robot3,mehackit_robot4,mehackit_robot5,mehackit_robot6,mehackit_robot7,misc_burp,misc_crow,misc_cineboom,perc_bell,perc_bell2,perc_snap,perc_snap2,perc_swash,perc_till,perc_door,perc_impact1,perc_impact2,perc_swoosh,sn_dub,sn_dolf,sn_zome,sn_generic,guit_harmonics,guit_e_fifths,guit_e_slide,guit_em9,loop_industrial,loop_compus,loop_amen,loop_amen_full,loop_garzul,loop_mika,loop_breakbeat,loop_safari,loop_tabla,loop_3d_printer,loop_drone_g_97,loop_electric,loop_mehackit1,loop_mehackit2,loop_perc1,loop_perc2,loop_weirdo,tabla_tas1,tabla_tas2,tabla_tas3,tabla_ke1,tabla_ke2,tabla_ke3,tabla_na,tabla_na_o,tabla_tun1,tabla_tun2,tabla_tun3,tabla_te1,tabla_te2,tabla_te_ne,tabla_te_m,tabla_ghe1,tabla_ghe2,tabla_ghe3,tabla_ghe4,tabla_ghe5,tabla_ghe6,tabla_ghe7,tabla_ghe8,tabla_dhec,tabla_na_s,tabla_re,vinyl_backspin,vinyl_rewind,vinyl_scratch,vinyl_hiss|}" 143 | } 144 | } 145 | -------------------------------------------------------------------------------- /src/config.ts: -------------------------------------------------------------------------------- 1 | import * as vscode from 'vscode'; 2 | 3 | export class Config { 4 | private getConfiguration = vscode.workspace.getConfiguration; 5 | private section: string = 'sonicpieditor'; 6 | private rubySection: string = 'ruby.interpreter'; 7 | 8 | // sonic-pi's config 9 | public flashBackgroundColor(): string { 10 | return this.getConfiguration(this.section).flashBackgroundColor; 11 | } 12 | public flashTextColor(): string { 13 | return this.getConfiguration(this.section).flashTextColor; 14 | } 15 | public launchSonicPiServerAutomatically(): string { 16 | return this.getConfiguration(this.section).launchSonicPiServerAutomatically; 17 | } 18 | public runFileWhenRunSelectedIsEmpty(): string { 19 | return this.getConfiguration(this.section).runFileWhenRunSelectedIsEmpty; 20 | } 21 | public sonicPiRootDirectory(): string { 22 | return this.getConfiguration(this.section).sonicPiRootDirectory; 23 | } 24 | public launchSonicPiServerCustomExtension(): string { 25 | return this.getConfiguration(this.section).launchSonicPiServerCustomExtension; 26 | } 27 | public invertStereo(): boolean { 28 | return this.getConfiguration(this.section).invertStereo; 29 | } 30 | public forceMono(): boolean { 31 | return this.getConfiguration(this.section).forceMono; 32 | } 33 | public logClearOnRun(): boolean { 34 | return this.getConfiguration(this.section).logClearOnRun; 35 | } 36 | public safeMode(): boolean { 37 | return this.getConfiguration(this.section).safeMode; 38 | } 39 | public updateRunFileWhenRunSelectedIsEmpty(value: string) { 40 | this.getConfiguration(this.section).update('runFileWhenRunSelectedIsEmpty', value, true); 41 | } 42 | 43 | // ruby interpreter's config 44 | public commandPath(): string { 45 | return this.getConfiguration(this.rubySection).commandPath; 46 | } 47 | } -------------------------------------------------------------------------------- /src/extension.ts: -------------------------------------------------------------------------------- 1 | // MIT License 2 | 3 | // Copyright (c) 2020 Luis Lloret 4 | 5 | // Permission is hereby granted, free of charge, to any person obtaining a copy 6 | // of this software and associated documentation files (the "Software"), to deal 7 | // in the Software without restriction, including without limitation the rights 8 | // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | // copies of the Software, and to permit persons to whom the Software is 10 | // furnished to do so, subject to the following conditions: 11 | 12 | // The above copyright notice and this permission notice shall be included in all 13 | // copies or substantial portions of the Software. 14 | 15 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | // SOFTWARE. 22 | 23 | // Important!! Note that for now, the code is sent as "workspace_nine", so this will overwrite 24 | // what you have in tab 9 on your Sonic Pi session. This should be smarter in the future, and 25 | // I think there are plans to make this more flexible on Sonic Pi's side 26 | 27 | // At the moment this has been tested on Windows... my Linux VM with Linux Mint refuses to 28 | // start Supercollider. Something to do with jackd and pulseaudio. Will investigate later. 29 | // It would be great if someone can check if it works in Linux, and provide a PR if it doesn't. 30 | 31 | import * as vscode from 'vscode'; 32 | 33 | import { Main } from './main'; 34 | import { Config } from './config'; 35 | 36 | export function activate(context: vscode.ExtensionContext) { 37 | console.log('Ruby detected. Sonic Pi editor extension active!'); 38 | 39 | let main = new Main(); 40 | 41 | main.checkSonicPiPath(); 42 | 43 | let config = new Config(); 44 | if (config.launchSonicPiServerAutomatically() === 'start'){ 45 | main.startServer(); 46 | } 47 | 48 | let isRecording = false; 49 | 50 | // Register the editor commands. For now, run, stop and recording. Those should be enough for 51 | // some initial fun... 52 | let disposable = vscode.commands.registerCommand('sonicpieditor.startserver', () => { 53 | main.startServer(); 54 | }); 55 | 56 | disposable = vscode.commands.registerTextEditorCommand('sonicpieditor.run', (textEditor) => { 57 | let doc = textEditor.document; 58 | // If the focus is on something that is not ruby (i.e. something on the output pane), 59 | // run the first found open ruby editor instead 60 | if (doc.languageId !== 'ruby'){ 61 | let textEditors = vscode.window.visibleTextEditors; 62 | let rubyEditors = textEditors.filter((editor) => { 63 | return editor.document.languageId === 'ruby'; 64 | }); 65 | 66 | // TODO: if no ruby editors, show a warning to indicate that this will not have effect 67 | if (!rubyEditors.length){ 68 | return; 69 | } 70 | doc = rubyEditors[0].document; 71 | } 72 | let code = doc.getText(); 73 | main.flashCode(textEditor, true); 74 | main.runCode(code); 75 | }); 76 | 77 | 78 | disposable = vscode.commands.registerTextEditorCommand('sonicpieditor.runselected', (textEditor) => { 79 | let doc = textEditor.document; 80 | // If the focus is on something that is not ruby (i.e. something on the output pane), 81 | // run the first found open ruby editor instead 82 | if (doc.languageId !== 'ruby'){ 83 | let textEditors = vscode.window.visibleTextEditors; 84 | let rubyEditors = textEditors.filter((editor) => { 85 | return editor.document.languageId === 'ruby'; 86 | }); 87 | 88 | // TODO: if no ruby editors, show a warning to indicate that this will not have effect 89 | if (!rubyEditors.length){ 90 | return; 91 | } 92 | doc = rubyEditors[0].document; 93 | } 94 | let code = doc.getText(textEditor.selection); 95 | if (!code){ 96 | let runFileWhenRunSelectedIsEmpty = config.runFileWhenRunSelectedIsEmpty(); 97 | if (!runFileWhenRunSelectedIsEmpty){ 98 | vscode.window.showWarningMessage('You tried to Run selected code with no code selected.' + 99 | 'Do you want to run the whole file when this happens?', 'Yes, once', 'Yes, always', 'No, never').then( 100 | item => { 101 | if (item === 'Yes, once'){ 102 | code = doc.getText(); 103 | main.flashCode(textEditor, true); 104 | main.runCode(code); 105 | } 106 | else if (item === 'Yes, always'){ 107 | config.updateRunFileWhenRunSelectedIsEmpty('always'); 108 | code = doc.getText(); 109 | main.flashCode(textEditor, true); 110 | main.runCode(code); 111 | } 112 | else if (item === 'No, never'){ 113 | config.updateRunFileWhenRunSelectedIsEmpty('never'); 114 | } 115 | } 116 | ); 117 | return; 118 | } 119 | else if (runFileWhenRunSelectedIsEmpty === 'never'){ 120 | return; 121 | } 122 | else if (runFileWhenRunSelectedIsEmpty === 'always'){ 123 | code = doc.getText(); 124 | main.flashCode(textEditor, true); 125 | main.runCode(code); 126 | } 127 | 128 | } 129 | main.flashCode(textEditor, false); 130 | main.runCode(code, textEditor.selection.start.line); 131 | }); 132 | 133 | 134 | disposable = vscode.commands.registerCommand('sonicpieditor.stop', () => { 135 | main.stopAllJobs(); 136 | }); 137 | 138 | disposable = vscode.commands.registerCommand('sonicpieditor.togglerecording', () => { 139 | isRecording = !isRecording; 140 | if (isRecording){ 141 | main.startRecording(); 142 | } 143 | else{ 144 | main.stopRecording(); 145 | vscode.window.showSaveDialog({filters: {'Wave file': ['wav']}}).then(uri => { 146 | if (uri){ 147 | main.saveRecording(uri.fsPath); 148 | } 149 | else{ 150 | main.deleteRecording(); 151 | } 152 | }); 153 | } 154 | }); 155 | 156 | context.subscriptions.push(disposable); 157 | } 158 | 159 | // this method is called when your extension is deactivated 160 | export function deactivate() {} 161 | -------------------------------------------------------------------------------- /src/main.ts: -------------------------------------------------------------------------------- 1 | // MIT License 2 | 3 | // Copyright (c) 2020 Luis Lloret 4 | 5 | // Permission is hereby granted, free of charge, to any person obtaining a copy 6 | // of this software and associated documentation files (the "Software"), to deal 7 | // in the Software without restriction, including without limitation the rights 8 | // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | // copies of the Software, and to permit persons to whom the Software is 10 | // furnished to do so, subject to the following conditions: 11 | 12 | // The above copyright notice and this permission notice shall be included in all 13 | // copies or substantial portions of the Software. 14 | 15 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | // SOFTWARE. 22 | 23 | import * as vscode from 'vscode'; 24 | import { TextDecoder } from 'util'; 25 | const fs = require('fs'); 26 | const os = require('os'); 27 | const child_process = require('child_process'); 28 | import { OscSender } from './oscsender'; 29 | const OSC = require('osc-js'); 30 | const utf8 = require('utf8'); 31 | const { v4: uuidv4 } = require('uuid'); 32 | import { Config } from './config'; 33 | import { Range, TextEditor, window } from 'vscode'; 34 | 35 | 36 | export class Main { 37 | rootPath: string; 38 | rubyPath: string; 39 | rubyServerPath: string; 40 | portDiscoveryPath: string; 41 | fetchUrlPath: string; 42 | samplePath: string; 43 | spUserPath: string; 44 | spUserTmpPath: string; 45 | logPath: string; 46 | serverErrorLogPath: string; 47 | serverOutputLogPath: string; 48 | guiLogPath: string; 49 | processLogPath: string; 50 | scsynthLogPath: string; 51 | initScriptPath: string; 52 | exitScriptPath: string; 53 | qtAppThemePath: string; 54 | qtBrowserDarkCss: string; 55 | qtBrowserLightCss: string; 56 | qtBrowserHcCss: string; 57 | 58 | guiSendToServerPort: number; 59 | guiListenToServerPort: number; 60 | serverListenToGuiPort: number; 61 | serverOscCuesPort: number; 62 | serverSendToHuiPort: number; 63 | scsynthPort: number; 64 | scsynthSendPort: number; 65 | erlangRouterPort: number; 66 | oscMidiOutPort: number; 67 | oscMidiInPort: number; 68 | websocketPort: number; 69 | 70 | logOutput: vscode.OutputChannel; 71 | cuesOutput: vscode.OutputChannel; 72 | 73 | oscSender: OscSender; 74 | 75 | serverStarted: boolean; 76 | 77 | platform: string; 78 | guiUuid: any; 79 | config: any; 80 | 81 | runOffset: number; 82 | 83 | errorHighlightDecorationType = vscode.window.createTextEditorDecorationType({ 84 | border: '2px solid red' 85 | }); 86 | 87 | 88 | constructor(){ 89 | // Set up path defaults based on platform 90 | this.platform = os.platform(); 91 | if (this.platform === 'win32'){ 92 | this.rootPath = "C:/Program Files/Sonic Pi"; 93 | this.rubyPath = this.rootPath + "/app/server/native/ruby/bin/ruby.exe"; 94 | } 95 | else if (this.platform === 'darwin'){ 96 | this.rootPath = "/Applications/Sonic Pi.app/Contents/Resources"; 97 | this.rubyPath = this.rootPath + "/app/server/native/ruby/bin/ruby"; 98 | } 99 | else{ 100 | this.rootPath = "/home/user/sonic-pi"; 101 | this.rubyPath = "ruby"; 102 | } 103 | this.config = new Config(); 104 | 105 | // Override default root path if found in settings 106 | if (this.config.sonicPiRootDirectory()){ 107 | this.rootPath = this.config.sonicPiRootDirectory(); 108 | } 109 | 110 | if (this.config.commandPath()){ 111 | this.rubyPath = this.config.commandPath(); 112 | } 113 | 114 | console.log('Using Sonic Pi root directory: ' + this.rootPath); 115 | console.log('Using ruby: ' + this.rubyPath); 116 | 117 | this.rubyServerPath = this.rootPath + "/app/server/ruby/bin/sonic-pi-server.rb"; 118 | this.portDiscoveryPath = this.rootPath + "/app/server/ruby/bin/port-discovery.rb"; 119 | this.fetchUrlPath = this.rootPath + "/app/server/ruby/bin/fetch-url.rb"; 120 | this.samplePath = this.rootPath + "/etc/samples"; 121 | this.spUserPath = this.sonicPiHomePath() + "/.sonic-pi"; 122 | this.spUserTmpPath = this.spUserPath + "/.writableTesterPath"; 123 | this.logPath = this.spUserPath + "/log"; 124 | this.serverErrorLogPath = this.logPath + "/server-errors.log"; 125 | this.serverOutputLogPath = this.logPath + "/server-output.log"; 126 | this.guiLogPath = this.logPath + "/gui.log"; 127 | this.processLogPath = this.logPath + "/processes.log"; 128 | this.scsynthLogPath = this.logPath + "/scsynth.log"; 129 | this.initScriptPath = this.rootPath + "/app/server/ruby/bin/init-script.rb"; 130 | this.exitScriptPath = this.rootPath + "/app/server/ruby/bin/exit-script.rb"; 131 | this.qtAppThemePath = this.rootPath + "/app/gui/qt/theme/app.qss"; 132 | this.qtBrowserDarkCss = this.rootPath + "/app/gui/qt/theme/dark/doc-styles.css"; 133 | this.qtBrowserLightCss = this.rootPath + "/app/gui/qt/theme/light/doc-styles.css"; 134 | this.qtBrowserHcCss = this.rootPath + "/app/gui/qt/theme/high_contrast/doc-styles.css"; 135 | 136 | this.guiSendToServerPort = -1; 137 | this.guiListenToServerPort = -1; 138 | this.serverListenToGuiPort = -1; 139 | this.serverOscCuesPort = -1; 140 | this.serverSendToHuiPort = -1; 141 | this.scsynthPort = -1; 142 | this.scsynthSendPort = -1; 143 | this.erlangRouterPort = -1; 144 | this.oscMidiOutPort = -1; 145 | this.oscMidiInPort = -1; 146 | this.websocketPort = -1; 147 | 148 | this.runOffset = 0; 149 | 150 | // attempt to create log directory 151 | if (!fs.existsSync(this.logPath)){ 152 | fs.mkdirSync(this.logPath, { recursive: true }); 153 | } 154 | 155 | this.cuesOutput = vscode.window.createOutputChannel('Cues'); 156 | this.logOutput = vscode.window.createOutputChannel('Log'); 157 | this.cuesOutput.show(); 158 | this.logOutput.show(); 159 | 160 | this.serverStarted = false; 161 | 162 | this.oscSender = new OscSender(); 163 | 164 | // create an uuid for the editor 165 | this.guiUuid = uuidv4(); 166 | 167 | // watch to see if the user opens a ruby or custom file and we need to start the server 168 | vscode.window.onDidChangeVisibleTextEditors((editors) => { 169 | let launchAuto = this.config.launchSonicPiServerAutomatically(); 170 | for (let i = 0; i < editors.length; i++){ 171 | if (launchAuto === 'ruby' && editors[i].document.languageId === 'ruby' && !this.serverStarted) { 172 | this.startServer(); 173 | break; 174 | } 175 | if (launchAuto === 'custom'){ 176 | let customExtension = this.config.launchSonicPiServerCustomExtension(); 177 | if (!customExtension){ 178 | vscode.window.showErrorMessage("Launch is set to custom, but custom extension is empty.", 179 | "Go to settings").then( 180 | item => { if (item) { 181 | vscode.commands.executeCommand('workbench.action.openSettings', 'sonicpieditor.launchSonicPiServerCustomExtension'); 182 | }}); 183 | } 184 | else if (editors[i].document.fileName.endsWith(customExtension) && !this.serverStarted) { 185 | this.startServer(); 186 | break; 187 | } 188 | } 189 | } 190 | }); 191 | 192 | // Update the mixer on the server if there are configuration changes 193 | vscode.workspace.onDidChangeConfiguration((event) => { 194 | if (event.affectsConfiguration('sonicpieditor')){ 195 | this.updateMixerSettings(); 196 | } 197 | }); 198 | } 199 | 200 | checkSonicPiPath() { 201 | if (!fs.existsSync(this.rubyServerPath)){ 202 | vscode.window.showErrorMessage("The Sonic Pi root path is not properly configured.", 203 | "Go to settings").then( 204 | item => { if (item) { 205 | vscode.commands.executeCommand('workbench.action.openSettings', 'sonicpieditor.sonicPiRootDirectory'); 206 | }}); 207 | } 208 | } 209 | 210 | sonicPiHomePath(){ 211 | return os.homedir(); 212 | } 213 | 214 | startServer(){ 215 | if (!this.serverStarted){ 216 | // Initialise the Sonic Pi server 217 | vscode.window.setStatusBarMessage("Starting Sonic Pi server"); 218 | vscode.window.showInformationMessage("Starting Sonic Pi server"); 219 | this.initAndCheckPorts(); 220 | this.setupOscReceiver(); 221 | this.startRubyServer(); 222 | this.serverStarted = true; 223 | } 224 | } 225 | 226 | log(str: string){ 227 | this.logOutput.appendLine(str); 228 | } 229 | 230 | cueLog(str: string){ 231 | this.cuesOutput.appendLine(str); 232 | } 233 | 234 | // This is where the incoming OSC messages are processed. 235 | // We are processing most of the incoming OSC messages, but not everything yet. 236 | setupOscReceiver(){ 237 | let osc = new OSC({ 238 | plugin: new OSC.DatagramPlugin({ open: { port: this.guiListenToServerPort, host: '127.0.0.1' } }) 239 | }); 240 | osc.open(); 241 | osc.on('/log/info', (message: { args: any; }) => { 242 | console.log("Got /log/info" + " -> " + message.args[0] + ", " + message.args[1]); 243 | this.log(message.args[1]); 244 | }); 245 | 246 | osc.on('/incoming/osc', (message: { args: any; }) => { 247 | console.log("Got /incoming/osc" + " -> " + message.args[0] + ", " + message.args[1] + ", " + 248 | message.args[2] + ", " + message.args[3]); 249 | this.cueLog(message.args[2] + ": " + message.args[3]); 250 | }); 251 | 252 | osc.on('/log/multi_message', (message: any) => { 253 | console.log("Got /log/multi_message"); 254 | this.processMultiMessage(message); 255 | }); 256 | 257 | osc.on('/syntax_error', (message: { args: any;}) => { 258 | console.log("Got /syntax_error" + message.args[0] + ", " + message.args[1] + ", " + 259 | message.args[2] + ", " + message.args[3] + ", " + message.args[4]); 260 | this.processSyntaxError(message); 261 | }); 262 | 263 | osc.on('/error', (message: any) => { 264 | console.log("Got /error"); 265 | this.processError(message); 266 | }); 267 | 268 | /* osc.on('*', (message: {address: string}) => { 269 | console.log("Got message of type: " + message.address); 270 | }); 271 | */ 272 | } 273 | 274 | // Show information about the syntax error to the user 275 | processSyntaxError(message: {args: any }){ 276 | let job_id = message.args[0]; 277 | let desc = message.args[1]; 278 | let error_line = message.args[2]; 279 | let line = message.args[3] + this.runOffset; 280 | 281 | vscode.window.showErrorMessage('Syntax error on job ' + job_id + ': ' + desc + '\nLine ' + line + ': ' + error_line, 'Goto error').then( 282 | item => { if (item) { 283 | let errorHighlight: vscode.DecorationOptions[] = []; 284 | let editor = vscode.window.activeTextEditor!; 285 | let range = editor.document.lineAt(line - 1).range; 286 | editor.selection = new vscode.Selection(range.start, range.end); 287 | editor.revealRange(range, vscode.TextEditorRevealType.InCenterIfOutsideViewport); 288 | errorHighlight.push({range}); 289 | editor.setDecorations(this.errorHighlightDecorationType, errorHighlight); 290 | }} 291 | ); 292 | } 293 | 294 | // Show information about the error to the user 295 | processError(message: {args: any }){ 296 | let job_id = message.args[0]; 297 | let desc = message.args[1]; 298 | let backtrace = message.args[2]; 299 | let line = message.args[3] + this.runOffset; 300 | 301 | vscode.window.showErrorMessage('Error on job ' + job_id + ': ' + desc + '\nLine ' + line + ': ' + backtrace, 'Goto error').then( 302 | item => { if (item) { 303 | let errorHighlight: vscode.DecorationOptions[] = []; 304 | let editor = vscode.window.activeTextEditor!; 305 | let range = editor.document.lineAt(line-1).range; 306 | editor.selection = new vscode.Selection(range.start, range.end); 307 | editor.revealRange(range, vscode.TextEditorRevealType.InCenterIfOutsideViewport); 308 | errorHighlight.push({range}); 309 | editor.setDecorations(this.errorHighlightDecorationType, errorHighlight); 310 | }} 311 | ); 312 | } 313 | 314 | 315 | // Process an incoming multi-message 316 | processMultiMessage(message: {args: any }){ 317 | let job_id = message.args[0]; 318 | let thread_name = message.args[1]; 319 | let runtime = message.args[2]; 320 | let count = message.args[3]; 321 | 322 | let toShow = "{run: " + job_id + ", time: " + runtime; 323 | if (thread_name){ 324 | toShow += ", thread: " + thread_name; 325 | } 326 | toShow += "}"; 327 | this.logOutput.appendLine(toShow); 328 | 329 | toShow = ""; 330 | for (let i = 0; i < count; i++){ 331 | //let type = message.args[4 + (2*i)]; 332 | let str = message.args[4 + 1 + (2*i)]; 333 | let lines = str.split(/\r?\n/); 334 | if (!str){ 335 | toShow = " |"; 336 | } 337 | else if (i === (count - 1)){ 338 | toShow = " └─ "; 339 | } 340 | else{ 341 | toShow = " ├─ "; 342 | } 343 | this.logOutput.append(toShow); 344 | 345 | lines.forEach((line: string) => { 346 | this.logOutput.appendLine(line); 347 | }); 348 | } 349 | } 350 | 351 | // This is where we see what ports to use, calling a ruby script 352 | initAndCheckPorts() { 353 | // Clear out old tasks from previous sessions if they still exist 354 | // in addtition to clearing out the logs 355 | this.log("[GUI] - Cleaning old sessions..."); 356 | child_process.spawnSync(this.rubyPath, [this.initScriptPath]); 357 | 358 | // Discover the port numbers 359 | let port_map = new Map(); 360 | this.log("[GUI] - Discovering port numbers..."); 361 | 362 | let determinePortNumbers = child_process.spawnSync(this.rubyPath, [this.portDiscoveryPath]); 363 | determinePortNumbers.output.forEach((item: any) => { 364 | let itemStr = this.ua82str(item); 365 | let port_strings = itemStr.split(/\r?\n/); 366 | port_strings.forEach((port_string) => { 367 | let tokens = port_string.split(':'); 368 | port_map.set(tokens[0], +tokens[1]); 369 | }); 370 | }); 371 | 372 | this.guiSendToServerPort = port_map.get("gui-send-to-server")!; 373 | this.guiListenToServerPort = port_map.get("gui-listen-to-server")!; 374 | this.serverListenToGuiPort = port_map.get("server-listen-to-gui")!; 375 | this.serverOscCuesPort = port_map.get("server-osc-cues")!; 376 | this.serverSendToHuiPort = port_map.get("server-send-to-gui")!; 377 | this.scsynthPort = port_map.get("scsynth")!; 378 | this.scsynthSendPort = port_map.get("scsynth-send")!; 379 | this.erlangRouterPort = port_map.get("erlang-router")!; 380 | this.oscMidiOutPort = port_map.get("osc-midi-out")!; 381 | this.oscMidiInPort = port_map.get("osc-midi-in")!; 382 | this.websocketPort = port_map.get("websocket")!; 383 | 384 | // FIXME: for now, we assume all ports are available. 385 | /* 386 | bool glts_available = checkPort(gui_listen_to_server_port); 387 | bool sltg_available = checkPort(server_listen_to_gui_port); 388 | bool soc_available = checkPort(server_osc_cues_port); 389 | bool s_available = checkPort(scsynth_port); 390 | bool sstg_available = checkPort(server_send_to_gui_port); 391 | bool gsts_available = checkPort(gui_send_to_server_port); 392 | bool ss_available = checkPort(scsynth_send_port); 393 | bool er_available = checkPort(erlang_router_port); 394 | bool omo_available = checkPort(osc_midi_out_port); 395 | bool omi_available = checkPort(osc_midi_in_port); 396 | bool ws_available = checkPort(websocket_port); 397 | if(!(glts_available && sltg_available && soc_available && s_available && sstg_available && gsts_available && ss_available && 398 | er_available && omo_available && omi_available && ws_available)){ 399 | std::cout << "[GUI] - Critical Error. One or more ports is not available." << std::endl; 400 | startupError("One or more ports is not available. Is Sonic Pi already running? If not, please reboot your machine and try again."); 401 | return false; 402 | 403 | } else { 404 | std::cout << "[GUI] - All ports OK" << std::endl; 405 | return true; 406 | } 407 | */ 408 | } 409 | 410 | // This is the main part of launching Sonic Pi's backend 411 | startRubyServer(){ 412 | let args = ["--enable-frozen-string-literal", "-E", "utf-8", this.rubyServerPath, "-u", 413 | this.serverListenToGuiPort, this.serverSendToHuiPort, this.scsynthPort, 414 | this.scsynthSendPort, this.serverOscCuesPort, this.erlangRouterPort, 415 | this.oscMidiOutPort, this.oscMidiInPort, this.websocketPort]; 416 | 417 | let ruby_server = child_process.spawn(this.rubyPath, args); 418 | ruby_server.stdout.on('data', (data: any) => { 419 | console.log(`stdout: ${data}`); 420 | this.log(`stdout: ${data}`); 421 | if (data.toString().match(/.*Sonic Pi Server successfully booted.*/)){ 422 | vscode.window.setStatusBarMessage("Sonic Pi server started"); 423 | vscode.window.showInformationMessage("Sonic Pi server started"); 424 | this.updateMixerSettings(); 425 | } 426 | }); 427 | 428 | ruby_server.stderr.on('data', (data: any) => { 429 | console.log(`stderr: ${data}`); 430 | this.log(`stderr: ${data}`); 431 | }); 432 | } 433 | 434 | updateMixerSettings(){ 435 | let invert_stereo = this.config.invertStereo(); 436 | let force_mono = this.config.forceMono(); 437 | if (invert_stereo) { 438 | this.mixerInvertStereo(); 439 | } else { 440 | this.mixerStandardStereo(); 441 | } 442 | 443 | if (force_mono) { 444 | this.mixerMonoMode(); 445 | } else { 446 | this.mixerStereoMode(); 447 | } 448 | } 449 | 450 | sendOsc(message: any){ 451 | this.oscSender.send(message); 452 | } 453 | 454 | runCode(code: string, offset: number = 0){ 455 | // The offset represents the line number of the selection, so we can apply it when we just send a 456 | // selection to Sonic Pi. If we send the full buffer, then this is 0. 457 | this.runOffset = offset; 458 | if (this.config.logClearOnRun()){ 459 | this.logOutput.clear(); 460 | } 461 | if (this.config.safeMode()){ 462 | code = "use_arg_checks true #__nosave__ set by Qt GUI user preferences.\n" + code ; 463 | } 464 | code = utf8.encode(code); 465 | this.clearErrorHighlight(); 466 | let message = new OSC.Message('/run-code', this.guiUuid, code); 467 | this.sendOsc(message); 468 | } 469 | 470 | flashCode(editor: TextEditor, isWhole: boolean) { 471 | const range = isWhole ? this.getWholeRange(editor) : this.getSelectedRange(editor); 472 | const flashDecorationType = window.createTextEditorDecorationType({ 473 | backgroundColor: this.config.flashBackgroundColor(), 474 | color: this.config.flashTextColor() 475 | }); 476 | editor.setDecorations(flashDecorationType, [range]); 477 | setTimeout(function () { 478 | flashDecorationType.dispose(); 479 | }, 250); 480 | } 481 | private getWholeRange(editor: TextEditor): Range { 482 | let startPos = editor.document.positionAt(0); 483 | let endPos = editor.document.positionAt(editor.document.getText().length - 1); 484 | return new Range(startPos, endPos); 485 | } 486 | private getSelectedRange(editor: TextEditor): Range { 487 | return new Range(editor.selection.anchor, editor.selection.active); 488 | } 489 | 490 | stopAllJobs() { 491 | var message = new OSC.Message('/stop-all-jobs', this.guiUuid); 492 | this.sendOsc(message); 493 | } 494 | 495 | startRecording(){ 496 | let message = new OSC.Message('/start-recording', this.guiUuid); 497 | this.sendOsc(message); 498 | } 499 | 500 | stopRecording(){ 501 | let message = new OSC.Message('/stop-recording', this.guiUuid); 502 | this.sendOsc(message); 503 | } 504 | 505 | saveRecording(path: string){ 506 | let message = new OSC.Message('/save-recording', this.guiUuid, path); 507 | this.sendOsc(message); 508 | } 509 | 510 | deleteRecording(){ 511 | let message = new OSC.Message('/delete-recording', this.guiUuid); 512 | this.sendOsc(message); 513 | } 514 | 515 | mixerInvertStereo(){ 516 | let message = new OSC.Message('/mixer-invert-stereo', this.guiUuid); 517 | this.sendOsc(message); 518 | } 519 | 520 | mixerStandardStereo(){ 521 | let message = new OSC.Message('/mixer-standard-stereo', this.guiUuid); 522 | this.sendOsc(message); 523 | } 524 | 525 | mixerMonoMode(){ 526 | let message = new OSC.Message('/mixer-mono-mode', this.guiUuid); 527 | this.sendOsc(message); 528 | } 529 | 530 | mixerStereoMode(){ 531 | let message = new OSC.Message('/mixer-stereo-mode', this.guiUuid); 532 | this.sendOsc(message); 533 | } 534 | 535 | // Remove the error highlight 536 | clearErrorHighlight(){ 537 | vscode.window.activeTextEditor?.setDecorations(this.errorHighlightDecorationType, []); 538 | } 539 | 540 | // Convert a uint array to a string 541 | ua82str(buf: Uint8Array): string { 542 | if (!buf){ 543 | return ""; 544 | } 545 | let str = new TextDecoder().decode(buf); 546 | return str; 547 | } 548 | 549 | 550 | } -------------------------------------------------------------------------------- /src/oscsender.ts: -------------------------------------------------------------------------------- 1 | // MIT License 2 | 3 | // Copyright (c) 2020 Luis Lloret 4 | 5 | // Permission is hereby granted, free of charge, to any person obtaining a copy 6 | // of this software and associated documentation files (the "Software"), to deal 7 | // in the Software without restriction, including without limitation the rights 8 | // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | // copies of the Software, and to permit persons to whom the Software is 10 | // furnished to do so, subject to the following conditions: 11 | 12 | // The above copyright notice and this permission notice shall be included in all 13 | // copies or substantial portions of the Software. 14 | 15 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | // SOFTWARE. 22 | 23 | // Very simple OSC send wrapper. At the moment the send port is harcoded, but we already know the 24 | // port where this is, since we are calling initAndCheckPorts(). Use that! 25 | // The OSC receiving handling is in main.ts. 26 | const OSC = require('osc-js'); 27 | 28 | export class OscSender { 29 | private osc: any; 30 | constructor(){ 31 | this.osc = new OSC({ 32 | plugin: new OSC.DatagramPlugin({ send: { port: 51235, host: '127.0.0.1' } }) 33 | }); 34 | this.osc.open(); 35 | } 36 | 37 | send(msg: any){ 38 | this.osc.send(msg); 39 | } 40 | } 41 | -------------------------------------------------------------------------------- /src/test/runTest.ts: -------------------------------------------------------------------------------- 1 | import * as path from 'path'; 2 | 3 | import { runTests } from 'vscode-test'; 4 | 5 | async function main() { 6 | try { 7 | // The folder containing the Extension Manifest package.json 8 | // Passed to `--extensionDevelopmentPath` 9 | const extensionDevelopmentPath = path.resolve(__dirname, '../../'); 10 | 11 | // The path to test runner 12 | // Passed to --extensionTestsPath 13 | const extensionTestsPath = path.resolve(__dirname, './suite/index'); 14 | 15 | // Download VS Code, unzip it and run the integration test 16 | await runTests({ extensionDevelopmentPath, extensionTestsPath }); 17 | } catch (err) { 18 | console.error('Failed to run tests'); 19 | process.exit(1); 20 | } 21 | } 22 | 23 | main(); 24 | -------------------------------------------------------------------------------- /src/test/suite/extension.test.ts: -------------------------------------------------------------------------------- 1 | import * as assert from 'assert'; 2 | 3 | // You can import and use all API from the 'vscode' module 4 | // as well as import your extension to test it 5 | import * as vscode from 'vscode'; 6 | // import * as myExtension from '../../extension'; 7 | 8 | suite('Extension Test Suite', () => { 9 | vscode.window.showInformationMessage('Start all tests.'); 10 | 11 | test('Sample test', () => { 12 | assert.equal(-1, [1, 2, 3].indexOf(5)); 13 | assert.equal(-1, [1, 2, 3].indexOf(0)); 14 | }); 15 | }); 16 | -------------------------------------------------------------------------------- /src/test/suite/index.ts: -------------------------------------------------------------------------------- 1 | import * as path from 'path'; 2 | import * as Mocha from 'mocha'; 3 | import * as glob from 'glob'; 4 | 5 | export function run(): Promise { 6 | // Create the mocha test 7 | const mocha = new Mocha({ 8 | ui: 'tdd', 9 | color: true 10 | }); 11 | 12 | const testsRoot = path.resolve(__dirname, '..'); 13 | 14 | return new Promise((c, e) => { 15 | glob('**/**.test.js', { cwd: testsRoot }, (err, files) => { 16 | if (err) { 17 | return e(err); 18 | } 19 | 20 | // Add files to the test suite 21 | files.forEach(f => mocha.addFile(path.resolve(testsRoot, f))); 22 | 23 | try { 24 | // Run the mocha test 25 | mocha.run(failures => { 26 | if (failures > 0) { 27 | e(new Error(`${failures} tests failed.`)); 28 | } else { 29 | c(); 30 | } 31 | }); 32 | } catch (err) { 33 | console.error(err); 34 | e(err); 35 | } 36 | }); 37 | }); 38 | } 39 | -------------------------------------------------------------------------------- /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 | "strict": true /* enable all strict type-checking options */ 12 | /* Additional Checks */ 13 | // "noImplicitReturns": true, /* Report error when not all code paths in function return a value. */ 14 | // "noFallthroughCasesInSwitch": true, /* Report errors for fallthrough cases in switch statement. */ 15 | // "noUnusedParameters": true, /* Report errors on unused parameters. */ 16 | }, 17 | "exclude": [ 18 | "node_modules", 19 | ".vscode-test" 20 | ] 21 | } 22 | -------------------------------------------------------------------------------- /vsc-extension-quickstart.md: -------------------------------------------------------------------------------- 1 | # Welcome to your VS Code Extension 2 | 3 | ## What's in the folder 4 | 5 | * This folder contains all of the files necessary for your extension. 6 | * `package.json` - this is the manifest file in which you declare your extension and command. 7 | * The sample plugin registers a command and defines its title and command name. With this information VS Code can show the command in the command palette. It doesn’t yet need to load the plugin. 8 | * `src/extension.ts` - this is the main file where you will provide the implementation of your command. 9 | * The file exports one function, `activate`, which is called the very first time your extension is activated (in this case by executing the command). Inside the `activate` function we call `registerCommand`. 10 | * We pass the function containing the implementation of the command as the second parameter to `registerCommand`. 11 | 12 | ## Get up and running straight away 13 | 14 | * Press `F5` to open a new window with your extension loaded. 15 | * Run your command from the command palette by pressing (`Ctrl+Shift+P` or `Cmd+Shift+P` on Mac) and typing `Hello World`. 16 | * Set breakpoints in your code inside `src/extension.ts` to debug your extension. 17 | * Find output from your extension in the debug console. 18 | 19 | ## Make changes 20 | 21 | * You can relaunch the extension from the debug toolbar after changing code in `src/extension.ts`. 22 | * You can also reload (`Ctrl+R` or `Cmd+R` on Mac) the VS Code window with your extension to load your changes. 23 | 24 | 25 | ## Explore the API 26 | 27 | * You can open the full set of our API when you open the file `node_modules/@types/vscode/index.d.ts`. 28 | 29 | ## Run tests 30 | 31 | * Open the debug viewlet (`Ctrl+Shift+D` or `Cmd+Shift+D` on Mac) and from the launch configuration dropdown pick `Extension Tests`. 32 | * Press `F5` to run the tests in a new window with your extension loaded. 33 | * See the output of the test result in the debug console. 34 | * Make changes to `src/test/suite/extension.test.ts` or create new test files inside the `test/suite` folder. 35 | * The provided test runner will only consider files matching the name pattern `**.test.ts`. 36 | * You can create folders inside the `test` folder to structure your tests any way you want. 37 | 38 | ## Go further 39 | 40 | * Reduce the extension size and improve the startup time by [bundling your extension](https://code.visualstudio.com/api/working-with-extensions/bundling-extension). 41 | * [Publish your extension](https://code.visualstudio.com/api/working-with-extensions/publishing-extension) on the VSCode extension marketplace. 42 | * Automate builds by setting up [Continuous Integration](https://code.visualstudio.com/api/working-with-extensions/continuous-integration). 43 | --------------------------------------------------------------------------------