├── docs
├── 0.png
├── 1-1.png
├── 2-1.png
├── 2-2.png
├── 2-3.png
├── 2-4.png
├── 3-1.png
├── 3-2.png
├── 4-1.png
├── 4-2.png
├── 4-3.png
├── 4-4.png
├── 5-2.png
├── 5-3.png
├── 5-1-0.png
├── 5-1-1.png
└── features.md
├── screenshot.png
├── resources
├── icon.png
├── playground.png
├── add.svg
├── runTask.svg
├── add-black.svg
├── runTask-black.svg
├── apps.svg
├── apps-black.svg
├── play-black.svg
├── play.svg
├── stopProcess.svg
├── purgeChain.svg
├── stopProcess-black.svg
├── purgeChain-black.svg
├── check-black.svg
├── check.svg
├── home.svg
├── home-black.svg
├── book.svg
├── book-black.svg
├── startNode.svg
├── startNode-black.svg
├── compileStartNode.svg
├── compileStartNode-black.svg
├── gear.svg
├── gear-black.svg
├── substrate.svg
├── github.svg
├── github-black.svg
└── playground.png.base64
├── .gitignore
├── src
├── constants.ts
├── views
│ ├── marketplace
│ │ ├── types.ts
│ │ ├── substrateDeps.ts
│ │ ├── fetchCategories.ts
│ │ └── MarketplaceProvider.ts
│ ├── processes
│ │ └── ProcessesProvider.ts
│ ├── nodes
│ │ └── NodesProvider.ts
│ ├── accounts
│ │ └── AccountsProvider.ts
│ └── contracts
│ │ └── ContractsProvider.ts
├── common
│ ├── TreeDataProvider.ts
│ └── Substrate.ts
├── runtimes
│ ├── parseDepsFromFile.ts
│ └── Runtime.ts
├── processes
│ └── Processes.ts
├── extension.ts
├── util.ts
└── nodes
│ └── Nodes.ts
├── .vscode
├── extensions.json
├── tasks.json
├── settings.json
└── launch.json
├── .vscodeignore
├── .eslintrc.json
├── tsconfig.json
├── README.md
├── .github
└── workflows
│ └── release.yml
├── CHANGELOG.md
├── package.json
└── yarn.lock
/docs/0.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/0.png
--------------------------------------------------------------------------------
/docs/1-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/1-1.png
--------------------------------------------------------------------------------
/docs/2-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/2-1.png
--------------------------------------------------------------------------------
/docs/2-2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/2-2.png
--------------------------------------------------------------------------------
/docs/2-3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/2-3.png
--------------------------------------------------------------------------------
/docs/2-4.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/2-4.png
--------------------------------------------------------------------------------
/docs/3-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/3-1.png
--------------------------------------------------------------------------------
/docs/3-2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/3-2.png
--------------------------------------------------------------------------------
/docs/4-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/4-1.png
--------------------------------------------------------------------------------
/docs/4-2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/4-2.png
--------------------------------------------------------------------------------
/docs/4-3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/4-3.png
--------------------------------------------------------------------------------
/docs/4-4.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/4-4.png
--------------------------------------------------------------------------------
/docs/5-2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/5-2.png
--------------------------------------------------------------------------------
/docs/5-3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/5-3.png
--------------------------------------------------------------------------------
/docs/5-1-0.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/5-1-0.png
--------------------------------------------------------------------------------
/docs/5-1-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/docs/5-1-1.png
--------------------------------------------------------------------------------
/screenshot.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/screenshot.png
--------------------------------------------------------------------------------
/resources/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/resources/icon.png
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | scratchpad
2 | out
3 | node_modules
4 | .vscode-test/
5 | *.vsix
6 | vsc-extension-quickstart.md
--------------------------------------------------------------------------------
/resources/playground.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/paritytech/vscode-substrate/HEAD/resources/playground.png
--------------------------------------------------------------------------------
/src/constants.ts:
--------------------------------------------------------------------------------
1 | // Source for the marketplace treeview
2 | export const GRAPHQL_API_ENDPOINT = 'https://marketplace-api-staging.substrate.dev/graphql';
--------------------------------------------------------------------------------
/resources/add.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/runTask.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/add-black.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/runTask-black.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/.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 |
--------------------------------------------------------------------------------
/.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 | scratchpad
12 | *.vsix
--------------------------------------------------------------------------------
/resources/apps.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/apps-black.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/play-black.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/play.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/stopProcess.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/purgeChain.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/stopProcess-black.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/purgeChain-black.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/check-black.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/check.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/src/views/marketplace/types.ts:
--------------------------------------------------------------------------------
1 | export interface Category {
2 | category: string;
3 | pallets: Pallet[];
4 | }
5 |
6 | export interface Pallet {
7 | id: number;
8 | name: string;
9 | description: string;
10 | homepage: string;
11 | github: string;
12 | documentation: string;
13 | icon: string;
14 | version: string;
15 | }
--------------------------------------------------------------------------------
/.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 |
--------------------------------------------------------------------------------
/.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 | }
--------------------------------------------------------------------------------
/.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 | "semi": "off"
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/resources/home.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/home-black.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/book.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/book-black.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/src/common/TreeDataProvider.ts:
--------------------------------------------------------------------------------
1 | import * as vscode from 'vscode';
2 |
3 | export abstract class TreeDataProvider implements vscode.TreeDataProvider {
4 | protected _onDidChangeTreeData: vscode.EventEmitter = new vscode.EventEmitter();
5 | readonly onDidChangeTreeData: vscode.Event = this._onDidChangeTreeData.event;
6 |
7 | public refresh(): void {
8 | this._onDidChangeTreeData.fire();
9 | }
10 |
11 | public getTreeItem(element: T): vscode.TreeItem {
12 | return element;
13 | }
14 |
15 | abstract getChildren(element?: T): Thenable;
16 | }
17 |
--------------------------------------------------------------------------------
/resources/startNode.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/src/runtimes/parseDepsFromFile.ts:
--------------------------------------------------------------------------------
1 | const fs = require('fs');
2 | const uniq = require('lodash/fp/uniq');
3 | const matchAll = require('string.prototype.matchall')
4 |
5 | /**
6 | * Given the contents of a Cargo.toml, extract the list of pallet names.
7 | */
8 | function parseDeps(text: string) : string[] {
9 | return uniq(Array.from(matchAll(text, /(pallet-[a-z-A-Z0-9-]+)/g)).map((match: any) => match[1]));
10 | }
11 |
12 | /**
13 | * Given a path to a Cargo.toml, extract the list of pallet names.
14 | */
15 | export default function parseDepsFromFile(file: string) {
16 | return parseDeps(fs.readFileSync(file).toString());
17 | }
--------------------------------------------------------------------------------
/resources/startNode-black.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/compileStartNode.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/compileStartNode-black.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/resources/gear.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/gear-black.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "module": "commonjs",
4 | "target": "es6",
5 | "outDir": "out",
6 | "lib": [
7 | "es2019"
8 | ],
9 | "esModuleInterop": true,
10 | "sourceMap": true,
11 | "rootDir": "src",
12 | "strict": true /* enable all strict type-checking options */
13 | /* Additional Checks */
14 | // "noImplicitReturns": true, /* Report error when not all code paths in function return a value. */
15 | // "noFallthroughCasesInSwitch": true, /* Report errors for fallthrough cases in switch statement. */
16 | // "noUnusedParameters": true, /* Report errors on unused parameters. */
17 | },
18 | "exclude": [
19 | "node_modules",
20 | ".vscode-test"
21 | ]
22 | }
23 |
--------------------------------------------------------------------------------
/src/processes/Processes.ts:
--------------------------------------------------------------------------------
1 | import { BehaviorSubject} from 'rxjs';
2 | import * as vscode from 'vscode';
3 |
4 | export type Process = {nodePath: string; term: vscode.Terminal; command: string; termCloseHandlerDispose: any;}
5 |
6 | // Manages Substrate processes spawned through the extension
7 | export default class Processes {
8 |
9 | processes$: BehaviorSubject = new BehaviorSubject([] as Process[]);
10 |
11 | new(process: Process) {
12 | this.processes$.next(this.processes$.getValue().concat([process]));
13 | process.termCloseHandlerDispose = vscode.window.onDidCloseTerminal(t => {
14 | if (t === process.term) {
15 | process.termCloseHandlerDispose.dispose();
16 | console.log('Terminal closed; removing process.');
17 | this.del(process);
18 | }
19 | });
20 | }
21 |
22 | del(process: Process) {
23 | this.processes$.next(this.processes$.getValue().filter(p => p != process));
24 | }
25 | }
--------------------------------------------------------------------------------
/.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 |
--------------------------------------------------------------------------------
/src/extension.ts:
--------------------------------------------------------------------------------
1 | import 'array-flat-polyfill';
2 | import * as vscode from 'vscode';
3 | import { Substrate } from './common/Substrate';
4 | import Nodes from './nodes/Nodes';
5 | import Processes from './processes/Processes';
6 | import { setupAccountsTreeView } from './views/accounts/AccountsProvider';
7 | import { setupContractsTreeView } from './views/contracts/ContractsProvider';
8 | import { setUpMarketplaceTreeView } from './views/marketplace/MarketplaceProvider';
9 | import { setUpNodesTreeView } from './views/nodes/NodesProvider';
10 | import { setupProcessesTreeView } from './views/processes/ProcessesProvider';
11 |
12 | export async function activate(context: vscode.ExtensionContext) {
13 | const substrate = new Substrate(context)
14 |
15 | const nodes = new Nodes();
16 | const processes = new Processes();
17 |
18 | // Set up runtimes
19 | const { selectedNode$ } = setUpNodesTreeView(nodes, processes);
20 |
21 | // Set up marketplace
22 | setUpMarketplaceTreeView(nodes, selectedNode$);
23 |
24 | // Set up processes
25 | const selectedProcess$ = setupProcessesTreeView(substrate, processes);
26 |
27 | // Set up accounts
28 | setupAccountsTreeView(substrate, context);
29 |
30 | // Set up contracts
31 | setupContractsTreeView(substrate, selectedProcess$, context);
32 | }
33 |
34 | export function deactivate() { }
--------------------------------------------------------------------------------
/src/runtimes/Runtime.ts:
--------------------------------------------------------------------------------
1 | import * as vscode from 'vscode';
2 | import parseDepsFromFile from './parseDepsFromFile';
3 | import { BehaviorSubject } from 'rxjs';
4 | import { startWith, distinctUntilChanged, map } from 'rxjs/operators';
5 | import { vscToObservable } from '../util';
6 | const isEqual = require('lodash/fp/isEqual');
7 | const path = require('path');
8 |
9 | // Manages the auto-detection of a runtime's dependencies
10 | export default class Runtime {
11 |
12 | deps$: BehaviorSubject;
13 |
14 | _watcher: vscode.FileSystemWatcher;
15 |
16 | constructor(public runtimePath: string) {
17 | this._watcher = vscode.workspace.createFileSystemWatcher(
18 | path.join(runtimePath,'Cargo.toml'), true, false, true
19 | );
20 | // Note that moving Cargo.toml around (e.g. removing/adding) will mess up the events.
21 | // TODO listen to all three events? Probably sounder.
22 | const fileChange$ = vscToObservable(this._watcher.onDidChange.bind(this._watcher));
23 |
24 | this.deps$ = new BehaviorSubject([] as string[]);
25 | fileChange$
26 | .pipe(
27 | startWith(null),
28 | map(() => parseDepsFromFile(path.join(runtimePath,'Cargo.toml'))),
29 | distinctUntilChanged(isEqual)
30 | ).subscribe(this.deps$);
31 | }
32 |
33 | dispose() {
34 | this._watcher.dispose(); // can Runtime still be garbage collected?
35 | // todo use setinterval unref to check if old runtime gets gc'd
36 | }
37 | }
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## DEPRECATION NOTICE
2 |
3 | This repo is not supported anymore. There are no replacement.
4 |
5 | # Substrate - VSCode extension
6 |
7 | Manage your node, runtime, browse and install pallets, manage your accounts and your smart contracts within VSCode.
8 |
9 | A complete list of features and walkthrough is available in [this document](./docs/features.md).
10 |
11 | Visit the extension's page on Visual Studio Marketplace: https://marketplace.visualstudio.com/items?itemName=paritytech.vscode-substrate
12 |
13 | 
14 |
15 | This extension is work in progress and is in active development. Please [report](https://github.com/paritytech/vscode-substrate/issues/new) any issues you encounter.
16 |
17 | ## Installation
18 |
19 | Launch VS Code Quick Open (Ctrl+P), paste the following command, and press enter.
20 |
21 | ```
22 | ext install paritytech.vscode-substrate
23 | ```
24 |
25 | Alternatively, search for "VSCode Substrate" in the extension marketplace.
26 |
27 | ## Development
28 |
29 | Clone, run `yarn` to install the dependencies, open the folder in VSCode, run the command "Debug: Start Debugging" to launch a new VSCode instance with the extension running.
30 |
31 | ## Publish
32 |
33 | Bump the version in `package.json`, tag your commit with the version number (`vX.X.X`) and push. This will trigger the GitHub Action and publish the new version on VSCode Marketplace as well as create a new release on GitHub. Release information can be edited manually and added to CHANGELOG.md
34 |
--------------------------------------------------------------------------------
/src/views/marketplace/substrateDeps.ts:
--------------------------------------------------------------------------------
1 |
2 | const child_process = require('child_process');
3 | import * as vscode from 'vscode';
4 | import { resolveWhenTerminalClosed } from '../../util';
5 |
6 | /**
7 | * Ensures that substrate-deps is installed; installs it otherwise.
8 | *
9 | * @return A promise that always resolves; either with true in case substrate-deps
10 | * is or got installed; false otherwise.
11 | */
12 | export async function substrateDepsInstalled(): Promise {
13 | try {
14 | child_process.execSync('substrate-deps --version');
15 | return true;
16 | } catch (e) {
17 | const clicked = await vscode.window.showWarningMessage(
18 | `substrate-deps is required for this extension to work but doesn't seem to be installed. Install it? This will run the following command: 'cargo install substrate-deps'`,
19 | { modal: true },
20 | 'Yes'
21 | );
22 |
23 | if (clicked !== 'Yes') {
24 | return false;
25 | }
26 |
27 | const term = vscode.window.createTerminal('Installing substrate-deps');
28 | term.sendText('cargo install substrate-deps && exit');
29 | term.show();
30 |
31 | await resolveWhenTerminalClosed(term);
32 |
33 | if (term.exitStatus?.code === 0) {
34 | vscode.window.showInformationMessage(`substrate-deps was successfully installed.`);
35 | return substrateDepsInstalled();
36 | } else {
37 | console.error('Terminal is closed; should have exit status; qed.')
38 | return false;
39 | }
40 |
41 | // TODO if users runs this function again with a terminal already running, we have two listeners, this is bad.
42 | }
43 | }
--------------------------------------------------------------------------------
/resources/substrate.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/.github/workflows/release.yml:
--------------------------------------------------------------------------------
1 | on:
2 | push:
3 | # Sequence of patterns matched against refs/tags
4 | tags:
5 | - 'v[0-9]+.[0-9]+.[0-9]+'
6 |
7 | name: Upload Release Asset
8 |
9 | jobs:
10 | build:
11 | name: Upload Release Asset
12 | runs-on: ubuntu-latest
13 | steps:
14 | - name: Checkout code
15 | uses: actions/checkout@v2
16 | - name: Use Node.js 12.8.1
17 | uses: actions/setup-node@v1
18 | with:
19 | node-version: '12.8.1'
20 | - name: Build extension package
21 | run: |
22 | npm install
23 | npx vsce package -o vscode-substrate.vsix
24 | - name: Publish Extension to Visual Studio Marketplace
25 | run: npx vsce publish --pat ${{ secrets.MARKETPLACE_TOKEN }} --packagePath ./vscode-substrate.vsix
26 | - name: Create GitHub Release
27 | id: create_release
28 | uses: actions/create-release@v1
29 | env:
30 | GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
31 | with:
32 | tag_name: ${{ github.ref }}
33 | release_name: ${{ github.ref }}
34 | draft: false
35 | prerelease: false
36 | - name: Upload Release Asset
37 | id: upload-release-asset
38 | uses: actions/upload-release-asset@v1
39 | env:
40 | GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
41 | with:
42 | upload_url: ${{ steps.create_release.outputs.upload_url }} # This pulls from the CREATE RELEASE step above, referencing its ID to get its outputs object, which include a `upload_url`. See this blog post for more info: https://jasonet.co/posts/new-features-of-github-actions/#passing-data-to-future-steps
43 | asset_path: ./vscode-substrate.vsix
44 | asset_name: vscode-substrate.vsix
45 | asset_content_type: application/vsix
--------------------------------------------------------------------------------
/src/util.ts:
--------------------------------------------------------------------------------
1 | import * as vscode from 'vscode';
2 | import { Observable } from 'rxjs';
3 |
4 | const path = require('path');
5 |
6 | export function resolveWhenTerminalClosed(term: vscode.Terminal): Promise {
7 | return new Promise((resolve, reject) => {
8 | const disp = vscode.window.onDidCloseTerminal(t => {
9 | if (t === term) {
10 | disp.dispose();
11 | resolve();
12 | }
13 | });
14 | });
15 | }
16 |
17 | export function vscToObservable(fn: (arg0: ((x: T) => any)) => any): Observable {
18 | return new Observable((subscriber) => {
19 | return fn((x: T) => subscriber.next(x));
20 | });
21 | }
22 |
23 | // Convert a full path to a human-friendly shorter path to display
24 | export function tryShortname(fullPath: string) {
25 | const fsPaths = vscode.workspace.workspaceFolders?.map((x) => x.uri.fsPath) || [];
26 | if (fsPaths.length !== 1) return fullPath;
27 | return path.relative(fsPaths[0], fullPath);
28 | }
29 |
30 | export async function showInputBoxValidate(options: vscode.InputBoxOptions, validateFn: (x: any) => Promise) {
31 | do {
32 | const a = await vscode.window.showInputBox(options);
33 | if (a === undefined)
34 | return a;
35 | else {
36 | let err = await validateFn(a);
37 | if (err !== '')
38 | vscode.window.showErrorMessage(err);
39 | else
40 | return a;
41 | }
42 | } while (true);
43 | }
44 |
45 | const INSTANCE = process.env.SUBSTRATE_PLAYGROUND_INSTANCE;
46 | const HOST = process.env.SUBSTRATE_PLAYGROUND_HOSTNAME;
47 |
48 | export function wsEndpointFromCommand(command: string) {
49 | const port = command.match(/--ws-port[ =]\d+/)?.[0] || '9944';
50 | const wsEndpoint = INSTANCE ? `wss://${INSTANCE}.${HOST}${port !== '9944' ? `:${port}` : ''}/wss` : `ws://127.0.0.1:${port}/`;
51 | return wsEndpoint;
52 | }
--------------------------------------------------------------------------------
/src/views/marketplace/fetchCategories.ts:
--------------------------------------------------------------------------------
1 | import fetch from 'node-fetch';
2 | const flow = require('lodash/fp/flow');
3 | const get = require('lodash/fp/get');
4 | const invoke = require('lodash/fp/invoke');
5 | const isEqual = require('lodash/fp/isEqual');
6 | const map = require('lodash/fp/map');
7 | const remove = require('lodash/fp/remove');
8 |
9 | import { Category, Pallet } from './types';
10 | import { GRAPHQL_API_ENDPOINT } from '../../constants';
11 |
12 | /**
13 | * Executes the given GraphQL query on the Marketplace endpoint, returns
14 | * the deserialized response value
15 | *
16 | * @returns {Promise} A resolved promise with the deserialized response in
17 | * case of success; a rejected promise with the error message in case the
18 | * request failed or the response payload indicates an error.
19 | */
20 | const gq = (query: string) => (variables?: object): Promise =>
21 | fetch(
22 | GRAPHQL_API_ENDPOINT,
23 | {
24 | method: 'POST',
25 | headers: {
26 | 'Accept': 'application/json',
27 | 'Content-Type': 'application/json'
28 | },
29 | body: JSON.stringify({
30 | query,
31 | variables
32 | })
33 | }
34 | )
35 | .then(invoke('json'))
36 | .then((d: any) => d.data || Promise.reject(map('message')(d.errors)));
37 |
38 | /**
39 | * Fetches all the pallets for a given Marketplace category.
40 | */
41 | const fetchPallets = (category: string): Promise =>
42 | gq(`query($category: String!){
43 | search(type: PALLET query: "" category: $category) {
44 | results {
45 | ...on Pallet {
46 | id
47 | name
48 | description
49 | documentation
50 | homepage
51 | github: repository
52 | icon
53 | version
54 | repository
55 | }
56 | }
57 | }
58 | }`)({ category })
59 | .then(
60 | get('search.results'),
61 | );
62 |
63 | /**
64 | * Fetches a category with all its pallets.
65 | */
66 | const fetchCategory = (category: string): Promise =>
67 | fetchPallets(category).then((pallets: Pallet[]) => ({ category, pallets }))
68 |
69 | /**
70 | * Fetches the list of all categories and their pallets.
71 | */
72 | const fetchCategories = (): Promise =>
73 | gq('{ marketplaceCategories(type:PALLET) {name} }')()
74 | .then(flow(
75 | get('marketplaceCategories'),
76 | map('name'),
77 | remove(isEqual('tutorial')),
78 | map(fetchCategory),
79 | Promise.all.bind(Promise)
80 | ));
81 | export default fetchCategories;
--------------------------------------------------------------------------------
/CHANGELOG.md:
--------------------------------------------------------------------------------
1 | # Changelog
2 |
3 | ## v0.5.1
4 | * Remove Getting Started (Substrate Playground)
5 | * Remove Tasks
6 | * Remove built-in default process
7 |
8 | ## v0.5.0
9 | * Smart contracts and accounts support
10 |
11 | ## v0.4.11
12 | * "Purge chain" will run the binary directly (instead of potentially recompiling); same for "Start node".
13 | * Add command "Compile & start node" that behaves like "Start node" previously, i.e. using "cargo run".
14 |
15 | ## v0.4.10
16 | * Playground: Fix ws endpoint for apps
17 |
18 | ## v0.4.9
19 | * Add "Getting started" command
20 |
21 | ## v0.4.8
22 | * Fix icons for light theme
23 | * Always show marketplace treeview, even when no node is selected
24 | * Playground: don't show Getting Started when a file is already opened
25 |
26 | ## v0.4.7
27 | * Playground: fix for Workspace tasks.
28 |
29 | ## v0.4.6
30 | * Workspace tasks (tasks defined by the template) are now listed in a treeview
31 |
32 | ## v0.4.5
33 | * Playground: add Polkadot Apps link
34 | * Ask user for flags when launching node
35 |
36 | ## v0.4.4
37 | * List running nodes
38 |
39 | ## v0.4.3
40 | * The start node command now compiles in release mode.
41 |
42 | ## v0.4.2
43 | * Retain Getting started view in cache when the tab loses focus
44 |
45 | ## v0.4.1
46 | * Lint
47 |
48 | ## v0.4.0
49 | * Reorganized UI : now shows the list of nodes in the workspace. The marketplace is based on the runtime of the selected node.
50 | * Node commands accessible in the Command Palette.
51 |
52 | ## v0.3.4
53 | * Fix commands "Start node" and "Purge chain"
54 | * Remove playground-specific commands
55 |
56 | ## v0.3.2
57 | * Fix Playground detection
58 | * Fix pallets GitHub target
59 |
60 | ## v0.3.1
61 | * Marketplace is now loaded on startup
62 | * Selected runtime now highlighted
63 | * Fix parsing runtime pallets
64 |
65 | ## v0.3.0
66 | * Support multiple nodes
67 | * Execute operations on the selected node
68 | * Runtime is selected by user and isn't based on the active editor's path any longer.
69 |
70 | ## v0.2.0
71 | * Support multiple runtimes in the same workspace. Your active editor will determine the runtime used by the extension.
72 | * Display installed pallets.
73 | * Removed setting `substrateMarketplace.runtimeManifestPath` & removed Substrate Playground hardcoded path. Runtimes are now detected and updated automatically.
74 |
75 | ## v0.1.1
76 | * Your project's runtime manifest location can now be provided using the `substrateMarketplace.runtimeManifestPath` setting.
77 | * If this setting is not provided, the extension will try to use `./runtime/Cargo.toml` or `./Cargo.toml`.
--------------------------------------------------------------------------------
/src/nodes/Nodes.ts:
--------------------------------------------------------------------------------
1 | import { BehaviorSubject, merge, of } from 'rxjs';
2 | import { map, tap, throttleTime } from 'rxjs/operators';
3 | import * as vscode from 'vscode';
4 | import parseDepsFromFile from '../runtimes/parseDepsFromFile';
5 | import { vscToObservable } from '../util';
6 |
7 | const path = require('path');
8 | const fs = require('fs');
9 | const glob = require('glob');
10 |
11 | /**
12 | * Given a folder path, return a list of substrate nodes contained inside.
13 | */
14 | const findNodesInFolder = (roots: string): string[] => {
15 | // TODO use vscode.workspace.findFiles instead
16 | return glob.sync(path.join(roots, '**/Cargo.toml'), { ignore: '**/node_modules/**' }).filter((b: string) => {
17 | return fs.readFileSync(b).toString().includes('[[bin]]')
18 | }).map((nodeRs: string) => path.dirname(nodeRs)); // (todo transducers)
19 | }
20 |
21 | const findNodesInWorkspace = () => {
22 | return vscode.workspace.workspaceFolders?.map((x) => x.uri.fsPath).map(root => {
23 | return findNodesInFolder(root);
24 | }).flat() || [];
25 | }
26 |
27 | export type Node = {nodePath: string; runtimePath?: string; deps?: any[]}
28 |
29 | // Manages the auto-detection of nodes in the current workspace
30 | export default class Nodes {
31 |
32 | nodes$: BehaviorSubject;
33 |
34 | constructor() {
35 | // TODO add to the deactivate function of extension.ts? Not sure if vscode
36 | // does this automatically. Evtl add a dispose() function to the class.
37 | const watcher = vscode.workspace.createFileSystemWatcher('**/Cargo.toml');
38 |
39 | const nodesPath$ = merge(
40 | // Scan on startup
41 | of(null),
42 | // Scan on file changes involving build.rs
43 | merge(
44 | vscToObservable(watcher.onDidChange.bind(watcher)),
45 | vscToObservable(watcher.onDidCreate.bind(watcher)),
46 | vscToObservable(watcher.onDidDelete.bind(watcher))
47 | ).pipe(throttleTime(100)) // e.g. in case of renaming folders, we receive a delete + added event.
48 | ).pipe
49 | (
50 | map(() => findNodesInWorkspace()),
51 | // tap(r => console.log('Found nodes',r))
52 | ); // TODO have a rescan button
53 |
54 | this.nodes$ = new BehaviorSubject([]);
55 |
56 | nodesPath$.pipe(
57 | map(nodePaths => {
58 | return nodePaths.map(nodePath => {
59 | try{
60 | const cargoToml = fs.readFileSync(path.join(nodePath, 'Cargo.toml')).toString(); // TODO ASYNC
61 | const matches = cargoToml.match(/^(?:[^#].*)?path ?= ?"([a-zA-Z./-]*runtime[a-zA-Z./-]*)"|^(?:[^#].*)?path ?= ?'([a-zA-Z./-]*runtime[a-zA-Z./-]*)'/m)
62 | if (matches === null)
63 | return {nodePath} as Node;
64 | else {
65 | // Important that this be called on any dep change (Cargo.toml)
66 | const runtimePath = path.join(nodePath, matches[1] || matches[2]);
67 |
68 | return {
69 | nodePath,
70 | runtimePath: runtimePath,
71 | deps: parseDepsFromFile(path.join(runtimePath, 'Cargo.toml'))
72 | } as Node;
73 | }
74 | } catch (e) { console.error('error',e); return {nodePath: ''} }
75 | });
76 | }),
77 | // tap(r => console.log('Found nodes infos', r))
78 | ).subscribe(this.nodes$);
79 | this.nodes$.subscribe(x => {});
80 | }
81 | }
--------------------------------------------------------------------------------
/resources/github.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/resources/github-black.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/src/views/processes/ProcessesProvider.ts:
--------------------------------------------------------------------------------
1 | import * as vscode from 'vscode';
2 | import { BehaviorSubject, of, combineLatest } from 'rxjs';
3 | import { tryShortname, wsEndpointFromCommand } from '../../util';
4 | import { switchMap, tap, distinctUntilChanged } from 'rxjs/operators';
5 | import Nodes, {Node} from '../../nodes/Nodes';
6 | import Processes, { Process } from '../../processes/Processes';
7 | import { Substrate } from '../../common/Substrate';
8 |
9 | export class ProcessesProvider implements vscode.TreeDataProvider {
10 | ProcessTreeItems: ProcessTreeItem[] = [];
11 |
12 | private _onDidChangeTreeData: vscode.EventEmitter = new vscode.EventEmitter();
13 | readonly onDidChangeTreeData: vscode.Event = this._onDidChangeTreeData.event;
14 |
15 | constructor(processes: Processes) {
16 | processes.processes$.subscribe((processes: Process[]) => {
17 | this.ProcessTreeItems = processes.map((process: Process) => {
18 | return new ProcessTreeItem(process);
19 | });
20 |
21 | this._onDidChangeTreeData.fire();
22 | });
23 | }
24 |
25 | getTreeItem(element: ProcessTreeItem): ProcessTreeItem | Thenable {
26 | return element;
27 | }
28 |
29 | getChildren(element?: ProcessTreeItem | undefined): vscode.ProviderResult {
30 | if (element === undefined) {
31 | return this.ProcessTreeItems;
32 | }
33 | return element.children;
34 | }
35 | }
36 |
37 | const isTheia = process.env.SUBSTRATE_PLAYGROUND !== undefined;
38 |
39 | export class ProcessTreeItem extends vscode.TreeItem {
40 | children: undefined;
41 | process: Process;
42 |
43 | constructor(process: Process) {
44 | const { nodePath, command } = process;
45 | super(
46 | tryShortname(nodePath) + ' • ' + command,
47 | vscode.TreeItemCollapsibleState.None);
48 | this.process = process;
49 | this.command = {
50 | command: "substrate.selectProcess",
51 | title: "Select Process",
52 | arguments: [this]
53 | };
54 |
55 | this.contextValue = isTheia ? 'theia' : 'vscode'
56 | }
57 | }
58 |
59 |
60 | async function quickPickProcesses(_processes: Processes) {
61 | let processes = _processes.processes$.getValue();
62 |
63 | if (processes.length === 1)
64 | return processes[0];
65 |
66 | if (processes.length === 0) {
67 | vscode.window.showErrorMessage('No node was found in the workspace.');
68 | return Promise.reject();
69 | }
70 |
71 | const processesReadable = processes.map(n => tryShortname(n.nodePath));
72 |
73 | const pick = await vscode.window.showQuickPick(processesReadable, { placeHolder: "Please choose a node." });
74 | if (pick === undefined)
75 | return Promise.reject();
76 |
77 | return processes[processesReadable.findIndex(x => x === pick)];
78 | }
79 |
80 | const INSTANCE = process.env.SUBSTRATE_PLAYGROUND_INSTANCE;
81 | const HOST = process.env.SUBSTRATE_PLAYGROUND_HOSTNAME;
82 |
83 | export function setupProcessesTreeView(substrate: Substrate, processes: Processes) {
84 |
85 | vscode.commands.registerCommand("substrate.polkadotApps", async (processTreeItem?: ProcessTreeItem) => {
86 | const process = processTreeItem?.process || await quickPickProcesses(processes);
87 |
88 | const port = process.command.match(/--ws-port[ =]\d+/)?.[0] || '9944';
89 | const wsEndpoint = `wss://${INSTANCE}.${HOST}${port !== '9944' ? `:${port}` : ''}/wss`;
90 | const apps = `https://polkadot.js.org/apps/?rpc=${wsEndpoint}`;
91 | vscode.commands.executeCommand('vscode.open', vscode.Uri.parse(apps));
92 | });
93 |
94 | const selectedProcess$ = new BehaviorSubject(null);
95 | vscode.commands.registerCommand("substrate.selectProcess", async (processTreeItem: ProcessTreeItem) => {
96 | selectedProcess$.next((processTreeItem as any).process || null);
97 | processTreeItem.process.term.show();
98 | });
99 |
100 | vscode.commands.registerCommand("substrate.stopProcess", async (processTreeItem?: ProcessTreeItem) => {
101 | const process = processTreeItem?.process || await quickPickProcesses(processes);
102 | processes.del(process);
103 | process.term.sendText('\x03');
104 | process.term.sendText('exit 0');
105 | });
106 |
107 | const treeDataProvider = new ProcessesProvider(processes);
108 | const treeView = vscode.window.createTreeView('substrateProcesses', { treeDataProvider });
109 |
110 | processes.processes$.subscribe((processes: Process[]) => {
111 | if (processes.length === 0 && isTheia)
112 | treeView.message = `No node is currently running.`;
113 | else
114 | treeView.message = undefined;
115 |
116 | if (!processes.find(process => process === selectedProcess$.getValue()))
117 | selectedProcess$.next(null);
118 |
119 |
120 | });
121 |
122 | selectedProcess$.pipe(distinctUntilChanged()).subscribe(process => {
123 | if (!process)
124 | substrate.disconnect();
125 | else {
126 | substrate.connectTo(wsEndpointFromCommand(process.command));
127 | }
128 | });
129 |
130 | return selectedProcess$;
131 | }
--------------------------------------------------------------------------------
/docs/features.md:
--------------------------------------------------------------------------------
1 | # Features
2 |
3 | - [Features](#features)
4 | * [1. Workspace actions](#1-workspace-actions)
5 | + [1.1 Run workspace actions](#11-run-workspace-actions)
6 | * [2. Node actions](#2-node-actions)
7 | + [2.1 Compile & run a node](#21-compile---run-a-node)
8 | + [2.2 Run an already compiled node](#22-run-an-already-compiled-node)
9 | + [2.3 Kill a process](#23-kill-a-process)
10 | + [2.4 Purge a node's chain](#24-purge-a-node-s-chain)
11 | * [3. Runtime actions](#3-runtime-actions)
12 | + [3.1 Get information on installed pallets](#31-get-information-on-installed-pallets)
13 | + [3.2 Add a new pallet](#32-add-a-new-pallet)
14 | * [4. Account management](#4-account-management)
15 | + [4.1 Import an account](#41-import-an-account)
16 | + [4.2 Import an account from JSON](#42-import-an-account-from-json)
17 | + [4.3 Generate a new account](#43-generate-a-new-account)
18 | + [4.4 Rename, copy address, export, remove account](#44-rename--copy-address--export--remove-account)
19 | * [5. Contracts](#5-contracts)
20 | + [5.1 Compile & deploy a contract](#51-compile---deploy-a-contract)
21 | + [5.2 Call a contract method](#52-call-a-contract-method)
22 | + [5.3 Copy hash, forget contract](#53-copy-hash--forget-contract)
23 |
24 | Access the extension's features by clicking the "Substrate" icon to the left.
25 |
26 | 
27 |
28 | ## 1. Workspace actions
29 | ### 1.1 Run workspace actions
30 | The extension will detect all VSCode actions provided in your workspace folders and list them under the "Actions" panel. Hover over an action and click the "Execute" icon to the right to run it.
31 |
32 | 
33 |
34 | ## 2. Node actions
35 | ### 2.1 Compile & run a node
36 | The extension will detect all nodes (binary Rust projects) in the workspace and list them in the "Nodes" panel. To compile and run a node, hover over the node in the list and click the corresponding icon. The new process will be listed under "Processes" and a new terminal will be opened with the process's ouput.
37 |
38 | 
39 |
40 | ### 2.2 Run an already compiled node
41 | If you have already compiled the node and which to run it again without compiling the new changes, click the "Start node" button. The new process will be listed under "Processes" and a new terminal will be opened with the process's ouput.
42 |
43 | 
44 |
45 | ### 2.3 Kill a process
46 | Once a node is running, you can exit it by clicking the "Stop process" icon.
47 |
48 | 
49 |
50 | ### 2.4 Purge a node's chain
51 | You can purge the chain of an already compiled node by clicking the "Purge chain" icon.
52 |
53 | 
54 |
55 | ## 3. Runtime actions
56 |
57 | ### 3.1 Get information on installed pallets
58 | First, select a node.
59 |
60 | 
61 |
62 | The runtime used by this node will be automatically detected and displayed in the "Marketplace" panel. The "Marketplace" panel will then list all pallets available, and add a checkmark next to those that are used by the runtime of the selected node. Hover over each pallet to get access to their homepage, documentation page and GitHub page.
63 |
64 | ### 3.2 Add a new pallet
65 | To add a pallet as a dependency of the runtime, click the "Add pallet" icon. This will install the pallet using the tool `substrate-deps`, installing it if necessary.
66 |
67 | 
68 |
69 | ## 4. Account management
70 | Please note that this extension's account management features are meant to be used for testing/development purposes only. The accounts are persisted across restarts.
71 |
72 | ### 4.1 Import an account
73 | To add an account, hover over the "Accounts" bar and click the "+" icon. You will then be asked for a human-friendly name, cryptographic type and the key to the account.
74 |
75 | 
76 |
77 | ### 4.2 Import an account from JSON
78 | You can also import an account from a JSON file by clicking on the "Next actions" icon next to the "+" icon, then choosing "Import account from JSON."
79 |
80 | 
81 |
82 | ### 4.3 Generate a new account
83 | You can also generate a new account with a random seed by clicking "Generate new account".
84 |
85 | 
86 |
87 | ### 4.4 Rename, copy address, export, remove account
88 | By right-clicking an account, you can rename it, copy its address, export it as a JSON file, or remove the account from the list.
89 |
90 | 
91 |
92 | ## 5. Contracts
93 | Contracts are persisted across restarts and are linked to a WebSocket endpoint (host and port). You need to have added an account in the extension in order to be able to deploy a contract.
94 |
95 | ### 5.1 Compile & deploy a contract
96 | Select a running process. This will list the contracts you already added to this endpoint in the extension (if any).
97 |
98 | 
99 |
100 | You can compile an __ink!__ project, upload its WASM code and instantiate the contract with one click by clicking the "..." icon after hovering over "Contracts", and then selecting "Compile & deploy contract". You will then be asked for the location of your __ink!__ project, a human-friendly name for the contract, the account (see section 4) with which to upload the WASM & deploy the contract, its password, the constructor to use when instantiating the contract, if necessary its arguments, the endowment and maximum amount of gas for the deployment. The contract will then be listed in the "Contracts" panel.
101 |
102 | 
103 |
104 | ### 5.2 Call a contract method
105 | Right-click a contract and click "Call contract method". You will then be asked various information and be informed of the progress of the transaction.
106 |
107 | 
108 |
109 | ### 5.3 Copy hash, forget contract
110 | By right-clicking a contract, you can also copy its hash or remove the contract from the list.
111 |
112 | 
--------------------------------------------------------------------------------
/src/common/Substrate.ts:
--------------------------------------------------------------------------------
1 | import * as vscode from 'vscode';
2 | import { Keyring } from '@polkadot/keyring';
3 | import { KeyringPair$Json } from '@polkadot/keyring/types';
4 | import { KeypairType } from '@polkadot/util-crypto/types';
5 | import { ApiPromise, WsProvider } from '@polkadot/api';
6 | const fs = require('fs');
7 |
8 | export type Contracts = { [index: string]: Contract[] };
9 |
10 | export type Contract = { name: string, address: string, abiJson: any };
11 |
12 | // Manages API connection and extension storage (accounts, contracts)
13 | export class Substrate {
14 | private keyring = new Keyring({ type: 'sr25519' });
15 | private api?: ApiPromise;
16 | private wsEndpoint?: string;
17 |
18 | getConnection(): ApiPromise | undefined {
19 | return this.api;
20 | }
21 |
22 | constructor(
23 | // Necessary to access the extension's persistent storage
24 | private context: vscode.ExtensionContext
25 | ) { }
26 |
27 | async connectTo(wsEndpoint: string) {
28 | this.disconnect();
29 | this.wsEndpoint = wsEndpoint;
30 | const api = new ApiPromise({provider: new WsProvider(wsEndpoint)});
31 | await api.isReady;
32 | this.api = api;
33 | }
34 |
35 | async disconnect() {
36 | if (this.api) {
37 | this.api!.disconnect();
38 | }
39 | this.wsEndpoint = undefined;
40 | this.api = undefined;
41 | }
42 |
43 | getAccounts(): KeyringPair$Json[] {
44 | const accounts = this.context.globalState.get('accounts');
45 | if (!accounts) {
46 | return [];
47 | }
48 | return JSON.parse(accounts);
49 | }
50 |
51 | getKeyring(): Keyring {
52 | return this.keyring;
53 | }
54 |
55 | // Verify if an account with the given name exists
56 | isAccountExists(name: string): boolean {
57 | const result = this.getAccounts();
58 | const exKey = result.find((val) => val.meta.name === name);
59 | if (!exKey) {
60 | return false;
61 | }
62 | return true;
63 | }
64 |
65 | // Replace all accounts with new accounts list
66 | async updateAccounts(accounts: KeyringPair$Json[]) {
67 | await this.context.globalState.update('accounts', JSON.stringify(accounts));
68 | }
69 |
70 | async createKeyringPair(key: string, name: string, type: KeypairType) {
71 | const pair = this.keyring.addFromUri(key, { name }, type);
72 | const accounts = this.getAccounts();
73 | accounts.push(pair.toJson());
74 | await this.updateAccounts(accounts);
75 | }
76 |
77 | async createKeyringPairWithPassword(key: string, name: string, type: KeypairType, pass: string) {
78 | const pair = this.keyring.addFromUri(key, { name }, type);
79 |
80 | const json = pair.toJson(pass);
81 | json.meta.whenEdited = Date.now();
82 |
83 | const accounts = this.getAccounts();
84 | accounts.push(json);
85 | await this.updateAccounts(accounts);
86 | }
87 |
88 | // Forget an account
89 | async removeAccount(name: string) {
90 | const accounts = this.getAccounts();
91 | const index = accounts.findIndex((val) => val.meta['name'] === name);
92 | accounts.splice(index, 1);
93 | await this.updateAccounts(accounts);
94 | }
95 |
96 | async renameAccount(oldName: string, newName: string) {
97 | const accounts = this.getAccounts();
98 | for (const account of accounts) {
99 | if (account.meta['name'] === oldName) {
100 | account.meta['name'] = newName;
101 | break;
102 | }
103 | }
104 | await this.updateAccounts(accounts);
105 | }
106 |
107 | // Import a new keyring pair globally
108 | // @TODO Keyring pairs are currently global, maybe scope them?
109 | async importKeyringPair(path: string) {
110 | const rawdata = fs.readFileSync(path);
111 | const pair: KeyringPair$Json = JSON.parse(rawdata.toString());
112 | if (this.isAccountExists(pair.meta['name'] as string)) {
113 | vscode.window.showWarningMessage('Account with same key already exists. Account not added');
114 | return;
115 | }
116 | const accounts = this.getAccounts();
117 | accounts.push(pair);
118 | await this.updateAccounts(accounts);
119 | }
120 |
121 | // Get the list of contracts for the current endpoint
122 | getConnectionContracts() : any[] {
123 | if (!this.wsEndpoint) return [];
124 | const contractCodes = this.getContracts();
125 | // @TODO Contracts are currently scoped to a websocket endpoint
126 | // We could use better heuristics instead
127 | const nodeContractCodes = contractCodes[this.wsEndpoint] || [];
128 | return nodeContractCodes;
129 | }
130 |
131 | // Get the list of all contracts for all endpoints
132 | getContracts() {
133 | const contractsString = this.context.globalState.get('contracts');
134 | if (!contractsString) {
135 | return {};
136 | }
137 | return JSON.parse(contractsString);
138 | }
139 |
140 | // Add a new contract for the current endpoint
141 | async saveContract(contractName: string, contractAddress: string, abiJson: any) {
142 | const contracts = this.getConnectionContracts();
143 | const existingContract = contracts.find(
144 | contract => contract.name === contractName || contract.address === contractAddress
145 | );
146 | if (existingContract) {
147 | existingContract.name = contractName;
148 | existingContract.address = contractAddress;
149 | existingContract.abiJson = abiJson;
150 | } else {
151 | contracts.push({
152 | name: contractName,
153 | address: contractAddress,
154 | abiJson: abiJson,
155 | });
156 | }
157 | await this.updateConnectionContracts(contracts);
158 | }
159 |
160 | // Update contracts for the current endpoint
161 | async updateConnectionContracts(codes: Contract[]) {
162 | const connectedNode = this.wsEndpoint;
163 | if (!connectedNode) {
164 | throw Error('Not connected to node');
165 | }
166 | const contracts = this.getContracts();
167 | contracts[connectedNode] = codes;
168 | await this.updateContracts(contracts);
169 | }
170 |
171 | async updateContracts(codes: Contracts) {
172 | await this.context.globalState.update('contracts', JSON.stringify(codes));
173 | await vscode.commands.executeCommand('substrate.refreshContracts'); // Refresh TreeView
174 | }
175 | }
176 |
--------------------------------------------------------------------------------
/src/views/marketplace/MarketplaceProvider.ts:
--------------------------------------------------------------------------------
1 | import * as vscode from 'vscode';
2 | import { BehaviorSubject } from 'rxjs';
3 | import fetchCategories from './fetchCategories';
4 | import { substrateDepsInstalled } from './substrateDeps';
5 | import { Category, Pallet } from './types';
6 | import Nodes, { Node } from '../../nodes/Nodes';
7 | import { tryShortname } from '../../util';
8 |
9 | const path = require('path');
10 |
11 | type TreeItem = TreePallet | TreeCategory;
12 |
13 | export class TreeDataProvider implements vscode.TreeDataProvider {
14 | treeCategories: TreeCategory[];
15 |
16 | private _selectedNode$: any;
17 |
18 | private _onDidChangeTreeData: vscode.EventEmitter = new vscode.EventEmitter();
19 | readonly onDidChangeTreeData: vscode.Event = this._onDidChangeTreeData.event;
20 |
21 | constructor(data: any, selectedNode$: BehaviorSubject) {
22 | this.treeCategories = data.map((category: any) => {
23 | return new TreeCategory(category, category.pallets.map(TreePallet.create.bind(TreePallet)));
24 | });
25 |
26 | this._selectedNode$ = selectedNode$;
27 |
28 | // TODO Reinstantiating all pallets would be cleaner (shared logic when instantiating)
29 | selectedNode$.subscribe((node) => {
30 | this.treeCategories.forEach(treeCategory => {
31 | treeCategory.children?.forEach(treePallet => {
32 | treePallet.contextValue = node && node.deps && node.deps.includes(treePallet.name)
33 | ? 'palletInstalled' // TODO work from a single source of truth and have a clear mapping instead (see above comment); pure functions
34 | : node ? 'pallet' : 'palletNoNode';
35 | treePallet.iconPath = treePallet.contextValue === 'palletInstalled' ? path.join(__filename, '..', '..', '..', '..', 'resources', 'check.svg') : false;
36 | });
37 | });
38 | this._onDidChangeTreeData.fire();
39 | });
40 | }
41 |
42 | getTreeItem(element: TreeCategory): TreeCategory | Thenable {
43 | return element;
44 | }
45 |
46 | getChildren(element?: TreeItem | undefined): vscode.ProviderResult {
47 | if (element === undefined) {
48 | return this.treeCategories;
49 | }
50 | return element.children;
51 | }
52 | }
53 |
54 | export class TreeCategory extends vscode.TreeItem {
55 | children: TreePallet[] | undefined;
56 |
57 | constructor(info: Category, children?: TreePallet[]) {
58 | const { category: label } = info;
59 | super(
60 | label,
61 | children === undefined ? vscode.TreeItemCollapsibleState.None :
62 | vscode.TreeItemCollapsibleState.Expanded);
63 | this.children = children;
64 | this.contextValue = 'category';
65 | }
66 | }
67 |
68 | export class TreePallet extends vscode.TreeItem {
69 | children: TreeItem[] | undefined;
70 | name: string;
71 | github: string;
72 | documentation: string;
73 | homepage: string;
74 |
75 | static create(info: Pallet) {
76 | return new this(info);
77 | }
78 |
79 | constructor(info: Pallet) {
80 | const { name, description, github, documentation, homepage } = info;
81 | super(
82 | name,
83 | vscode.TreeItemCollapsibleState.None);
84 | this.name = name;
85 | this.description = description;
86 | this.tooltip = `${name} - ${description}`;
87 | this.contextValue = 'palletNoNode';
88 | this.github = github;
89 | this.documentation = documentation;
90 | this.homepage = homepage;
91 | }
92 | }
93 |
94 | type NodePath = string | null;
95 | export function setUpMarketplaceTreeView(nodes: Nodes, selectedNode$: BehaviorSubject) {
96 | fetchCategories().then((categories: Category[]) => {
97 | const treeView = vscode.window.createTreeView('substrateMarketplace', { treeDataProvider: new TreeDataProvider(categories, selectedNode$) });
98 | selectedNode$.subscribe((node) => {
99 | if (node && node.runtimePath)
100 | treeView.message = `Runtime: ${tryShortname(node.runtimePath)}`;
101 | else
102 | treeView.message = `No node selected`;
103 | });
104 |
105 | // Set up commands: documentation, github, homepage
106 | ([
107 | { command: 'substrateMarketplace.palletDocumentation', name: 'Documentation', property: 'documentation' },
108 | { command: 'substrateMarketplace.palletGithub', name: 'GitHub page', property: 'github' },
109 | { command: 'substrateMarketplace.palletHomepage', name: 'Homepage', property: 'homepage' }
110 | ] as const).forEach(({ command, name, property }) => {
111 | vscode.commands.registerCommand(command, (item: TreePallet) => {
112 | if (!item[property].startsWith('http')) { // Also acts as a safeguard
113 | vscode.window.showErrorMessage(`${name} is unavailable for this pallet.`);
114 | return;
115 | }
116 | vscode.commands.executeCommand('vscode.open', vscode.Uri.parse(item[property]));
117 | });
118 | });
119 |
120 | // Set up command: install
121 | vscode.commands.registerCommand("substrateMarketplace.installPallet", async (item: vscode.TreeItem) => {
122 | // Install substrate-deps if needed
123 | if (!await substrateDepsInstalled()) {
124 | return;
125 | }
126 |
127 | // Verify pallet name to prevent shell injection & derive alias
128 | const palletName = item.label as string;
129 | if (!/^[a-z-]+$/.test(palletName)) {
130 | vscode.window.showErrorMessage('Pallet name is invalid.');
131 | return;
132 | }
133 | const alias = (alias => alias === palletName ? null : alias)(palletName.replace(/^pallet-/, ''));
134 |
135 | // Ask for user confirmation
136 | // TODO Indicate current runtime in the message in case we have more than
137 | // one runtime in the workspace.
138 | const clicked = await vscode.window.showInformationMessage(`Install the pallet ${palletName}?`, { modal: true }, 'Yes');
139 | if (clicked !== 'Yes') {
140 | return;
141 | }
142 |
143 | // Get manifest path
144 | let manifestPath: string;
145 | try {
146 | let selectedNodeRuntimePath = selectedNode$.getValue()?.runtimePath;
147 | if (!selectedNodeRuntimePath) {
148 | vscode.window.showErrorMessage('Please first select a node.');
149 | return;
150 | }
151 | manifestPath = path.join(selectedNodeRuntimePath, 'Cargo.toml');
152 | } catch (e) {
153 | return;
154 | }
155 |
156 | // Prepare command
157 | const termCommand = [
158 | 'substrate-deps',
159 | `add ${palletName}`,
160 | ...alias ? [`--alias ${alias}`] : [],
161 | `--manifest-path '${manifestPath.replace(/'/, `'\\''`)}'`, // Allow spaces in path, prevent command injection (TODO Windows?)
162 | `&& echo '${palletName} was successfully added to the project${alias ? ` as '${alias}'` : ''}.'`
163 | ].join(' ');
164 |
165 | // Create terminal and run command
166 | const term = vscode.window.createTerminal({ name: `Installing ${palletName}` });
167 | term.sendText(termCommand);
168 | term.show();
169 |
170 | });
171 | }, (async r => { // Offer to retry in case fetching the categories failed
172 | const clicked = await vscode.window.showErrorMessage(`An error occured when fetching the list of pallets from the Substrate Marketplace: ${r}`, 'Try again');
173 | if (clicked === 'Try again') {
174 | return setUpMarketplaceTreeView(nodes, selectedNode$);
175 | }
176 | }));
177 |
178 | }
--------------------------------------------------------------------------------
/src/views/nodes/NodesProvider.ts:
--------------------------------------------------------------------------------
1 | import * as vscode from 'vscode';
2 | import { BehaviorSubject, of, combineLatest } from 'rxjs';
3 | import { tryShortname } from '../../util';
4 | import { switchMap, tap } from 'rxjs/operators';
5 | import Nodes, {Node} from '../../nodes/Nodes';
6 |
7 | const path = require('path');
8 | const fs = require('fs');
9 |
10 | export class NodesProvider implements vscode.TreeDataProvider {
11 | nodeTreeItems: NodeTreeItem[] = [];
12 |
13 | private _onDidChangeTreeData: vscode.EventEmitter = new vscode.EventEmitter();
14 | readonly onDidChangeTreeData: vscode.Event = this._onDidChangeTreeData.event;
15 |
16 | constructor(nodes: Nodes, selectedNode$: BehaviorSubject) {
17 | nodes.nodes$.subscribe((nodes: Node[]) => {
18 |
19 | this.nodeTreeItems = nodes.map((node: Node) => {
20 | return new NodeTreeItem(node);
21 | });
22 |
23 | this.changeSelected(selectedNode$.getValue()?.nodePath || null);
24 | this._onDidChangeTreeData.fire();
25 | });
26 |
27 | selectedNode$.subscribe(node => {
28 | this.changeSelected(node?.nodePath || null);
29 | });
30 | }
31 |
32 | changeSelected(selectedNodePath: string | null) {
33 | this.nodeTreeItems.forEach((nodeTreeItem) => {
34 | if (nodeTreeItem.nodePath === selectedNodePath)
35 | nodeTreeItem.select();
36 | else
37 | nodeTreeItem.unselect();
38 | });
39 | this._onDidChangeTreeData.fire();
40 |
41 | }
42 |
43 | getTreeItem(element: NodeTreeItem): NodeTreeItem | Thenable {
44 | return element;
45 | }
46 |
47 | getChildren(element?: NodeTreeItem | undefined): vscode.ProviderResult {
48 | if (element === undefined) {
49 | return this.nodeTreeItems;
50 | }
51 | return element.children;
52 | }
53 | }
54 |
55 | export class NodeTreeItem extends vscode.TreeItem {
56 | children: undefined;
57 | nodePath: string;
58 |
59 | static create(info: Node) {
60 | return new this(info);
61 | }
62 |
63 | constructor(node: Node) {
64 | const { nodePath } = node;
65 | super(
66 | tryShortname(nodePath),
67 | vscode.TreeItemCollapsibleState.None);
68 | this.nodePath = nodePath;
69 | this.command = {
70 | command: "substrateNodes.selectNode",
71 | title: "Select Node",
72 | arguments: [this]
73 | };
74 | }
75 |
76 | select() {
77 | this.label = '▶️ ' + tryShortname(this.nodePath);
78 | }
79 |
80 | unselect() {
81 | this.label = tryShortname(this.nodePath);
82 | }
83 | }
84 |
85 | // Prompt the user to select a node among a list
86 | async function quickPickNodePath(nodes: Nodes) {
87 | let nodePaths = nodes.nodes$.getValue().map((node: Node) => node.nodePath);
88 |
89 | if (nodePaths.length === 1)
90 | return nodePaths[0];
91 |
92 | if (nodePaths.length === 0) {
93 | vscode.window.showErrorMessage('No node was found in the workspace.');
94 | return Promise.reject();
95 | }
96 |
97 | const nodesReadable = nodePaths.map(n => tryShortname(n));
98 |
99 | const pick = await vscode.window.showQuickPick(nodesReadable, { placeHolder: "Please choose a node." });
100 | if (pick === undefined)
101 | return Promise.reject();
102 |
103 | return nodePaths[nodesReadable.findIndex(x => x === pick)];
104 | }
105 |
106 | function getCargoName(directory: string) {
107 | const toml = fs.readFileSync(path.join(directory,'Cargo.toml')).toString();
108 | const ap = toml.match(/name ?= ?'(.+)'/);
109 | if (ap)
110 | return ap[1];
111 | else return toml.match(/name ?= ?"(.+)"/)[1];
112 | }
113 |
114 | function getWorkspaceRoot(directory: string) {
115 | let currDir = directory;
116 | do {
117 | if (fs.existsSync(path.join(currDir, 'Cargo.lock')))
118 | return currDir;
119 | currDir = path.join(currDir, '..');
120 | } while (currDir.split(path.sep).length > 0)
121 | throw new Error('No workspace root was found');
122 | }
123 |
124 | export function setUpNodesTreeView(nodes: Nodes, processes: any) {
125 |
126 | // Can be called by the NodesProvider, in which case a NodeTreeItem is provided;
127 | // or remotely (across iframe in Substrate Playground), in which case we provide
128 | // the path of the node to run, and optional additional flags.
129 | vscode.commands.registerCommand("substrate.startNode", async (nodePathLike?: string | NodeTreeItem, _flags?: string | string[], options: any = {compile: false}) => {
130 | try {
131 | let nodePath = nodePathLike instanceof NodeTreeItem ? nodePathLike.nodePath : nodePathLike;
132 | if (nodePath) {
133 | selectedNodePath$.next(nodePath); // select the item we launch the command on
134 | }
135 | const defNodePath = nodePath || await quickPickNodePath(nodes); // e.g. When run through Command Palette
136 | const term = vscode.window.createTerminal({ name: 'Start node ' + tryShortname(defNodePath), cwd: defNodePath });
137 |
138 | const flags = _flags ? (Array.isArray(_flags) ? _flags.join(' ') : _flags) : await vscode.window.showInputBox({
139 | value: '--dev --ws-external',
140 | prompt: 'Flags to run Substrate with',
141 | ignoreFocusOut: true
142 | });
143 | if (!flags) return; // user cancelled
144 |
145 | // todo use ws port to use polkadot apps
146 | // does it make sense to have two different polkadot apps endpoints ? two processes, one polkadot app endpoint ?
147 |
148 | if (options.compile) {
149 | term.sendText(`cargo run --release -- ${flags}`);
150 | } else {
151 | term.sendText(path.join(getWorkspaceRoot(defNodePath),`target/release/${getCargoName(defNodePath)}`) + ' ' + flags);
152 | }
153 | term.show();
154 |
155 | processes.new({nodePath: defNodePath, command: flags, term: term});
156 | } catch (e) {
157 | vscode.window.showErrorMessage(e);
158 | console.error(e);
159 | }
160 | });
161 |
162 | vscode.commands.registerCommand("substrate.compileStartNode", async (nodePathLike?: string | NodeTreeItem, _flags?: string | string[]) => {
163 | vscode.commands.executeCommand("substrate.startNode", nodePathLike, _flags, {compile: true});
164 | });
165 |
166 | vscode.commands.registerCommand("substrate.purgeChain", async (nodePathLike?: string | NodeTreeItem) => {
167 | let nodePath = nodePathLike instanceof NodeTreeItem ? nodePathLike.nodePath : nodePathLike;
168 | if (nodePath) {
169 | selectedNodePath$.next(nodePath); // select the item we launch the command on
170 | }
171 | const defoNodePath = nodePath || await quickPickNodePath(nodes);
172 | const term = vscode.window.createTerminal({ name: 'Purge chain', cwd: defoNodePath });
173 |
174 | term.sendText(path.join(getWorkspaceRoot(defoNodePath), `target/release/${getCargoName(defoNodePath)}`) + ' purge-chain --dev'); // TODO s
175 | term.show();
176 | });
177 |
178 | const selectedNodePath$ = new BehaviorSubject(null); // TODO NULL ON UNSELECT
179 | const selectedNode$ = new BehaviorSubject(null);
180 |
181 | combineLatest(selectedNodePath$, nodes.nodes$)
182 | .pipe(
183 | switchMap(([nodePath, nodes]: [string | null, Node[]]) => {
184 | if (!nodePath) return of(null);
185 | const selectedNode = nodes.find(node => node.nodePath === nodePath);
186 | if (selectedNode === undefined) {
187 | console.error("Selected node but doesn't match any.");
188 | return of(null);
189 | }
190 | return of(selectedNode);
191 | }),
192 | // tap(r => console.log('Selected node \'changes\' fired with', r))
193 | ).subscribe(selectedNode$);
194 |
195 | const treeDataProvider = new NodesProvider(nodes, selectedNode$);
196 | vscode.window.createTreeView('substrateNodes', { treeDataProvider });
197 | vscode.commands.registerCommand("substrateNodes.selectNode", (item: vscode.TreeItem) => {
198 | selectedNodePath$.next((item as any).nodePath || null);
199 | });
200 |
201 | return { selectedNode$ };
202 | }
--------------------------------------------------------------------------------
/src/views/accounts/AccountsProvider.ts:
--------------------------------------------------------------------------------
1 | import { u8aToHex } from '@polkadot/util';
2 | import { mnemonicGenerate, randomAsU8a } from '@polkadot/util-crypto';
3 | import * as clipboard from 'clipboardy';
4 | import * as vscode from 'vscode';
5 | import { Substrate } from '../../common/Substrate';
6 | import { TreeDataProvider } from '../../common/TreeDataProvider';
7 | import { showInputBoxValidate } from '../../util';
8 |
9 | const fs = require('fs');
10 |
11 | type Account = any;
12 |
13 | export class AccountsProvider extends TreeDataProvider {
14 | substrate: Substrate;
15 |
16 | constructor(substrate: Substrate) {
17 | super();
18 | this.substrate = substrate;
19 | }
20 |
21 | getChildren(element?: AccountTreeItem): Thenable {
22 | if (element === undefined) {
23 | return Promise.resolve(this.substrate.getAccounts().map(a => new AccountTreeItem(a)));
24 | }
25 | return Promise.resolve(element.children);
26 | }
27 | }
28 |
29 | export class AccountTreeItem extends vscode.TreeItem {
30 | children: [] = [];
31 | account: Account;
32 |
33 | constructor(account: Account) {
34 | super(
35 | account.meta.name,
36 | vscode.TreeItemCollapsibleState.None);
37 | this.description = account.address;
38 | this.account = account;
39 | }
40 | }
41 |
42 | async function quickPickAccount(substrate: Substrate, placeholder: string) {
43 | const quickPickItem = await vscode.window.showQuickPick(substrate.getAccounts().map(pair => ({
44 | label: pair.address,
45 | decription: pair.meta.name,
46 | account: pair
47 | }),{placeHolder: placeholder}));
48 | return quickPickItem?.account;
49 | }
50 |
51 | export async function setupAccountsTreeView(substrate: Substrate, context: vscode.ExtensionContext) {
52 | const treeDataProvider = new AccountsProvider(substrate);
53 | vscode.window.createTreeView('substrateAccounts', { treeDataProvider });
54 |
55 | // Add an existing account
56 | vscode.commands.registerCommand("substrate.addAccount", async () => {
57 | const name: string | undefined = await showInputBoxValidate({
58 | ignoreFocusOut: true,
59 | prompt: 'Account name',
60 | placeHolder: 'ex. Alice'
61 | }, async (value: any) => {
62 | if (!value || !value.trim()) {
63 | return 'Name is required';
64 | }
65 | if (substrate.isAccountExists(value)) {
66 | return 'Account with same name already exists';
67 | }
68 | return '';
69 | });
70 | if (!name) return;
71 |
72 | const type: 'ed25519' | 'sr25519' | undefined = await vscode.window.showQuickPick(['ed25519', 'sr25519']) as 'ed25519' | 'sr25519' | undefined;
73 | if (!type) return;
74 |
75 | const key: string | undefined = await showInputBoxValidate({
76 | ignoreFocusOut: true,
77 | prompt: 'The key to the account',
78 | placeHolder: 'ex. //Alice'
79 | }, async (value: any) => {
80 | if (!value || !value.trim()) {
81 | return 'Key is required';
82 | }
83 | return '';
84 | });
85 | if (!key) return;
86 |
87 | await substrate.createKeyringPair(key, name, type);
88 |
89 | treeDataProvider.refresh();
90 | });
91 |
92 | // Generate a new account
93 | vscode.commands.registerCommand("substrate.createAccount", async () => {
94 | const name: string | undefined = await showInputBoxValidate({
95 | ignoreFocusOut: true,
96 | prompt: 'Account name',
97 | placeHolder: 'ex. Alice'
98 | }, async (value: any) => {
99 | if (!value || !value.trim()) {
100 | return 'Name is required';
101 | }
102 | if (substrate.isAccountExists(value)) {
103 | return 'Account with same name already exists';
104 | }
105 | return '';
106 | });
107 | if (!name) return;
108 |
109 | const cryptoType: 'ed25519' | 'sr25519' | undefined = await vscode.window.showQuickPick(['ed25519', 'sr25519']) as 'ed25519' | 'sr25519' | undefined;
110 | if (!cryptoType) return;
111 |
112 | const keyTypeR: 'Raw seed' | 'Mnemonic seed' | undefined = await vscode.window.showQuickPick(['Raw seed', 'Mnemonic seed']) as 'Raw seed' | 'Mnemonic seed' | undefined;
113 | if (!keyTypeR) return;
114 | const keyType = keyTypeR === 'Raw seed' ? 'seed' : 'mnemonic';
115 |
116 | const placeholder = keyType !== 'seed' ?
117 | 'crunch aspect strong flavor enable final display general shy debate stable final'
118 | : 'ex. 0x89abd2b6b79f4e2df7e89cb6b44c7f02d416719f6970b17d6ad34178423fa922';
119 | const seed = keyType !== 'seed' ?
120 | mnemonicGenerate()
121 | : u8aToHex(randomAsU8a());
122 | const key: string | undefined = await showInputBoxValidate({
123 | ignoreFocusOut: true,
124 | prompt: 'The key to the account',
125 | placeHolder: placeholder,
126 | value: seed
127 | }, async (value: any) => {
128 | if (!value || !value.trim()) {
129 | return 'Key is required';
130 | }
131 | return '';
132 | });
133 | if (!key) return;
134 |
135 | const password: string | undefined = await vscode.window.showInputBox({
136 | ignoreFocusOut: true,
137 | prompt: 'Account password',
138 | placeHolder: 'ex. StrongPassword',
139 | password: true,
140 | value: '',
141 | });
142 | if (typeof password === 'undefined') return;
143 |
144 | substrate.createKeyringPairWithPassword(key, name, cryptoType, password);
145 |
146 | treeDataProvider.refresh();
147 | });
148 |
149 | // Rename the selected account in the TreeView
150 | vscode.commands.registerCommand("substrate.renameAccount", async (item?: AccountTreeItem) => {
151 | let itemAccount = item?.account || await quickPickAccount(substrate, 'Please pick the account you want to rename');
152 | if (!itemAccount) return;
153 | const name: string | undefined = await showInputBoxValidate({
154 | ignoreFocusOut: true,
155 | prompt: 'Account name',
156 | value: itemAccount.meta.name,
157 | placeHolder: 'ex. Alice'
158 | }, async (value: any) => {
159 | if (!value || !value.trim()) {
160 | return 'Name is required';
161 | }
162 | if (substrate.isAccountExists(value) && value !== itemAccount.meta.name) {
163 | return 'Account with same name already exists';
164 | }
165 | return '';
166 | });
167 | if (!name) return;
168 |
169 | await substrate.renameAccount(itemAccount.meta.name, name);
170 |
171 | treeDataProvider.refresh();
172 | });
173 |
174 | // Remove the selected account in the TreeView
175 | vscode.commands.registerCommand("substrate.removeAccount", async (item?: AccountTreeItem) => {
176 | let itemAccount = item?.account || await quickPickAccount(substrate, 'Please pick the account you want to remove');
177 | if (!itemAccount) return;
178 | await substrate.removeAccount(itemAccount.meta.name);
179 | treeDataProvider.refresh();
180 | });
181 |
182 | // Copy the address of the selected account in the TreeView
183 | vscode.commands.registerCommand("substrate.copyAddress", async (item?: AccountTreeItem) => {
184 | let itemAccount = item?.account || await quickPickAccount(substrate, 'Please pick the account whose address you want to copy');
185 | if (!itemAccount) return;
186 | try {
187 | await clipboard.write(itemAccount.address);
188 | vscode.window.showInformationMessage(`Address copied to clipboard.`);
189 | } catch (err) {
190 | vscode.window.showErrorMessage(`Failed to copy address to clipboard: ${err.message}`);
191 | }
192 | });
193 |
194 | // Import a new account
195 | vscode.commands.registerCommand("substrate.importAccount", async () => {
196 | const res = await vscode.window.showOpenDialog({
197 | openLabel: 'Import',
198 | canSelectFiles: true,
199 | canSelectFolders: false,
200 | canSelectMany: false,
201 | filters: {
202 | 'JSON': ['json'],
203 | },
204 | });
205 | if (!res) return;
206 | await substrate.importKeyringPair(res[0].path);
207 |
208 | treeDataProvider.refresh();
209 | });
210 |
211 | // Export the selected account in the TreeView
212 | vscode.commands.registerCommand("substrate.exportAccount", async (item?: AccountTreeItem) => {
213 | let itemAccount = item?.account || await quickPickAccount(substrate,'Please pick an account to export');
214 | if (!itemAccount) return;
215 |
216 | const result = await vscode.window.showSaveDialog({});
217 | if (!result) return;
218 |
219 | fs.writeFileSync(result.fsPath, JSON.stringify(itemAccount), 'utf8');
220 | });
221 | }
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "vscode-substrate",
3 | "displayName": "VSCode Substrate",
4 | "description": "Manage your Substrate node and add pallets from the Marketplace to your runtime",
5 | "version": "0.5.2",
6 | "publisher": "paritytech",
7 | "icon": "resources/icon.png",
8 | "repository": {
9 | "type": "git",
10 | "url": "https://github.com/paritytech/vscode-substrate.git"
11 | },
12 | "bugs": {
13 | "url": "https://github.com/paritytech/vscode-substrate/issues"
14 | },
15 | "homepage": "https://github.com/paritytech/vscode-substrate/blob/master/README.md",
16 | "engines": {
17 | "vscode": "^1.42.0"
18 | },
19 | "keywords": [
20 | "pallets",
21 | "polkadot",
22 | "parity",
23 | "web3"
24 | ],
25 | "categories": [
26 | "Other"
27 | ],
28 | "activationEvents": [
29 | "*"
30 | ],
31 | "main": "./out/extension.js",
32 | "contributes": {
33 | "viewsWelcome": [
34 | {
35 | "view": "substrateAccounts",
36 | "contents": "No accounts found."
37 | },
38 | {
39 | "view": "substrateProcesses",
40 | "contents": "No node is currently running."
41 | },
42 | {
43 | "view": "substrateNodes",
44 | "contents": "No node was found in your current workspace.\n[Open folder](command:vscode.openFolder)"
45 | }
46 | ],
47 | "viewsContainers": {
48 | "activitybar": [
49 | {
50 | "id": "substrate",
51 | "title": "Substrate",
52 | "icon": "resources/substrate.svg"
53 | }
54 | ]
55 | },
56 | "views": {
57 | "substrate": [
58 | {
59 | "id": "substrateProcesses",
60 | "name": "Processes",
61 | "group": "@1"
62 | },
63 | {
64 | "id": "substrateNodes",
65 | "name": "Nodes",
66 | "group": "@2"
67 | },
68 | {
69 | "id": "substrateMarketplace",
70 | "name": "Marketplace",
71 | "group": "@3"
72 | },
73 | {
74 | "id": "substrateAccounts",
75 | "name": "Accounts",
76 | "group": "@4"
77 | },
78 | {
79 | "id": "substrateContracts",
80 | "name": "Contracts",
81 | "group": "@5"
82 | }
83 | ]
84 | },
85 | "commands": [
86 | {
87 | "command": "substrate.stopProcess",
88 | "title": "Substrate: Stop process",
89 | "icon": {
90 | "dark": "resources/stopProcess.svg",
91 | "light": "resources/stopProcess-black.svg"
92 | }
93 | },
94 | {
95 | "command": "substrate.polkadotApps",
96 | "title": "Substrate: Polkadot Apps",
97 | "icon": {
98 | "dark": "resources/apps.svg",
99 | "light": "resources/apps-black.svg"
100 | }
101 | },
102 | {
103 | "command": "substrate.purgeChain",
104 | "title": "Substrate: Purge chain",
105 | "icon": {
106 | "dark": "resources/purgeChain.svg",
107 | "light": "resources/purgeChain-black.svg"
108 | }
109 | },
110 | {
111 | "command": "substrate.startNode",
112 | "title": "Substrate: Start node",
113 | "icon": {
114 | "dark": "resources/startNode.svg",
115 | "light": "resources/startNode-black.svg"
116 | }
117 | },
118 | {
119 | "command": "substrate.compileStartNode",
120 | "title": "Substrate: Compile & start node",
121 | "icon": {
122 | "dark": "resources/compileStartNode.svg",
123 | "light": "resources/compileStartNode-black.svg"
124 | }
125 | },
126 | {
127 | "command": "substrateMarketplace.palletHomepage",
128 | "title": "Homepage",
129 | "icon": {
130 | "dark": "resources/home.svg",
131 | "light": "resources/home-black.svg"
132 | }
133 | },
134 | {
135 | "command": "substrateMarketplace.palletGithub",
136 | "title": "GitHub",
137 | "icon": {
138 | "dark": "resources/github.svg",
139 | "light": "resources/github-black.svg"
140 | }
141 | },
142 | {
143 | "command": "substrateMarketplace.palletDocumentation",
144 | "title": "Documentation",
145 | "icon": {
146 | "dark": "resources/book.svg",
147 | "light": "resources/book-black.svg"
148 | }
149 | },
150 | {
151 | "command": "substrateMarketplace.installPallet",
152 | "title": "Add pallet to runtime",
153 | "icon": {
154 | "dark": "resources/add.svg",
155 | "light": "resources/add-black.svg"
156 | }
157 | },
158 | {
159 | "command": "substrate.addAccount",
160 | "title": "Substrate: Add account",
161 | "icon": {
162 | "dark": "resources/add.svg",
163 | "light": "resources/add-black.svg"
164 | }
165 | },
166 | {
167 | "command": "substrate.createAccount",
168 | "title": "Generate new account"
169 | },
170 | {
171 | "command": "substrate.importAccount",
172 | "title": "Import account from JSON"
173 | },
174 | {
175 | "command": "substrate.removeAccount",
176 | "title": "Remove"
177 | },
178 | {
179 | "command": "substrate.exportAccount",
180 | "title": "Export"
181 | },
182 | {
183 | "command": "substrate.renameAccount",
184 | "title": "Change name"
185 | },
186 | {
187 | "command": "substrate.copyAddress",
188 | "title": "Copy address"
189 | },
190 | {
191 | "command": "substrate.compileAndDeploy",
192 | "title": "Compile & deploy contract"
193 | },
194 | {
195 | "command": "substrate.forgetContract",
196 | "title": "Forget contract"
197 | },
198 | {
199 | "command": "substrate.copyContractHash",
200 | "title": "Copy hash"
201 | },
202 | {
203 | "command": "substrate.callContractMethod",
204 | "title": "Call a contract method"
205 | },
206 | {
207 | "command": "substrate.refreshContracts",
208 | "title": "Refresh contracts list"
209 | }
210 | ],
211 | "menus": {
212 | "commandPalette": [
213 | {
214 | "command": "substrateMarketplace.palletHomepage",
215 | "when": "false"
216 | },
217 | {
218 | "command": "substrateMarketplace.palletGithub",
219 | "when": "false"
220 | },
221 | {
222 | "command": "substrateMarketplace.palletDocumentation",
223 | "when": "false"
224 | },
225 | {
226 | "command": "substrateMarketplace.installPallet",
227 | "when": "false"
228 | }
229 | ],
230 | "view/title": [
231 | {
232 | "command": "substrate.addAccount",
233 | "when": "view == substrateAccounts",
234 | "group": "navigation"
235 | },
236 | {
237 | "command": "substrate.createAccount",
238 | "when": "view == substrateAccounts"
239 | },
240 | {
241 | "command": "substrate.importAccount",
242 | "when": "view == substrateAccounts"
243 | },
244 | {
245 | "command": "substrate.compileAndDeploy",
246 | "when": "view == substrateContracts",
247 | "group": "section-1"
248 | }
249 | ],
250 | "view/item/context": [
251 | {
252 | "command": "substrate.polkadotApps",
253 | "when": "view == substrateProcesses && viewItem == theia",
254 | "group": "inline@0"
255 | },
256 | {
257 | "command": "substrate.stopProcess",
258 | "when": "view == substrateProcesses",
259 | "group": "inline@1"
260 | },
261 | {
262 | "command": "substrate.purgeChain",
263 | "when": "view == substrateNodes",
264 | "group": "inline@0"
265 | },
266 | {
267 | "command": "substrate.startNode",
268 | "when": "view == substrateNodes",
269 | "group": "inline@1"
270 | },
271 | {
272 | "command": "substrate.compileStartNode",
273 | "when": "view == substrateNodes",
274 | "group": "inline@2"
275 | },
276 | {
277 | "command": "substrateMarketplace.palletHomepage",
278 | "when": "view == substrateMarketplace && viewItem == pallet || view == substrateMarketplace && viewItem == palletInstalled || view == substrateMarketplace && viewItem == palletNoNode",
279 | "group": "inline@0"
280 | },
281 | {
282 | "command": "substrateMarketplace.palletGithub",
283 | "when": "view == substrateMarketplace && viewItem == pallet || view == substrateMarketplace && viewItem == palletInstalled || view == substrateMarketplace && viewItem == palletNoNode",
284 | "group": "inline@1"
285 | },
286 | {
287 | "command": "substrateMarketplace.palletDocumentation",
288 | "when": "view == substrateMarketplace && viewItem == pallet || view == substrateMarketplace && viewItem == palletInstalled || view == substrateMarketplace && viewItem == palletNoNode",
289 | "group": "inline@2"
290 | },
291 | {
292 | "command": "substrateMarketplace.installPallet",
293 | "when": "view == substrateMarketplace && viewItem == pallet",
294 | "group": "inline@3"
295 | },
296 | {
297 | "command": "substrate.removeAccount",
298 | "when": "view == substrateAccounts"
299 | },
300 | {
301 | "command": "substrate.exportAccount",
302 | "when": "view == substrateAccounts"
303 | },
304 | {
305 | "command": "substrate.renameAccount",
306 | "when": "view == substrateAccounts"
307 | },
308 | {
309 | "command": "substrate.copyAddress",
310 | "when": "view == substrateAccounts"
311 | },
312 | {
313 | "command": "substrate.callContractMethod",
314 | "when": "view == substrateContracts",
315 | "group": "section-1"
316 | },
317 | {
318 | "command": "substrate.copyContractHash",
319 | "when": "view == substrateContracts",
320 | "group": "section-1"
321 | },
322 | {
323 | "command": "substrate.forgetContract",
324 | "when": "view == substrateContracts",
325 | "group": "section-2"
326 | }
327 | ]
328 | }
329 | },
330 | "scripts": {
331 | "vscode:prepublish": "yarn run compile",
332 | "compile": "tsc -p ./",
333 | "watch": "tsc -watch -p ./"
334 | },
335 | "devDependencies": {
336 | "@types/node": "^10.12.21",
337 | "@types/vscode": "^1.42.0",
338 | "tslint": "^5.12.1",
339 | "typescript": "^3.5.1"
340 | },
341 | "dependencies": {
342 | "@polkadot/api": "^1.27.1",
343 | "@polkadot/api-contract": "^1.27.1",
344 | "@types/lodash": "^4.14.149",
345 | "@types/node-fetch": "^2.5.5",
346 | "array-flat-polyfill": "^1.0.1",
347 | "clipboardy": "^2.3.0",
348 | "glob": "^7.1.6",
349 | "lodash": "^4.17.19",
350 | "node-fetch": "^2.6.1",
351 | "rxjs": "^6.5.5",
352 | "string.prototype.matchall": "^4.0.2"
353 | }
354 | }
355 |
--------------------------------------------------------------------------------
/src/views/contracts/ContractsProvider.ts:
--------------------------------------------------------------------------------
1 | import * as vscode from 'vscode';
2 | import { tryShortname, resolveWhenTerminalClosed, showInputBoxValidate, wsEndpointFromCommand } from '../../util';
3 | import { Abi } from '@polkadot/api-contract';
4 | import { compactAddLength } from '@polkadot/util';
5 |
6 | import * as clipboard from 'clipboardy';
7 | import { TreeDataProvider } from '../../common/TreeDataProvider';
8 | import { Substrate, Contract } from '../../common/Substrate';
9 | import { Process } from '../../processes/Processes';
10 |
11 | const fs = require('fs');
12 | const path = require('path');
13 |
14 | export class ContractsProvider extends TreeDataProvider {
15 | substrate: Substrate;
16 |
17 | constructor(substrate: Substrate) {
18 | super();
19 | this.substrate = substrate;
20 | }
21 |
22 | getChildren(element?: ContractTreeItem): Thenable {
23 | if (element === undefined) {
24 | return Promise.resolve(this.substrate.getConnectionContracts().map(a => new ContractTreeItem(a)));
25 | }
26 | return Promise.resolve(element.children);
27 | }
28 | }
29 |
30 | export class ContractTreeItem extends vscode.TreeItem {
31 | children: [] = [];
32 | contract: Contract;
33 |
34 | constructor(contract: Contract) {
35 | super(
36 | contract.name,
37 | vscode.TreeItemCollapsibleState.None);
38 | this.description = contract.address;
39 | this.contract = contract;
40 | }
41 | }
42 |
43 | export async function setupContractsTreeView(substrate: Substrate, selectedProcess$: any, context: vscode.ExtensionContext) {
44 | const treeDataProvider = new ContractsProvider(substrate);
45 | const treeView = vscode.window.createTreeView('substrateContracts', { treeDataProvider });
46 |
47 | selectedProcess$.subscribe((process: Process) => {
48 | if (process)
49 | treeView.message = `Connected to: ${wsEndpointFromCommand(process.command)} (${tryShortname(process.nodePath) + ' • ' + process.command})`;
50 | else
51 | treeView.message = `Not connected to any node.`;
52 | vscode.commands.executeCommand('substrate.refreshContracts');
53 | });
54 |
55 | vscode.commands.registerCommand("substrate.refreshContracts", async (contractItem: ContractTreeItem) => {
56 | treeDataProvider.refresh();
57 | });
58 |
59 | vscode.commands.registerCommand("substrate.callContractMethod", async (contractItem: ContractTreeItem) => {
60 | try {
61 | const api = substrate.getConnection();
62 | if (!api) {
63 | vscode.window.showErrorMessage(`Not connected to a node.`);
64 | return;
65 | }
66 |
67 | const abi = new Abi(api.registry, contractItem.contract.abiJson);
68 |
69 | const methods = abi.abi.contract.messages;
70 | const items = methods.map(method => {
71 | const args = method.args.map((arg) => `${arg.name}: ${arg.type}`);
72 | const retDisplayName = method.returnType && method.returnType.displayName;
73 | return {
74 | label: `🧭 ${method.name}(${args.join(', ')})${retDisplayName ? `: ${retDisplayName}` : ''}`,
75 | description: method.mutates ? 'will mutate storage' : 'won\'t mutate storage',
76 | detail: `Method selector: ${method.selector}`,
77 | method
78 | };
79 | });
80 | const pickedMessage = await vscode.window.showQuickPick(items, { placeHolder: 'ex. get(): bool' });
81 | if (!pickedMessage) return;
82 | const method = pickedMessage.method;
83 |
84 | const valueTo: string | undefined = await showInputBoxValidate({
85 | ignoreFocusOut: true,
86 | prompt: 'The allotted value for this contract, i.e. the amount transferred to the contract as part of this call',
87 | value: '1000000000000000',
88 | placeHolder: 'ex. 1000000000000000'
89 | }, async (value: any) => {
90 | if (!value || !value.trim()) {
91 | return 'Value is required';
92 | }
93 | if (!value.match(/^-{0,1}\d+$/)) {
94 | return 'The value specified is not a number';
95 | }
96 | return '';
97 | });
98 | if (!valueTo) return;
99 |
100 | const maxGas: string | undefined = await showInputBoxValidate({
101 | ignoreFocusOut: true,
102 | prompt: 'The maximum amount of gas that can be used by this call',
103 | value: '1000000000000', // too much?
104 | placeHolder: 'ex. 1000000000000'
105 | }, async (value: any) => {
106 | if (!value || !value.trim()) {
107 | return 'A value is required';
108 | }
109 | if (!value.match(/^-{0,1}\d+$/)) {
110 | return 'The maximum gas specified is not a number';
111 | }
112 | return '';
113 | });
114 | if (!maxGas) return;
115 |
116 | const argsVals = [];
117 | const argsDefs = pickedMessage.method.args;
118 |
119 | while (argsDefs.length > argsVals.length) {
120 | const i: number = argsVals.length;
121 | const prompt = `${argsDefs[i].name}: ${argsDefs[i].type.displayName}`;
122 |
123 | const vale = await showInputBoxValidate({
124 | ignoreFocusOut: true,
125 | prompt
126 | }, async (value: any) => {
127 | return ''
128 | });
129 | if (vale === undefined) return; // User cancelled
130 | argsVals.push(vale);
131 | }
132 |
133 | const accounts = substrate.getAccounts();
134 | const _account = await vscode.window.showQuickPick(substrate.getAccounts().map((x: any) => x.meta.name));
135 | if (!_account) return;
136 | const accountJson = accounts.find(a => a.meta.name === _account)!;
137 |
138 | const keyring = substrate.getKeyring();
139 | const account = keyring.addFromJson(accountJson);
140 |
141 | const password: string | undefined = await showInputBoxValidate({
142 | ignoreFocusOut: true,
143 | prompt: 'Account password',
144 | placeHolder: 'ex. StrongPassword',
145 | password: true,
146 | value: '',
147 | }, async (value: any) => {
148 | try {
149 | account.decodePkcs8(value);
150 | if (account.isLocked) {
151 | return 'Failed to decode account';
152 | }
153 | } catch (e) {
154 | return 'Failed to decode account';
155 | }
156 | return '';
157 | });
158 | if (typeof password === 'undefined') return;
159 |
160 | // METHOD CALL
161 |
162 | vscode.window.showInformationMessage('Calling contract method...');
163 |
164 | try {
165 | const { nonce } = await api.query.system.account(account.address);
166 | const contractApi = api.tx.contracts ? api.tx['contracts'] : api.tx['contract'];
167 |
168 | const methodExec = abi.messages[method.name];
169 | const unsignedTransaction = contractApi.call(
170 | contractItem.description,
171 | valueTo,
172 | maxGas,
173 | methodExec(...argsVals),
174 | );
175 |
176 | const cb = await unsignedTransaction.sign(account, { nonce: nonce as any }).send(({ events = [], status }: any) => {
177 | if (status.isFinalized) {
178 | const finalized = status.asFinalized.toHex();
179 | console.log(`Completed at block hash: ${finalized}`);
180 |
181 | console.log(`Events:`);
182 | let error: string = '';
183 | events.forEach(({ phase, event: { data, method, section } }: any) => {
184 | const res = `\t ${phase.toString()} : ${section}.${method} ${data.toString()}`;
185 | if (res.indexOf('Failed') !== -1) {
186 | error += res;
187 | }
188 | console.log(res);
189 | });
190 | if (error !== '') {
191 | // Todo: Get error
192 | vscode.window.showErrorMessage(`Failed on block "${finalized}" with error: ${error}`);
193 | return;
194 | }
195 | vscode.window.showInformationMessage(`Completed on block ${finalized}`);
196 | }
197 | });
198 | // TODO Get results from contract
199 | } catch (err) {
200 | vscode.window.showErrorMessage(`Error on put code: ${err.message}`);
201 | }
202 | }
203 | catch (err) {
204 | console.error(err);
205 | vscode.window.showErrorMessage(err);
206 | }
207 | });
208 |
209 | vscode.commands.registerCommand("substrate.forgetContract", async (contractItem: ContractTreeItem) => {
210 | try {
211 | const contracts = substrate.getConnectionContracts();
212 | for (let i = 0; i < contracts.length; i++) {
213 | const contract = contracts[i];
214 | if (contract.name === contractItem.label || contract.address === contractItem.description) {
215 | contracts.splice(i, 1);
216 | }
217 | }
218 | await substrate.updateConnectionContracts(contracts);
219 | } catch (err) {
220 | vscode.window.showErrorMessage('You are not connected to a node');
221 | }
222 | await vscode.commands.executeCommand('substrate.refreshContracts');
223 | vscode.window.showInformationMessage(`Successfully removed contract "${contractItem.label}"`);
224 | });
225 |
226 | vscode.commands.registerCommand("substrate.copyContractHash", async (contractItem: ContractTreeItem) => {
227 | try {
228 | await clipboard.write((contractItem as any).description);
229 | vscode.window.showInformationMessage('Hash copied to clipboard');
230 | } catch (err) {
231 | vscode.window.showErrorMessage(`Failed to copy hash to clipboard: ${err.message}`);
232 | }
233 | });
234 |
235 | vscode.commands.registerCommand("substrate.compileAndDeploy", async () => {
236 | try {
237 | const api = substrate.getConnection();
238 | if (!api) {
239 | vscode.window.showErrorMessage('Please connect to a node first.');
240 | return;
241 | }
242 | if (!api.tx.contracts && !api.tx.contract) {
243 | vscode.window.showErrorMessage('The selected process doesn\'t support smart contracts.');
244 | return;
245 | }
246 |
247 | const folders = await vscode.window.showOpenDialog({ canSelectFiles: false, canSelectFolders: true, canSelectMany: false, openLabel: 'Select contract folder' })
248 | if (folders === undefined) return;
249 |
250 | const term = vscode.window.createTerminal({ name: 'Compiling contract', cwd: folders[0] });
251 | term.sendText('cargo +nightly contract build && cargo +nightly contract generate-metadata && exit');
252 | term.show();
253 |
254 | await resolveWhenTerminalClosed(term);
255 |
256 | const wasmPath = path.join(folders[0].fsPath, 'target', path.basename(folders[0].fsPath) + '.wasm');
257 | const abiPath = path.join(folders[0].fsPath, 'target', 'metadata.json');
258 | const wasm: Uint8Array = await fs.promises.readFile(wasmPath);
259 | const isWasmValid = wasm.subarray(0, 4).join(',') === '0,97,115,109'; // '\0asm'
260 | if (!isWasmValid) {
261 | console.error('Invalid code');
262 | throw Error('Invalid code');
263 | }
264 | const compiledContract = compactAddLength(wasm);
265 |
266 | const name: string | undefined = await showInputBoxValidate({
267 | ignoreFocusOut: true,
268 | prompt: 'Contract name',
269 | placeHolder: 'ex. Flipper contract',
270 | value: 'flip'
271 | }, async (value: any) => {
272 | if (!value || !value.trim()) {
273 | return 'A name is required';
274 | }
275 | return '';
276 | });
277 | if (!name) return;
278 |
279 | const accounts = substrate.getAccounts();
280 | const _account = await vscode.window.showQuickPick(substrate.getAccounts().map((x: any) => x.meta.name));
281 | if (!_account) return;
282 | const accountJson = accounts.find(a => a.meta.name === _account)!;
283 |
284 | const keyring = substrate.getKeyring();
285 | const account = keyring.addFromJson(accountJson);
286 |
287 | const password: string | undefined = await showInputBoxValidate({
288 | ignoreFocusOut: true,
289 | prompt: 'Account password',
290 | placeHolder: 'ex. StrongPassword',
291 | password: true,
292 | value: '',
293 | }, async (value: any) => {
294 | try {
295 | account.decodePkcs8(value);
296 | if (account.isLocked) {
297 | return 'Failed to decode account';
298 | }
299 | } catch (e) {
300 | return 'Failed to decode account';
301 | }
302 | return '';
303 | });
304 | if (typeof password === 'undefined') return;
305 |
306 | const abiBytes: Uint8Array = await fs.promises.readFile(abiPath);
307 | const abiJson = JSON.parse(abiBytes.toString());
308 | const abi = new Abi(api.registry, abiJson);
309 | const constructors = abi.abi.contract.constructors;
310 | const _constructor = await vscode.window.showQuickPick(constructors.map((x: any) => x.name), { placeHolder: 'Choose constructor' });
311 | if (!_constructor) return;
312 | const constructorI = constructors.findIndex(x => x.name === _constructor)!;
313 | const constructor = constructors.find(x => x.name === _constructor)!;
314 | const args = constructor.args
315 |
316 | const constructorParams: any[] = [];
317 | while (args.length > constructorParams.length) {
318 | const i = constructorParams.length;
319 | const prompt = `${args[i].name}: ${args[i].type.displayName}`;
320 |
321 | constructorParams.push(await showInputBoxValidate({
322 | ignoreFocusOut: true,
323 | prompt
324 | }, async (value: any) => {
325 | return ''
326 | }));
327 | }
328 |
329 | const endowment: string | undefined = await showInputBoxValidate({
330 | ignoreFocusOut: true,
331 | prompt: 'The allotted endowment for this contract, i.e. the amount transferred to the contract upon instantiation.',
332 | value: '1000000000000000',
333 | placeHolder: 'ex. 1000000000000000'
334 | }, async (value: any) => {
335 | if (!value || !value.trim()) {
336 | return 'Endowment is required';
337 | }
338 | if (!value.match(/^-{0,1}\d+$/)) {
339 | return 'The endowment specified is not a number';
340 | }
341 | return '';
342 | });
343 | if (!endowment) return;
344 |
345 | const maxGasDeployment: string | undefined = await showInputBoxValidate({
346 | ignoreFocusOut: true,
347 | prompt: 'The maximum amount of gas that can be used for the deployment',
348 | value: '1000000000000',
349 | placeHolder: 'ex. 1000000000000'
350 | }, async (value: any) => {
351 | if (!value || !value.trim()) {
352 | return 'A value is required';
353 | }
354 | if (!value.match(/^-{0,1}\d+$/)) {
355 | return 'The maximum gas specified is not a number';
356 | }
357 | return '';
358 | });
359 | if (!maxGasDeployment) return;
360 |
361 | // PUT CODE
362 |
363 | vscode.window.showInformationMessage('Uploading WASM...');
364 |
365 | try {
366 | const { nonce } = await api.query.system.account(account.address);
367 | const contractApi = api.tx.contracts ? api.tx['contracts'] : api.tx['contract'];
368 | const unsignedTransaction = contractApi.putCode(compiledContract);
369 |
370 | let code_hash = '';
371 | unsignedTransaction.sign(account, { nonce: nonce as any }).send(async ({ events = [], status }: any) => {
372 | if (status.isFinalized) {
373 | const finalized = status.asFinalized.toHex();
374 | console.log(`Completed at block hash: ${finalized}`);
375 |
376 | console.log('Events:');
377 | let error: string = '';
378 | let resultHash: string = '';
379 | events.forEach(({ phase, event: { data, method, section } }: any) => {
380 | const res = `\t ${phase.toString()} : ${section}.${method} ${data.toString()}`;
381 | if (res.indexOf('Failed') !== -1) {
382 | error += res;
383 | }
384 | if (res.indexOf('contracts.CodeStored') !== -1) {
385 | resultHash = res.substring(
386 | res.lastIndexOf('["') + 2,
387 | res.lastIndexOf('"]'),
388 | );
389 | }
390 | console.log(res);
391 | });
392 | if (error !== '') {
393 | // Todo: Get error
394 | vscode.window.showErrorMessage(`Failed on block "${finalized}" with error: ${error}`);
395 | return;
396 | }
397 | if (resultHash === '') {
398 | vscode.window.showErrorMessage(`Completed on block "${finalized}" but failed to get event result`);
399 | return;
400 | }
401 | console.log(`Completed on block ${finalized} with code hash ${resultHash}`);
402 | code_hash = resultHash;
403 |
404 | // Deploying contract
405 |
406 | vscode.window.showInformationMessage('Instantiating contract...');
407 |
408 | // DEPLOY CONTRACT
409 | try {
410 | const { nonce } = await api.query.system.account(account.address);
411 | const unsignedTransaction = contractApi.instantiate(
412 | endowment, // +3 more 0's
413 | maxGasDeployment,
414 | code_hash,
415 | abi.constructors[constructorI](...constructorParams) // TODO test when passing parameters
416 | );
417 | await unsignedTransaction.sign(account, { nonce: nonce as any }).send(({ events = [], status }: any) => {
418 | if (status.isFinalized) {
419 | const finalized = status.asFinalized.toHex();
420 | console.log(`Completed at block hash: ${finalized}`);
421 |
422 | console.log('Events:');
423 | let error: string = '';
424 | let resultHash: string = '';
425 | events.forEach(({ phase, event: { data, method, section } }: any) => {
426 | const res = `\t ${phase.toString()} : ${section}.${method} ${data.toString()}`;
427 | if (res.indexOf('Failed') !== -1) {
428 | error += res;
429 | }
430 | if (res.indexOf('contracts.Instantiated') !== -1) {
431 | resultHash = res.substring(
432 | res.lastIndexOf('["') + 2,
433 | res.lastIndexOf('",'),
434 | );
435 | }
436 | console.log(res);
437 | });
438 | if (error !== '') {
439 | vscode.window.showErrorMessage(`Failed on block "${finalized}" with error: ${error}`);
440 | return;
441 | }
442 | if (resultHash === '') {
443 | vscode.window.showWarningMessage(`Completed on block "${finalized}" but failed to get event result`);
444 | return;
445 | }
446 | substrate.saveContract(name, resultHash, abiJson).catch(err => {
447 | vscode.window.showErrorMessage(`Failed to store contract: ${err.message}`);
448 | });
449 | vscode.window.showInformationMessage(`Completed on block ${finalized} with code hash ${resultHash}`);
450 | }
451 | });
452 | } catch (err) {
453 | vscode.window.showErrorMessage(`Error on deploy contract: ${err.message}`);
454 | }
455 | }
456 | });
457 | } catch (err) {
458 | vscode.window.showErrorMessage(`Error on put code: ${err.message}`);
459 | }
460 | } catch (surerr) {
461 | vscode.window.showErrorMessage(surerr);
462 | }
463 | });
464 | }
--------------------------------------------------------------------------------
/resources/playground.png.base64:
--------------------------------------------------------------------------------
1 | 
--------------------------------------------------------------------------------
/yarn.lock:
--------------------------------------------------------------------------------
1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
2 | # yarn lockfile v1
3 |
4 |
5 | "@babel/code-frame@^7.0.0":
6 | version "7.8.3"
7 | resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.8.3.tgz#33e25903d7481181534e12ec0a25f16b6fcf419e"
8 | integrity sha512-a9gxpmdXtZEInkCSHUJDLHZVBgb1QS0jhss4cPP93EW7s+uC5bikET2twEF3KV+7rDblJcmNvTR7VJejqd2C2g==
9 | dependencies:
10 | "@babel/highlight" "^7.8.3"
11 |
12 | "@babel/highlight@^7.8.3":
13 | version "7.8.3"
14 | resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.8.3.tgz#28f173d04223eaaa59bc1d439a3836e6d1265797"
15 | integrity sha512-PX4y5xQUvy0fnEVHrYOarRPXVWafSjTW9T0Hab8gVIawpl2Sj0ORyrygANq+KjcNlSSTw0YCLSNA8OyZ1I4yEg==
16 | dependencies:
17 | chalk "^2.0.0"
18 | esutils "^2.0.2"
19 | js-tokens "^4.0.0"
20 |
21 | "@babel/runtime@^7.10.5", "@babel/runtime@^7.11.0":
22 | version "7.11.2"
23 | resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.11.2.tgz#f549c13c754cc40b87644b9fa9f09a6a95fe0736"
24 | integrity sha512-TeWkU52so0mPtDcaCTxNBI/IHiz0pZgr8VEFqXFtZWpYD08ZB6FaSwVAS8MKRQAP3bYKiVjwysOJgMFY28o6Tw==
25 | dependencies:
26 | regenerator-runtime "^0.13.4"
27 |
28 | "@polkadot/api-contract@^1.27.1":
29 | version "1.27.1"
30 | resolved "https://registry.yarnpkg.com/@polkadot/api-contract/-/api-contract-1.27.1.tgz#52b45b7de8ce35b0d278f4c37eafdcd0b0b7b8c7"
31 | integrity sha512-4Md0LXysEctONz38u7TG9EEFHL2rd9s25nzskGOuFm3FoLomFrvUHGsheIr0zM6wDg8yScRh9zvdkc8EvG4sWQ==
32 | dependencies:
33 | "@babel/runtime" "^7.11.0"
34 | "@polkadot/api" "1.27.1"
35 | "@polkadot/rpc-core" "1.27.1"
36 | "@polkadot/types" "1.27.1"
37 | "@polkadot/util" "^3.0.1"
38 | bn.js "^5.1.2"
39 | rxjs "^6.6.2"
40 |
41 | "@polkadot/api-derive@1.27.1":
42 | version "1.27.1"
43 | resolved "https://registry.yarnpkg.com/@polkadot/api-derive/-/api-derive-1.27.1.tgz#5001f19cb29ea4d7924412950b41e46fcca45d81"
44 | integrity sha512-WPZEZ9THeYOkkGzDDsCGSTrWJuWu6qegmWLo52pqD7enmP3dADDoKqPzszHVbEAfc+wYtc5Ye4EogUAiK30iMg==
45 | dependencies:
46 | "@babel/runtime" "^7.11.0"
47 | "@polkadot/api" "1.27.1"
48 | "@polkadot/rpc-core" "1.27.1"
49 | "@polkadot/rpc-provider" "1.27.1"
50 | "@polkadot/types" "1.27.1"
51 | "@polkadot/util" "^3.0.1"
52 | "@polkadot/util-crypto" "^3.0.1"
53 | bn.js "^5.1.2"
54 | memoizee "^0.4.14"
55 | rxjs "^6.6.2"
56 |
57 | "@polkadot/api@1.27.1", "@polkadot/api@^1.27.1":
58 | version "1.27.1"
59 | resolved "https://registry.yarnpkg.com/@polkadot/api/-/api-1.27.1.tgz#dcd3ae5d8699ea5555369b9594888eb1008dd7fe"
60 | integrity sha512-ANMLmNR1PfXLoQ7Ysep9sM/2U3oh3fjFI+9P4Yy9Pv+XMvCROaGAcvxoG+hHfWFz6OAT7ABwyiocV93PuOpAww==
61 | dependencies:
62 | "@babel/runtime" "^7.11.0"
63 | "@polkadot/api-derive" "1.27.1"
64 | "@polkadot/keyring" "^3.0.1"
65 | "@polkadot/metadata" "1.27.1"
66 | "@polkadot/rpc-core" "1.27.1"
67 | "@polkadot/rpc-provider" "1.27.1"
68 | "@polkadot/types" "1.27.1"
69 | "@polkadot/types-known" "1.27.1"
70 | "@polkadot/util" "^3.0.1"
71 | "@polkadot/util-crypto" "^3.0.1"
72 | bn.js "^5.1.2"
73 | eventemitter3 "^4.0.4"
74 | rxjs "^6.6.2"
75 |
76 | "@polkadot/keyring@^3.0.1":
77 | version "3.0.1"
78 | resolved "https://registry.yarnpkg.com/@polkadot/keyring/-/keyring-3.0.1.tgz#3944079697c15b2af81e1f57b1c4aeab703a4fef"
79 | integrity sha512-vAHSBnisiDYHsBbEzAgIpuwQp3vIDN2uWQ/1wAE2BrKzXCBQM7RrF3LRcLFySk0xzQoDs7AP1TlPoakxJ/C/Qw==
80 | dependencies:
81 | "@babel/runtime" "^7.10.5"
82 | "@polkadot/util" "3.0.1"
83 | "@polkadot/util-crypto" "3.0.1"
84 |
85 | "@polkadot/metadata@1.27.1":
86 | version "1.27.1"
87 | resolved "https://registry.yarnpkg.com/@polkadot/metadata/-/metadata-1.27.1.tgz#40b8e26f9eee30df5b6790aaab406dae1e6d3152"
88 | integrity sha512-jOsWvpGNFkV3NGSGVTzyir0PKinxoa97CYEQj8tJTU7iHDPnSs5R/BEmI0+1hg4ZM/47n+YU8d0lLktD6wbMOA==
89 | dependencies:
90 | "@babel/runtime" "^7.11.0"
91 | "@polkadot/types" "1.27.1"
92 | "@polkadot/types-known" "1.27.1"
93 | "@polkadot/util" "^3.0.1"
94 | "@polkadot/util-crypto" "^3.0.1"
95 | bn.js "^5.1.2"
96 |
97 | "@polkadot/rpc-core@1.27.1":
98 | version "1.27.1"
99 | resolved "https://registry.yarnpkg.com/@polkadot/rpc-core/-/rpc-core-1.27.1.tgz#afe7d1890c779eaf2bec9bb1336b5ba2ff1ba15b"
100 | integrity sha512-HxLHHdV3bDqTltsEedCRGiZeTGmeahnk6OEGyysFOW3PFIrygwwuYa0Mo10lS93dwy9xZw4oE3h9qZqox2mGmQ==
101 | dependencies:
102 | "@babel/runtime" "^7.11.0"
103 | "@polkadot/metadata" "1.27.1"
104 | "@polkadot/rpc-provider" "1.27.1"
105 | "@polkadot/types" "1.27.1"
106 | "@polkadot/util" "^3.0.1"
107 | memoizee "^0.4.14"
108 | rxjs "^6.6.2"
109 |
110 | "@polkadot/rpc-provider@1.27.1":
111 | version "1.27.1"
112 | resolved "https://registry.yarnpkg.com/@polkadot/rpc-provider/-/rpc-provider-1.27.1.tgz#c0307a9e9c658f4072ad94ac6e98e2bd78bb1ee6"
113 | integrity sha512-12lpsmHNYLZ3p0X0h643U+sw+WE/kzVB9Q0Y0RA9Recq794cBGiEgK/nmVjT5hrTEk+2+qdtu+CqficWt2FeQw==
114 | dependencies:
115 | "@babel/runtime" "^7.11.0"
116 | "@polkadot/metadata" "1.27.1"
117 | "@polkadot/types" "1.27.1"
118 | "@polkadot/util" "^3.0.1"
119 | "@polkadot/util-crypto" "^3.0.1"
120 | bn.js "^5.1.2"
121 | eventemitter3 "^4.0.4"
122 | isomorphic-fetch "^2.2.1"
123 | websocket "^1.0.31"
124 |
125 | "@polkadot/types-known@1.27.1":
126 | version "1.27.1"
127 | resolved "https://registry.yarnpkg.com/@polkadot/types-known/-/types-known-1.27.1.tgz#d6c48c7b075cd0a9772fcc5df8d98f77b0da35f3"
128 | integrity sha512-9HzHJXznuuG0EQPR3XN931smIzA5usOP2D4Te/uyjiqHgXGUv1EU/vPYaacYYRrjLNdytNcF3HbW97fxu+fC6w==
129 | dependencies:
130 | "@babel/runtime" "^7.11.0"
131 | "@polkadot/types" "1.27.1"
132 | "@polkadot/util" "^3.0.1"
133 | bn.js "^5.1.2"
134 |
135 | "@polkadot/types@1.27.1":
136 | version "1.27.1"
137 | resolved "https://registry.yarnpkg.com/@polkadot/types/-/types-1.27.1.tgz#243f796178f9308a8e58f13fa13d0f3ed1fd2bc8"
138 | integrity sha512-6sX+0/cBUEd/Pt6Y4JSjUI44nqH6qyuRcs6bte2OsEsm06XTEMoIqICPHoOS36Sd/ux7IA/gTZgEj39C9mJWBQ==
139 | dependencies:
140 | "@babel/runtime" "^7.11.0"
141 | "@polkadot/metadata" "1.27.1"
142 | "@polkadot/util" "^3.0.1"
143 | "@polkadot/util-crypto" "^3.0.1"
144 | "@types/bn.js" "^4.11.6"
145 | bn.js "^5.1.2"
146 | memoizee "^0.4.14"
147 | rxjs "^6.6.2"
148 |
149 | "@polkadot/util-crypto@3.0.1", "@polkadot/util-crypto@^3.0.1":
150 | version "3.0.1"
151 | resolved "https://registry.yarnpkg.com/@polkadot/util-crypto/-/util-crypto-3.0.1.tgz#800746a39a00e5aa7dc7b901bbde0f5e3e0be60b"
152 | integrity sha512-4G5kzNfqa/nQGuTtoFsy3DESApc8BTgTHbAvLwSkxzM3j8YsvC5ayJ3AFYvM2UT2PDwXmrFx4cwRnYsYZvhC9A==
153 | dependencies:
154 | "@babel/runtime" "^7.10.5"
155 | "@polkadot/util" "3.0.1"
156 | "@polkadot/wasm-crypto" "^1.2.1"
157 | base-x "^3.0.8"
158 | bip39 "^3.0.2"
159 | blakejs "^1.1.0"
160 | bn.js "^5.1.2"
161 | elliptic "^6.5.3"
162 | js-sha3 "^0.8.0"
163 | pbkdf2 "^3.1.1"
164 | scryptsy "^2.1.0"
165 | tweetnacl "^1.0.3"
166 | xxhashjs "^0.2.2"
167 |
168 | "@polkadot/util@3.0.1", "@polkadot/util@^3.0.1":
169 | version "3.0.1"
170 | resolved "https://registry.yarnpkg.com/@polkadot/util/-/util-3.0.1.tgz#f7ed9d81d745136aa6d6ad57277ee05c88f32784"
171 | integrity sha512-WvH+seT03YQ+6dWJqo285uYHsDvMEGzgYQILEclzQo8xExeCYLIX6GptpW0vGycVxdZmmCdDmUFbcQSRsFawYA==
172 | dependencies:
173 | "@babel/runtime" "^7.10.5"
174 | "@types/bn.js" "^4.11.6"
175 | bn.js "^5.1.2"
176 | camelcase "^5.3.1"
177 | chalk "^4.1.0"
178 | ip-regex "^4.1.0"
179 |
180 | "@polkadot/wasm-crypto@^1.2.1":
181 | version "1.2.1"
182 | resolved "https://registry.yarnpkg.com/@polkadot/wasm-crypto/-/wasm-crypto-1.2.1.tgz#2189702447acd28d763886359576c87562241767"
183 | integrity sha512-nckIoZBV4nBZdeKwFwH5t7skS7L7GO5EFUl5B1F6uCjUfdNpDz3DtqbYQHcLdCZNmG4TDLg6w/1J+rkl2SiUZw==
184 |
185 | "@types/bn.js@^4.11.6":
186 | version "4.11.6"
187 | resolved "https://registry.yarnpkg.com/@types/bn.js/-/bn.js-4.11.6.tgz#c306c70d9358aaea33cd4eda092a742b9505967c"
188 | integrity sha512-pqr857jrp2kPuO9uRjZ3PwnJTjoQy+fcdxvBTvHm6dkmEL9q+hDD/2j/0ELOBPtPnS8LjCX0gI9nbl8lVkadpg==
189 | dependencies:
190 | "@types/node" "*"
191 |
192 | "@types/color-name@^1.1.1":
193 | version "1.1.1"
194 | resolved "https://registry.yarnpkg.com/@types/color-name/-/color-name-1.1.1.tgz#1c1261bbeaa10a8055bbc5d8ab84b7b2afc846a0"
195 | integrity sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==
196 |
197 | "@types/lodash@^4.14.149":
198 | version "4.14.149"
199 | resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.149.tgz#1342d63d948c6062838fbf961012f74d4e638440"
200 | integrity sha512-ijGqzZt/b7BfzcK9vTrS6MFljQRPn5BFWOx8oE0GYxribu6uV+aA9zZuXI1zc/etK9E8nrgdoF2+LgUw7+9tJQ==
201 |
202 | "@types/node-fetch@^2.5.5":
203 | version "2.5.5"
204 | resolved "https://registry.yarnpkg.com/@types/node-fetch/-/node-fetch-2.5.5.tgz#cd264e20a81f4600a6c52864d38e7fef72485e92"
205 | integrity sha512-IWwjsyYjGw+em3xTvWVQi5MgYKbRs0du57klfTaZkv/B24AEQ/p/IopNeqIYNy3EsfHOpg8ieQSDomPcsYMHpA==
206 | dependencies:
207 | "@types/node" "*"
208 | form-data "^3.0.0"
209 |
210 | "@types/node@*":
211 | version "13.9.1"
212 | resolved "https://registry.yarnpkg.com/@types/node/-/node-13.9.1.tgz#96f606f8cd67fb018847d9b61e93997dabdefc72"
213 | integrity sha512-E6M6N0blf/jiZx8Q3nb0vNaswQeEyn0XlupO+xN6DtJ6r6IT4nXrTry7zhIfYvFCl3/8Cu6WIysmUBKiqV0bqQ==
214 |
215 | "@types/node@11.11.6":
216 | version "11.11.6"
217 | resolved "https://registry.yarnpkg.com/@types/node/-/node-11.11.6.tgz#df929d1bb2eee5afdda598a41930fe50b43eaa6a"
218 | integrity sha512-Exw4yUWMBXM3X+8oqzJNRqZSwUAaS4+7NdvHqQuFi/d+synz++xmX3QIf+BFqneW8N31R8Ky+sikfZUXq07ggQ==
219 |
220 | "@types/node@^10.12.21":
221 | version "10.17.18"
222 | resolved "https://registry.yarnpkg.com/@types/node/-/node-10.17.18.tgz#ae364d97382aacdebf583fa4e7132af2dfe56a0c"
223 | integrity sha512-DQ2hl/Jl3g33KuAUOcMrcAOtsbzb+y/ufakzAdeK9z/H/xsvkpbETZZbPNMIiQuk24f5ZRMCcZIViAwyFIiKmg==
224 |
225 | "@types/vscode@^1.42.0":
226 | version "1.43.0"
227 | resolved "https://registry.yarnpkg.com/@types/vscode/-/vscode-1.43.0.tgz#22276e60034c693b33117f1068ffaac0e89522db"
228 | integrity sha512-kIaR9qzd80rJOxePKpCB/mdy00mz8Apt2QA5Y6rdrKFn13QNFNeP3Hzmsf37Bwh/3cS7QjtAeGSK7wSqAU0sYQ==
229 |
230 | ansi-styles@^3.2.1:
231 | version "3.2.1"
232 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d"
233 | integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==
234 | dependencies:
235 | color-convert "^1.9.0"
236 |
237 | ansi-styles@^4.1.0:
238 | version "4.2.1"
239 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.2.1.tgz#90ae75c424d008d2624c5bf29ead3177ebfcf359"
240 | integrity sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==
241 | dependencies:
242 | "@types/color-name" "^1.1.1"
243 | color-convert "^2.0.1"
244 |
245 | arch@^2.1.1:
246 | version "2.1.2"
247 | resolved "https://registry.yarnpkg.com/arch/-/arch-2.1.2.tgz#0c52bbe7344bb4fa260c443d2cbad9c00ff2f0bf"
248 | integrity sha512-NTBIIbAfkJeIletyABbVtdPgeKfDafR+1mZV/AyyfC1UkVkp9iUjV+wwmqtUgphHYajbI86jejBJp5e+jkGTiQ==
249 |
250 | argparse@^1.0.7:
251 | version "1.0.10"
252 | resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.10.tgz#bcd6791ea5ae09725e17e5ad988134cd40b3d911"
253 | integrity sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==
254 | dependencies:
255 | sprintf-js "~1.0.2"
256 |
257 | array-flat-polyfill@^1.0.1:
258 | version "1.0.1"
259 | resolved "https://registry.yarnpkg.com/array-flat-polyfill/-/array-flat-polyfill-1.0.1.tgz#1e3a4255be619dfbffbfd1d635c1cf357cd034e7"
260 | integrity sha512-hfJmKupmQN0lwi0xG6FQ5U8Rd97RnIERplymOv/qpq8AoNKPPAnxJadjFA23FNWm88wykh9HmpLJUUwUtNU/iw==
261 |
262 | asynckit@^0.4.0:
263 | version "0.4.0"
264 | resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79"
265 | integrity sha1-x57Zf380y48robyXkLzDZkdLS3k=
266 |
267 | balanced-match@^1.0.0:
268 | version "1.0.0"
269 | resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz#89b4d199ab2bee49de164ea02b89ce462d71b767"
270 | integrity sha1-ibTRmasr7kneFk6gK4nORi1xt2c=
271 |
272 | base-x@^3.0.8:
273 | version "3.0.8"
274 | resolved "https://registry.yarnpkg.com/base-x/-/base-x-3.0.8.tgz#1e1106c2537f0162e8b52474a557ebb09000018d"
275 | integrity sha512-Rl/1AWP4J/zRrk54hhlxH4drNxPJXYUaKffODVI53/dAsV4t9fBxyxYKAVPU1XBHxYwOWP9h9H0hM2MVw4YfJA==
276 | dependencies:
277 | safe-buffer "^5.0.1"
278 |
279 | bip39@^3.0.2:
280 | version "3.0.2"
281 | resolved "https://registry.yarnpkg.com/bip39/-/bip39-3.0.2.tgz#2baf42ff3071fc9ddd5103de92e8f80d9257ee32"
282 | integrity sha512-J4E1r2N0tUylTKt07ibXvhpT2c5pyAFgvuA5q1H9uDy6dEGpjV8jmymh3MTYJDLCNbIVClSB9FbND49I6N24MQ==
283 | dependencies:
284 | "@types/node" "11.11.6"
285 | create-hash "^1.1.0"
286 | pbkdf2 "^3.0.9"
287 | randombytes "^2.0.1"
288 |
289 | blakejs@^1.1.0:
290 | version "1.1.0"
291 | resolved "https://registry.yarnpkg.com/blakejs/-/blakejs-1.1.0.tgz#69df92ef953aa88ca51a32df6ab1c54a155fc7a5"
292 | integrity sha1-ad+S75U6qIylGjLfarHFShVfx6U=
293 |
294 | bn.js@^4.4.0:
295 | version "4.11.9"
296 | resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.11.9.tgz#26d556829458f9d1e81fc48952493d0ba3507828"
297 | integrity sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==
298 |
299 | bn.js@^5.1.2:
300 | version "5.1.2"
301 | resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.1.2.tgz#c9686902d3c9a27729f43ab10f9d79c2004da7b0"
302 | integrity sha512-40rZaf3bUNKTVYu9sIeeEGOg7g14Yvnj9kH7b50EiwX0Q7A6umbvfI5tvHaOERH0XigqKkfLkFQxzb4e6CIXnA==
303 |
304 | brace-expansion@^1.1.7:
305 | version "1.1.11"
306 | resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd"
307 | integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==
308 | dependencies:
309 | balanced-match "^1.0.0"
310 | concat-map "0.0.1"
311 |
312 | brorand@^1.0.1:
313 | version "1.1.0"
314 | resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f"
315 | integrity sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=
316 |
317 | builtin-modules@^1.1.1:
318 | version "1.1.1"
319 | resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-1.1.1.tgz#270f076c5a72c02f5b65a47df94c5fe3a278892f"
320 | integrity sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=
321 |
322 | camelcase@^5.3.1:
323 | version "5.3.1"
324 | resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.3.1.tgz#e3c9b31569e106811df242f715725a1f4c494320"
325 | integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==
326 |
327 | chalk@^2.0.0, chalk@^2.3.0:
328 | version "2.4.2"
329 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424"
330 | integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==
331 | dependencies:
332 | ansi-styles "^3.2.1"
333 | escape-string-regexp "^1.0.5"
334 | supports-color "^5.3.0"
335 |
336 | chalk@^4.1.0:
337 | version "4.1.0"
338 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.0.tgz#4e14870a618d9e2edd97dd8345fd9d9dc315646a"
339 | integrity sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==
340 | dependencies:
341 | ansi-styles "^4.1.0"
342 | supports-color "^7.1.0"
343 |
344 | cipher-base@^1.0.1, cipher-base@^1.0.3:
345 | version "1.0.4"
346 | resolved "https://registry.yarnpkg.com/cipher-base/-/cipher-base-1.0.4.tgz#8760e4ecc272f4c363532f926d874aae2c1397de"
347 | integrity sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==
348 | dependencies:
349 | inherits "^2.0.1"
350 | safe-buffer "^5.0.1"
351 |
352 | clipboardy@^2.3.0:
353 | version "2.3.0"
354 | resolved "https://registry.yarnpkg.com/clipboardy/-/clipboardy-2.3.0.tgz#3c2903650c68e46a91b388985bc2774287dba290"
355 | integrity sha512-mKhiIL2DrQIsuXMgBgnfEHOZOryC7kY7YO//TN6c63wlEm3NG5tz+YgY5rVi29KCmq/QQjKYvM7a19+MDOTHOQ==
356 | dependencies:
357 | arch "^2.1.1"
358 | execa "^1.0.0"
359 | is-wsl "^2.1.1"
360 |
361 | color-convert@^1.9.0:
362 | version "1.9.3"
363 | resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8"
364 | integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==
365 | dependencies:
366 | color-name "1.1.3"
367 |
368 | color-convert@^2.0.1:
369 | version "2.0.1"
370 | resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3"
371 | integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==
372 | dependencies:
373 | color-name "~1.1.4"
374 |
375 | color-name@1.1.3:
376 | version "1.1.3"
377 | resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25"
378 | integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=
379 |
380 | color-name@~1.1.4:
381 | version "1.1.4"
382 | resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
383 | integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
384 |
385 | combined-stream@^1.0.8:
386 | version "1.0.8"
387 | resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f"
388 | integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==
389 | dependencies:
390 | delayed-stream "~1.0.0"
391 |
392 | commander@^2.12.1:
393 | version "2.20.3"
394 | resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33"
395 | integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==
396 |
397 | concat-map@0.0.1:
398 | version "0.0.1"
399 | resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b"
400 | integrity sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=
401 |
402 | create-hash@^1.1.0, create-hash@^1.1.2:
403 | version "1.2.0"
404 | resolved "https://registry.yarnpkg.com/create-hash/-/create-hash-1.2.0.tgz#889078af11a63756bcfb59bd221996be3a9ef196"
405 | integrity sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==
406 | dependencies:
407 | cipher-base "^1.0.1"
408 | inherits "^2.0.1"
409 | md5.js "^1.3.4"
410 | ripemd160 "^2.0.1"
411 | sha.js "^2.4.0"
412 |
413 | create-hmac@^1.1.4:
414 | version "1.1.7"
415 | resolved "https://registry.yarnpkg.com/create-hmac/-/create-hmac-1.1.7.tgz#69170c78b3ab957147b2b8b04572e47ead2243ff"
416 | integrity sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==
417 | dependencies:
418 | cipher-base "^1.0.3"
419 | create-hash "^1.1.0"
420 | inherits "^2.0.1"
421 | ripemd160 "^2.0.0"
422 | safe-buffer "^5.0.1"
423 | sha.js "^2.4.8"
424 |
425 | cross-spawn@^6.0.0:
426 | version "6.0.5"
427 | resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-6.0.5.tgz#4a5ec7c64dfae22c3a14124dbacdee846d80cbc4"
428 | integrity sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==
429 | dependencies:
430 | nice-try "^1.0.4"
431 | path-key "^2.0.1"
432 | semver "^5.5.0"
433 | shebang-command "^1.2.0"
434 | which "^1.2.9"
435 |
436 | cuint@^0.2.2:
437 | version "0.2.2"
438 | resolved "https://registry.yarnpkg.com/cuint/-/cuint-0.2.2.tgz#408086d409550c2631155619e9fa7bcadc3b991b"
439 | integrity sha1-QICG1AlVDCYxFVYZ6fp7ytw7mRs=
440 |
441 | d@1, d@^1.0.1:
442 | version "1.0.1"
443 | resolved "https://registry.yarnpkg.com/d/-/d-1.0.1.tgz#8698095372d58dbee346ffd0c7093f99f8f9eb5a"
444 | integrity sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==
445 | dependencies:
446 | es5-ext "^0.10.50"
447 | type "^1.0.1"
448 |
449 | debug@^2.2.0:
450 | version "2.6.9"
451 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
452 | integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==
453 | dependencies:
454 | ms "2.0.0"
455 |
456 | define-properties@^1.1.2, define-properties@^1.1.3:
457 | version "1.1.3"
458 | resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.3.tgz#cf88da6cbee26fe6db7094f61d870cbd84cee9f1"
459 | integrity sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==
460 | dependencies:
461 | object-keys "^1.0.12"
462 |
463 | delayed-stream@~1.0.0:
464 | version "1.0.0"
465 | resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619"
466 | integrity sha1-3zrhmayt+31ECqrgsp4icrJOxhk=
467 |
468 | diff@^4.0.1:
469 | version "4.0.2"
470 | resolved "https://registry.yarnpkg.com/diff/-/diff-4.0.2.tgz#60f3aecb89d5fae520c11aa19efc2bb982aade7d"
471 | integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==
472 |
473 | elliptic@^6.5.3:
474 | version "6.5.3"
475 | resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.3.tgz#cb59eb2efdaf73a0bd78ccd7015a62ad6e0f93d6"
476 | integrity sha512-IMqzv5wNQf+E6aHeIqATs0tOLeOTwj1QKbRcS3jBbYkl5oLAserA8yJTT7/VyHUYG91PRmPyeQDObKLPpeS4dw==
477 | dependencies:
478 | bn.js "^4.4.0"
479 | brorand "^1.0.1"
480 | hash.js "^1.0.0"
481 | hmac-drbg "^1.0.0"
482 | inherits "^2.0.1"
483 | minimalistic-assert "^1.0.0"
484 | minimalistic-crypto-utils "^1.0.0"
485 |
486 | encoding@^0.1.11:
487 | version "0.1.12"
488 | resolved "https://registry.yarnpkg.com/encoding/-/encoding-0.1.12.tgz#538b66f3ee62cd1ab51ec323829d1f9480c74beb"
489 | integrity sha1-U4tm8+5izRq1HsMjgp0flIDHS+s=
490 | dependencies:
491 | iconv-lite "~0.4.13"
492 |
493 | end-of-stream@^1.1.0:
494 | version "1.4.4"
495 | resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0"
496 | integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==
497 | dependencies:
498 | once "^1.4.0"
499 |
500 | es-abstract@^1.17.0, es-abstract@^1.17.0-next.1, es-abstract@^1.17.5:
501 | version "1.17.5"
502 | resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.17.5.tgz#d8c9d1d66c8981fb9200e2251d799eee92774ae9"
503 | integrity sha512-BR9auzDbySxOcfog0tLECW8l28eRGpDpU3Dm3Hp4q/N+VtLTmyj4EUN088XZWQDW/hzj6sYRDXeOFsaAODKvpg==
504 | dependencies:
505 | es-to-primitive "^1.2.1"
506 | function-bind "^1.1.1"
507 | has "^1.0.3"
508 | has-symbols "^1.0.1"
509 | is-callable "^1.1.5"
510 | is-regex "^1.0.5"
511 | object-inspect "^1.7.0"
512 | object-keys "^1.1.1"
513 | object.assign "^4.1.0"
514 | string.prototype.trimleft "^2.1.1"
515 | string.prototype.trimright "^2.1.1"
516 |
517 | es-to-primitive@^1.2.1:
518 | version "1.2.1"
519 | resolved "https://registry.yarnpkg.com/es-to-primitive/-/es-to-primitive-1.2.1.tgz#e55cd4c9cdc188bcefb03b366c736323fc5c898a"
520 | integrity sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==
521 | dependencies:
522 | is-callable "^1.1.4"
523 | is-date-object "^1.0.1"
524 | is-symbol "^1.0.2"
525 |
526 | es5-ext@^0.10.35, es5-ext@^0.10.45, es5-ext@^0.10.46, es5-ext@^0.10.50, es5-ext@~0.10.14, es5-ext@~0.10.2, es5-ext@~0.10.46:
527 | version "0.10.53"
528 | resolved "https://registry.yarnpkg.com/es5-ext/-/es5-ext-0.10.53.tgz#93c5a3acfdbef275220ad72644ad02ee18368de1"
529 | integrity sha512-Xs2Stw6NiNHWypzRTY1MtaG/uJlwCk8kH81920ma8mvN8Xq1gsfhZvpkImLQArw8AHnv8MT2I45J3c0R8slE+Q==
530 | dependencies:
531 | es6-iterator "~2.0.3"
532 | es6-symbol "~3.1.3"
533 | next-tick "~1.0.0"
534 |
535 | es6-iterator@^2.0.3, es6-iterator@~2.0.3:
536 | version "2.0.3"
537 | resolved "https://registry.yarnpkg.com/es6-iterator/-/es6-iterator-2.0.3.tgz#a7de889141a05a94b0854403b2d0a0fbfa98f3b7"
538 | integrity sha1-p96IkUGgWpSwhUQDstCg+/qY87c=
539 | dependencies:
540 | d "1"
541 | es5-ext "^0.10.35"
542 | es6-symbol "^3.1.1"
543 |
544 | es6-symbol@^3.1.1, es6-symbol@~3.1.3:
545 | version "3.1.3"
546 | resolved "https://registry.yarnpkg.com/es6-symbol/-/es6-symbol-3.1.3.tgz#bad5d3c1bcdac28269f4cb331e431c78ac705d18"
547 | integrity sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==
548 | dependencies:
549 | d "^1.0.1"
550 | ext "^1.1.2"
551 |
552 | es6-weak-map@^2.0.2:
553 | version "2.0.3"
554 | resolved "https://registry.yarnpkg.com/es6-weak-map/-/es6-weak-map-2.0.3.tgz#b6da1f16cc2cc0d9be43e6bdbfc5e7dfcdf31d53"
555 | integrity sha512-p5um32HOTO1kP+w7PRnB+5lQ43Z6muuMuIMffvDN8ZB4GcnjLBV6zGStpbASIMk4DCAvEaamhe2zhyCb/QXXsA==
556 | dependencies:
557 | d "1"
558 | es5-ext "^0.10.46"
559 | es6-iterator "^2.0.3"
560 | es6-symbol "^3.1.1"
561 |
562 | escape-string-regexp@^1.0.5:
563 | version "1.0.5"
564 | resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4"
565 | integrity sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=
566 |
567 | esprima@^4.0.0:
568 | version "4.0.1"
569 | resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71"
570 | integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==
571 |
572 | esutils@^2.0.2:
573 | version "2.0.3"
574 | resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64"
575 | integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==
576 |
577 | event-emitter@^0.3.5:
578 | version "0.3.5"
579 | resolved "https://registry.yarnpkg.com/event-emitter/-/event-emitter-0.3.5.tgz#df8c69eef1647923c7157b9ce83840610b02cc39"
580 | integrity sha1-34xp7vFkeSPHFXuc6DhAYQsCzDk=
581 | dependencies:
582 | d "1"
583 | es5-ext "~0.10.14"
584 |
585 | eventemitter3@^4.0.4:
586 | version "4.0.4"
587 | resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.4.tgz#b5463ace635a083d018bdc7c917b4c5f10a85384"
588 | integrity sha512-rlaVLnVxtxvoyLsQQFBx53YmXHDxRIzzTLbdfxqi4yocpSjAxXwkU0cScM5JgSKMqEhrZpnvQ2D9gjylR0AimQ==
589 |
590 | execa@^1.0.0:
591 | version "1.0.0"
592 | resolved "https://registry.yarnpkg.com/execa/-/execa-1.0.0.tgz#c6236a5bb4df6d6f15e88e7f017798216749ddd8"
593 | integrity sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==
594 | dependencies:
595 | cross-spawn "^6.0.0"
596 | get-stream "^4.0.0"
597 | is-stream "^1.1.0"
598 | npm-run-path "^2.0.0"
599 | p-finally "^1.0.0"
600 | signal-exit "^3.0.0"
601 | strip-eof "^1.0.0"
602 |
603 | ext@^1.1.2:
604 | version "1.4.0"
605 | resolved "https://registry.yarnpkg.com/ext/-/ext-1.4.0.tgz#89ae7a07158f79d35517882904324077e4379244"
606 | integrity sha512-Key5NIsUxdqKg3vIsdw9dSuXpPCQ297y6wBjL30edxwPgt2E44WcWBZey/ZvUc6sERLTxKdyCu4gZFmUbk1Q7A==
607 | dependencies:
608 | type "^2.0.0"
609 |
610 | form-data@^3.0.0:
611 | version "3.0.0"
612 | resolved "https://registry.yarnpkg.com/form-data/-/form-data-3.0.0.tgz#31b7e39c85f1355b7139ee0c647cf0de7f83c682"
613 | integrity sha512-CKMFDglpbMi6PyN+brwB9Q/GOw0eAnsrEZDgcsH5Krhz5Od/haKHAX0NmQfha2zPPz0JpWzA7GJHGSnvCRLWsg==
614 | dependencies:
615 | asynckit "^0.4.0"
616 | combined-stream "^1.0.8"
617 | mime-types "^2.1.12"
618 |
619 | fs.realpath@^1.0.0:
620 | version "1.0.0"
621 | resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f"
622 | integrity sha1-FQStJSMVjKpA20onh8sBQRmU6k8=
623 |
624 | function-bind@^1.1.1:
625 | version "1.1.1"
626 | resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d"
627 | integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==
628 |
629 | get-stream@^4.0.0:
630 | version "4.1.0"
631 | resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-4.1.0.tgz#c1b255575f3dc21d59bfc79cd3d2b46b1c3a54b5"
632 | integrity sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==
633 | dependencies:
634 | pump "^3.0.0"
635 |
636 | glob@^7.1.1, glob@^7.1.6:
637 | version "7.1.6"
638 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.6.tgz#141f33b81a7c2492e125594307480c46679278a6"
639 | integrity sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==
640 | dependencies:
641 | fs.realpath "^1.0.0"
642 | inflight "^1.0.4"
643 | inherits "2"
644 | minimatch "^3.0.4"
645 | once "^1.3.0"
646 | path-is-absolute "^1.0.0"
647 |
648 | has-flag@^3.0.0:
649 | version "3.0.0"
650 | resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd"
651 | integrity sha1-tdRU3CGZriJWmfNGfloH87lVuv0=
652 |
653 | has-flag@^4.0.0:
654 | version "4.0.0"
655 | resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b"
656 | integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==
657 |
658 | has-symbols@^1.0.0, has-symbols@^1.0.1:
659 | version "1.0.1"
660 | resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.1.tgz#9f5214758a44196c406d9bd76cebf81ec2dd31e8"
661 | integrity sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==
662 |
663 | has@^1.0.3:
664 | version "1.0.3"
665 | resolved "https://registry.yarnpkg.com/has/-/has-1.0.3.tgz#722d7cbfc1f6aa8241f16dd814e011e1f41e8796"
666 | integrity sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==
667 | dependencies:
668 | function-bind "^1.1.1"
669 |
670 | hash-base@^3.0.0:
671 | version "3.1.0"
672 | resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-3.1.0.tgz#55c381d9e06e1d2997a883b4a3fddfe7f0d3af33"
673 | integrity sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==
674 | dependencies:
675 | inherits "^2.0.4"
676 | readable-stream "^3.6.0"
677 | safe-buffer "^5.2.0"
678 |
679 | hash.js@^1.0.0, hash.js@^1.0.3:
680 | version "1.1.7"
681 | resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42"
682 | integrity sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==
683 | dependencies:
684 | inherits "^2.0.3"
685 | minimalistic-assert "^1.0.1"
686 |
687 | hmac-drbg@^1.0.0:
688 | version "1.0.1"
689 | resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1"
690 | integrity sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=
691 | dependencies:
692 | hash.js "^1.0.3"
693 | minimalistic-assert "^1.0.0"
694 | minimalistic-crypto-utils "^1.0.1"
695 |
696 | iconv-lite@~0.4.13:
697 | version "0.4.24"
698 | resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b"
699 | integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==
700 | dependencies:
701 | safer-buffer ">= 2.1.2 < 3"
702 |
703 | inflight@^1.0.4:
704 | version "1.0.6"
705 | resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9"
706 | integrity sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=
707 | dependencies:
708 | once "^1.3.0"
709 | wrappy "1"
710 |
711 | inherits@2, inherits@^2.0.1, inherits@^2.0.3, inherits@^2.0.4:
712 | version "2.0.4"
713 | resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
714 | integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
715 |
716 | internal-slot@^1.0.2:
717 | version "1.0.2"
718 | resolved "https://registry.yarnpkg.com/internal-slot/-/internal-slot-1.0.2.tgz#9c2e9fb3cd8e5e4256c6f45fe310067fcfa378a3"
719 | integrity sha512-2cQNfwhAfJIkU4KZPkDI+Gj5yNNnbqi40W9Gge6dfnk4TocEVm00B3bdiL+JINrbGJil2TeHvM4rETGzk/f/0g==
720 | dependencies:
721 | es-abstract "^1.17.0-next.1"
722 | has "^1.0.3"
723 | side-channel "^1.0.2"
724 |
725 | ip-regex@^4.1.0:
726 | version "4.1.0"
727 | resolved "https://registry.yarnpkg.com/ip-regex/-/ip-regex-4.1.0.tgz#5ad62f685a14edb421abebc2fff8db94df67b455"
728 | integrity sha512-pKnZpbgCTfH/1NLIlOduP/V+WRXzC2MOz3Qo8xmxk8C5GudJLgK5QyLVXOSWy3ParAH7Eemurl3xjv/WXYFvMA==
729 |
730 | is-callable@^1.1.4, is-callable@^1.1.5:
731 | version "1.1.5"
732 | resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.1.5.tgz#f7e46b596890456db74e7f6e976cb3273d06faab"
733 | integrity sha512-ESKv5sMCJB2jnHTWZ3O5itG+O128Hsus4K4Qh1h2/cgn2vbgnLSVqfV46AeJA9D5EeeLa9w81KUXMtn34zhX+Q==
734 |
735 | is-date-object@^1.0.1:
736 | version "1.0.2"
737 | resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.2.tgz#bda736f2cd8fd06d32844e7743bfa7494c3bfd7e"
738 | integrity sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==
739 |
740 | is-docker@^2.0.0:
741 | version "2.0.0"
742 | resolved "https://registry.yarnpkg.com/is-docker/-/is-docker-2.0.0.tgz#2cb0df0e75e2d064fe1864c37cdeacb7b2dcf25b"
743 | integrity sha512-pJEdRugimx4fBMra5z2/5iRdZ63OhYV0vr0Dwm5+xtW4D1FvRkB8hamMIhnWfyJeDdyr/aa7BDyNbtG38VxgoQ==
744 |
745 | is-promise@^2.1:
746 | version "2.2.2"
747 | resolved "https://registry.yarnpkg.com/is-promise/-/is-promise-2.2.2.tgz#39ab959ccbf9a774cf079f7b40c7a26f763135f1"
748 | integrity sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ==
749 |
750 | is-regex@^1.0.5:
751 | version "1.0.5"
752 | resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.0.5.tgz#39d589a358bf18967f726967120b8fc1aed74eae"
753 | integrity sha512-vlKW17SNq44owv5AQR3Cq0bQPEb8+kF3UKZ2fiZNOWtztYE5i0CzCZxFDwO58qAOWtxdBRVO/V5Qin1wjCqFYQ==
754 | dependencies:
755 | has "^1.0.3"
756 |
757 | is-stream@^1.0.1, is-stream@^1.1.0:
758 | version "1.1.0"
759 | resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-1.1.0.tgz#12d4a3dd4e68e0b79ceb8dbc84173ae80d91ca44"
760 | integrity sha1-EtSj3U5o4Lec6428hBc66A2RykQ=
761 |
762 | is-symbol@^1.0.2:
763 | version "1.0.3"
764 | resolved "https://registry.yarnpkg.com/is-symbol/-/is-symbol-1.0.3.tgz#38e1014b9e6329be0de9d24a414fd7441ec61937"
765 | integrity sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==
766 | dependencies:
767 | has-symbols "^1.0.1"
768 |
769 | is-typedarray@^1.0.0:
770 | version "1.0.0"
771 | resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a"
772 | integrity sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=
773 |
774 | is-wsl@^2.1.1:
775 | version "2.2.0"
776 | resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-2.2.0.tgz#74a4c76e77ca9fd3f932f290c17ea326cd157271"
777 | integrity sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==
778 | dependencies:
779 | is-docker "^2.0.0"
780 |
781 | isexe@^2.0.0:
782 | version "2.0.0"
783 | resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10"
784 | integrity sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=
785 |
786 | isomorphic-fetch@^2.2.1:
787 | version "2.2.1"
788 | resolved "https://registry.yarnpkg.com/isomorphic-fetch/-/isomorphic-fetch-2.2.1.tgz#611ae1acf14f5e81f729507472819fe9733558a9"
789 | integrity sha1-YRrhrPFPXoH3KVB0coGf6XM1WKk=
790 | dependencies:
791 | node-fetch "^1.0.1"
792 | whatwg-fetch ">=0.10.0"
793 |
794 | js-sha3@^0.8.0:
795 | version "0.8.0"
796 | resolved "https://registry.yarnpkg.com/js-sha3/-/js-sha3-0.8.0.tgz#b9b7a5da73afad7dedd0f8c463954cbde6818840"
797 | integrity sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==
798 |
799 | js-tokens@^4.0.0:
800 | version "4.0.0"
801 | resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499"
802 | integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==
803 |
804 | js-yaml@^3.13.1:
805 | version "3.13.1"
806 | resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.13.1.tgz#aff151b30bfdfa8e49e05da22e7415e9dfa37847"
807 | integrity sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==
808 | dependencies:
809 | argparse "^1.0.7"
810 | esprima "^4.0.0"
811 |
812 | lodash@^4.17.19:
813 | version "4.17.19"
814 | resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.19.tgz#e48ddedbe30b3321783c5b4301fbd353bc1e4a4b"
815 | integrity sha512-JNvd8XER9GQX0v2qJgsaN/mzFCNA5BRe/j8JN9d+tWyGLSodKQHKFicdwNYzWwI3wjRnaKPsGj1XkBjx/F96DQ==
816 |
817 | lru-queue@0.1:
818 | version "0.1.0"
819 | resolved "https://registry.yarnpkg.com/lru-queue/-/lru-queue-0.1.0.tgz#2738bd9f0d3cf4f84490c5736c48699ac632cda3"
820 | integrity sha1-Jzi9nw089PhEkMVzbEhpmsYyzaM=
821 | dependencies:
822 | es5-ext "~0.10.2"
823 |
824 | md5.js@^1.3.4:
825 | version "1.3.5"
826 | resolved "https://registry.yarnpkg.com/md5.js/-/md5.js-1.3.5.tgz#b5d07b8e3216e3e27cd728d72f70d1e6a342005f"
827 | integrity sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==
828 | dependencies:
829 | hash-base "^3.0.0"
830 | inherits "^2.0.1"
831 | safe-buffer "^5.1.2"
832 |
833 | memoizee@^0.4.14:
834 | version "0.4.14"
835 | resolved "https://registry.yarnpkg.com/memoizee/-/memoizee-0.4.14.tgz#07a00f204699f9a95c2d9e77218271c7cd610d57"
836 | integrity sha512-/SWFvWegAIYAO4NQMpcX+gcra0yEZu4OntmUdrBaWrJncxOqAziGFlHxc7yjKVK2uu3lpPW27P27wkR82wA8mg==
837 | dependencies:
838 | d "1"
839 | es5-ext "^0.10.45"
840 | es6-weak-map "^2.0.2"
841 | event-emitter "^0.3.5"
842 | is-promise "^2.1"
843 | lru-queue "0.1"
844 | next-tick "1"
845 | timers-ext "^0.1.5"
846 |
847 | mime-db@1.43.0:
848 | version "1.43.0"
849 | resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.43.0.tgz#0a12e0502650e473d735535050e7c8f4eb4fae58"
850 | integrity sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ==
851 |
852 | mime-types@^2.1.12:
853 | version "2.1.26"
854 | resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.26.tgz#9c921fc09b7e149a65dfdc0da4d20997200b0a06"
855 | integrity sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==
856 | dependencies:
857 | mime-db "1.43.0"
858 |
859 | minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1:
860 | version "1.0.1"
861 | resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7"
862 | integrity sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==
863 |
864 | minimalistic-crypto-utils@^1.0.0, minimalistic-crypto-utils@^1.0.1:
865 | version "1.0.1"
866 | resolved "https://registry.yarnpkg.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz#f6c00c1c0b082246e5c4d99dfb8c7c083b2b582a"
867 | integrity sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo=
868 |
869 | minimatch@^3.0.4:
870 | version "3.0.4"
871 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083"
872 | integrity sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==
873 | dependencies:
874 | brace-expansion "^1.1.7"
875 |
876 | minimist@0.0.8:
877 | version "0.0.8"
878 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-0.0.8.tgz#857fcabfc3397d2625b8228262e86aa7a011b05d"
879 | integrity sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=
880 |
881 | mkdirp@^0.5.1:
882 | version "0.5.1"
883 | resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.1.tgz#30057438eac6cf7f8c4767f38648d6697d75c903"
884 | integrity sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=
885 | dependencies:
886 | minimist "0.0.8"
887 |
888 | ms@2.0.0:
889 | version "2.0.0"
890 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8"
891 | integrity sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=
892 |
893 | nan@^2.14.0:
894 | version "2.14.1"
895 | resolved "https://registry.yarnpkg.com/nan/-/nan-2.14.1.tgz#d7be34dfa3105b91494c3147089315eff8874b01"
896 | integrity sha512-isWHgVjnFjh2x2yuJ/tj3JbwoHu3UC2dX5G/88Cm24yB6YopVgxvBObDY7n5xW6ExmFhJpSEQqFPvq9zaXc8Jw==
897 |
898 | next-tick@1:
899 | version "1.1.0"
900 | resolved "https://registry.yarnpkg.com/next-tick/-/next-tick-1.1.0.tgz#1836ee30ad56d67ef281b22bd199f709449b35eb"
901 | integrity sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==
902 |
903 | next-tick@~1.0.0:
904 | version "1.0.0"
905 | resolved "https://registry.yarnpkg.com/next-tick/-/next-tick-1.0.0.tgz#ca86d1fe8828169b0120208e3dc8424b9db8342c"
906 | integrity sha1-yobR/ogoFpsBICCOPchCS524NCw=
907 |
908 | nice-try@^1.0.4:
909 | version "1.0.5"
910 | resolved "https://registry.yarnpkg.com/nice-try/-/nice-try-1.0.5.tgz#a3378a7696ce7d223e88fc9b764bd7ef1089e366"
911 | integrity sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==
912 |
913 | node-fetch@^1.0.1:
914 | version "1.7.3"
915 | resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-1.7.3.tgz#980f6f72d85211a5347c6b2bc18c5b84c3eb47ef"
916 | integrity sha512-NhZ4CsKx7cYm2vSrBAr2PvFOe6sWDf0UYLRqA6svUYg7+/TSfVAu49jYC4BvQ4Sms9SZgdqGBgroqfDhJdTyKQ==
917 | dependencies:
918 | encoding "^0.1.11"
919 | is-stream "^1.0.1"
920 |
921 | node-fetch@^2.6.1:
922 | version "2.6.1"
923 | resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.6.1.tgz#045bd323631f76ed2e2b55573394416b639a0052"
924 | integrity sha512-V4aYg89jEoVRxRb2fJdAg8FHvI7cEyYdVAh94HH0UIK8oJxUfkjlDQN9RbMx+bEjP7+ggMiFRprSti032Oipxw==
925 |
926 | npm-run-path@^2.0.0:
927 | version "2.0.2"
928 | resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-2.0.2.tgz#35a9232dfa35d7067b4cb2ddf2357b1871536c5f"
929 | integrity sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=
930 | dependencies:
931 | path-key "^2.0.0"
932 |
933 | object-inspect@^1.7.0:
934 | version "1.7.0"
935 | resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.7.0.tgz#f4f6bd181ad77f006b5ece60bd0b6f398ff74a67"
936 | integrity sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==
937 |
938 | object-keys@^1.0.11, object-keys@^1.0.12, object-keys@^1.1.1:
939 | version "1.1.1"
940 | resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e"
941 | integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==
942 |
943 | object.assign@^4.1.0:
944 | version "4.1.0"
945 | resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.0.tgz#968bf1100d7956bb3ca086f006f846b3bc4008da"
946 | integrity sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==
947 | dependencies:
948 | define-properties "^1.1.2"
949 | function-bind "^1.1.1"
950 | has-symbols "^1.0.0"
951 | object-keys "^1.0.11"
952 |
953 | once@^1.3.0, once@^1.3.1, once@^1.4.0:
954 | version "1.4.0"
955 | resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1"
956 | integrity sha1-WDsap3WWHUsROsF9nFC6753Xa9E=
957 | dependencies:
958 | wrappy "1"
959 |
960 | p-finally@^1.0.0:
961 | version "1.0.0"
962 | resolved "https://registry.yarnpkg.com/p-finally/-/p-finally-1.0.0.tgz#3fbcfb15b899a44123b34b6dcc18b724336a2cae"
963 | integrity sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=
964 |
965 | path-is-absolute@^1.0.0:
966 | version "1.0.1"
967 | resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f"
968 | integrity sha1-F0uSaHNVNP+8es5r9TpanhtcX18=
969 |
970 | path-key@^2.0.0, path-key@^2.0.1:
971 | version "2.0.1"
972 | resolved "https://registry.yarnpkg.com/path-key/-/path-key-2.0.1.tgz#411cadb574c5a140d3a4b1910d40d80cc9f40b40"
973 | integrity sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=
974 |
975 | path-parse@^1.0.6:
976 | version "1.0.6"
977 | resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.6.tgz#d62dbb5679405d72c4737ec58600e9ddcf06d24c"
978 | integrity sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==
979 |
980 | pbkdf2@^3.0.9, pbkdf2@^3.1.1:
981 | version "3.1.1"
982 | resolved "https://registry.yarnpkg.com/pbkdf2/-/pbkdf2-3.1.1.tgz#cb8724b0fada984596856d1a6ebafd3584654b94"
983 | integrity sha512-4Ejy1OPxi9f2tt1rRV7Go7zmfDQ+ZectEQz3VGUQhgq62HtIRPDyG/JtnwIxs6x3uNMwo2V7q1fMvKjb+Tnpqg==
984 | dependencies:
985 | create-hash "^1.1.2"
986 | create-hmac "^1.1.4"
987 | ripemd160 "^2.0.1"
988 | safe-buffer "^5.0.1"
989 | sha.js "^2.4.8"
990 |
991 | pump@^3.0.0:
992 | version "3.0.0"
993 | resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64"
994 | integrity sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==
995 | dependencies:
996 | end-of-stream "^1.1.0"
997 | once "^1.3.1"
998 |
999 | randombytes@^2.0.1:
1000 | version "2.1.0"
1001 | resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a"
1002 | integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==
1003 | dependencies:
1004 | safe-buffer "^5.1.0"
1005 |
1006 | readable-stream@^3.6.0:
1007 | version "3.6.0"
1008 | resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.0.tgz#337bbda3adc0706bd3e024426a286d4b4b2c9198"
1009 | integrity sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==
1010 | dependencies:
1011 | inherits "^2.0.3"
1012 | string_decoder "^1.1.1"
1013 | util-deprecate "^1.0.1"
1014 |
1015 | regenerator-runtime@^0.13.4:
1016 | version "0.13.5"
1017 | resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.13.5.tgz#d878a1d094b4306d10b9096484b33ebd55e26697"
1018 | integrity sha512-ZS5w8CpKFinUzOwW3c83oPeVXoNsrLsaCoLtJvAClH135j/R77RuymhiSErhm2lKcwSCIpmvIWSbDkIfAqKQlA==
1019 |
1020 | regexp.prototype.flags@^1.3.0:
1021 | version "1.3.0"
1022 | resolved "https://registry.yarnpkg.com/regexp.prototype.flags/-/regexp.prototype.flags-1.3.0.tgz#7aba89b3c13a64509dabcf3ca8d9fbb9bdf5cb75"
1023 | integrity sha512-2+Q0C5g951OlYlJz6yu5/M33IcsESLlLfsyIaLJaG4FA2r4yP8MvVMJUUP/fVBkSpbbbZlS5gynbEWLipiiXiQ==
1024 | dependencies:
1025 | define-properties "^1.1.3"
1026 | es-abstract "^1.17.0-next.1"
1027 |
1028 | resolve@^1.3.2:
1029 | version "1.15.1"
1030 | resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.15.1.tgz#27bdcdeffeaf2d6244b95bb0f9f4b4653451f3e8"
1031 | integrity sha512-84oo6ZTtoTUpjgNEr5SJyzQhzL72gaRodsSfyxC/AXRvwu0Yse9H8eF9IpGo7b8YetZhlI6v7ZQ6bKBFV/6S7w==
1032 | dependencies:
1033 | path-parse "^1.0.6"
1034 |
1035 | ripemd160@^2.0.0, ripemd160@^2.0.1:
1036 | version "2.0.2"
1037 | resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.2.tgz#a1c1a6f624751577ba5d07914cbc92850585890c"
1038 | integrity sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==
1039 | dependencies:
1040 | hash-base "^3.0.0"
1041 | inherits "^2.0.1"
1042 |
1043 | rxjs@^6.5.5:
1044 | version "6.5.5"
1045 | resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-6.5.5.tgz#c5c884e3094c8cfee31bf27eb87e54ccfc87f9ec"
1046 | integrity sha512-WfQI+1gohdf0Dai/Bbmk5L5ItH5tYqm3ki2c5GdWhKjalzjg93N3avFjVStyZZz+A2Em+ZxKH5bNghw9UeylGQ==
1047 | dependencies:
1048 | tslib "^1.9.0"
1049 |
1050 | rxjs@^6.6.2:
1051 | version "6.6.2"
1052 | resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-6.6.2.tgz#8096a7ac03f2cc4fe5860ef6e572810d9e01c0d2"
1053 | integrity sha512-BHdBMVoWC2sL26w//BCu3YzKT4s2jip/WhwsGEDmeKYBhKDZeYezVUnHatYB7L85v5xs0BAQmg6BEYJEKxBabg==
1054 | dependencies:
1055 | tslib "^1.9.0"
1056 |
1057 | safe-buffer@^5.0.1, safe-buffer@^5.1.0, safe-buffer@^5.1.2, safe-buffer@^5.2.0, safe-buffer@~5.2.0:
1058 | version "5.2.1"
1059 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6"
1060 | integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==
1061 |
1062 | "safer-buffer@>= 2.1.2 < 3":
1063 | version "2.1.2"
1064 | resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a"
1065 | integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==
1066 |
1067 | scryptsy@^2.1.0:
1068 | version "2.1.0"
1069 | resolved "https://registry.yarnpkg.com/scryptsy/-/scryptsy-2.1.0.tgz#8d1e8d0c025b58fdd25b6fa9a0dc905ee8faa790"
1070 | integrity sha512-1CdSqHQowJBnMAFyPEBRfqag/YP9OF394FV+4YREIJX4ljD7OxvQRDayyoyyCk+senRjSkP6VnUNQmVQqB6g7w==
1071 |
1072 | semver@^5.3.0, semver@^5.5.0:
1073 | version "5.7.1"
1074 | resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7"
1075 | integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==
1076 |
1077 | sha.js@^2.4.0, sha.js@^2.4.8:
1078 | version "2.4.11"
1079 | resolved "https://registry.yarnpkg.com/sha.js/-/sha.js-2.4.11.tgz#37a5cf0b81ecbc6943de109ba2960d1b26584ae7"
1080 | integrity sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==
1081 | dependencies:
1082 | inherits "^2.0.1"
1083 | safe-buffer "^5.0.1"
1084 |
1085 | shebang-command@^1.2.0:
1086 | version "1.2.0"
1087 | resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-1.2.0.tgz#44aac65b695b03398968c39f363fee5deafdf1ea"
1088 | integrity sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=
1089 | dependencies:
1090 | shebang-regex "^1.0.0"
1091 |
1092 | shebang-regex@^1.0.0:
1093 | version "1.0.0"
1094 | resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-1.0.0.tgz#da42f49740c0b42db2ca9728571cb190c98efea3"
1095 | integrity sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=
1096 |
1097 | side-channel@^1.0.2:
1098 | version "1.0.2"
1099 | resolved "https://registry.yarnpkg.com/side-channel/-/side-channel-1.0.2.tgz#df5d1abadb4e4bf4af1cd8852bf132d2f7876947"
1100 | integrity sha512-7rL9YlPHg7Ancea1S96Pa8/QWb4BtXL/TZvS6B8XFetGBeuhAsfmUspK6DokBeZ64+Kj9TCNRD/30pVz1BvQNA==
1101 | dependencies:
1102 | es-abstract "^1.17.0-next.1"
1103 | object-inspect "^1.7.0"
1104 |
1105 | signal-exit@^3.0.0:
1106 | version "3.0.3"
1107 | resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.3.tgz#a1410c2edd8f077b08b4e253c8eacfcaf057461c"
1108 | integrity sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==
1109 |
1110 | sprintf-js@~1.0.2:
1111 | version "1.0.3"
1112 | resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c"
1113 | integrity sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=
1114 |
1115 | string.prototype.matchall@^4.0.2:
1116 | version "4.0.2"
1117 | resolved "https://registry.yarnpkg.com/string.prototype.matchall/-/string.prototype.matchall-4.0.2.tgz#48bb510326fb9fdeb6a33ceaa81a6ea04ef7648e"
1118 | integrity sha512-N/jp6O5fMf9os0JU3E72Qhf590RSRZU/ungsL/qJUYVTNv7hTG0P/dbPjxINVN9jpscu3nzYwKESU3P3RY5tOg==
1119 | dependencies:
1120 | define-properties "^1.1.3"
1121 | es-abstract "^1.17.0"
1122 | has-symbols "^1.0.1"
1123 | internal-slot "^1.0.2"
1124 | regexp.prototype.flags "^1.3.0"
1125 | side-channel "^1.0.2"
1126 |
1127 | string.prototype.trimend@^1.0.0:
1128 | version "1.0.1"
1129 | resolved "https://registry.yarnpkg.com/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz#85812a6b847ac002270f5808146064c995fb6913"
1130 | integrity sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==
1131 | dependencies:
1132 | define-properties "^1.1.3"
1133 | es-abstract "^1.17.5"
1134 |
1135 | string.prototype.trimleft@^2.1.1:
1136 | version "2.1.2"
1137 | resolved "https://registry.yarnpkg.com/string.prototype.trimleft/-/string.prototype.trimleft-2.1.2.tgz#4408aa2e5d6ddd0c9a80739b087fbc067c03b3cc"
1138 | integrity sha512-gCA0tza1JBvqr3bfAIFJGqfdRTyPae82+KTnm3coDXkZN9wnuW3HjGgN386D7hfv5CHQYCI022/rJPVlqXyHSw==
1139 | dependencies:
1140 | define-properties "^1.1.3"
1141 | es-abstract "^1.17.5"
1142 | string.prototype.trimstart "^1.0.0"
1143 |
1144 | string.prototype.trimright@^2.1.1:
1145 | version "2.1.2"
1146 | resolved "https://registry.yarnpkg.com/string.prototype.trimright/-/string.prototype.trimright-2.1.2.tgz#c76f1cef30f21bbad8afeb8db1511496cfb0f2a3"
1147 | integrity sha512-ZNRQ7sY3KroTaYjRS6EbNiiHrOkjihL9aQE/8gfQ4DtAC/aEBRHFJa44OmoWxGGqXuJlfKkZW4WcXErGr+9ZFg==
1148 | dependencies:
1149 | define-properties "^1.1.3"
1150 | es-abstract "^1.17.5"
1151 | string.prototype.trimend "^1.0.0"
1152 |
1153 | string.prototype.trimstart@^1.0.0:
1154 | version "1.0.1"
1155 | resolved "https://registry.yarnpkg.com/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz#14af6d9f34b053f7cfc89b72f8f2ee14b9039a54"
1156 | integrity sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==
1157 | dependencies:
1158 | define-properties "^1.1.3"
1159 | es-abstract "^1.17.5"
1160 |
1161 | string_decoder@^1.1.1:
1162 | version "1.3.0"
1163 | resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e"
1164 | integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==
1165 | dependencies:
1166 | safe-buffer "~5.2.0"
1167 |
1168 | strip-eof@^1.0.0:
1169 | version "1.0.0"
1170 | resolved "https://registry.yarnpkg.com/strip-eof/-/strip-eof-1.0.0.tgz#bb43ff5598a6eb05d89b59fcd129c983313606bf"
1171 | integrity sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=
1172 |
1173 | supports-color@^5.3.0:
1174 | version "5.5.0"
1175 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz#e2e69a44ac8772f78a1ec0b35b689df6530efc8f"
1176 | integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==
1177 | dependencies:
1178 | has-flag "^3.0.0"
1179 |
1180 | supports-color@^7.1.0:
1181 | version "7.1.0"
1182 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.1.0.tgz#68e32591df73e25ad1c4b49108a2ec507962bfd1"
1183 | integrity sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==
1184 | dependencies:
1185 | has-flag "^4.0.0"
1186 |
1187 | timers-ext@^0.1.5:
1188 | version "0.1.7"
1189 | resolved "https://registry.yarnpkg.com/timers-ext/-/timers-ext-0.1.7.tgz#6f57ad8578e07a3fb9f91d9387d65647555e25c6"
1190 | integrity sha512-b85NUNzTSdodShTIbky6ZF02e8STtVVfD+fu4aXXShEELpozH+bCpJLYMPZbsABN2wDH7fJpqIoXxJpzbf0NqQ==
1191 | dependencies:
1192 | es5-ext "~0.10.46"
1193 | next-tick "1"
1194 |
1195 | tslib@^1.8.0, tslib@^1.8.1, tslib@^1.9.0:
1196 | version "1.11.1"
1197 | resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.11.1.tgz#eb15d128827fbee2841549e171f45ed338ac7e35"
1198 | integrity sha512-aZW88SY8kQbU7gpV19lN24LtXh/yD4ZZg6qieAJDDg+YBsJcSmLGK9QpnUjAKVG/xefmvJGd1WUmfpT/g6AJGA==
1199 |
1200 | tslint@^5.12.1:
1201 | version "5.20.1"
1202 | resolved "https://registry.yarnpkg.com/tslint/-/tslint-5.20.1.tgz#e401e8aeda0152bc44dd07e614034f3f80c67b7d"
1203 | integrity sha512-EcMxhzCFt8k+/UP5r8waCf/lzmeSyVlqxqMEDQE7rWYiQky8KpIBz1JAoYXfROHrPZ1XXd43q8yQnULOLiBRQg==
1204 | dependencies:
1205 | "@babel/code-frame" "^7.0.0"
1206 | builtin-modules "^1.1.1"
1207 | chalk "^2.3.0"
1208 | commander "^2.12.1"
1209 | diff "^4.0.1"
1210 | glob "^7.1.1"
1211 | js-yaml "^3.13.1"
1212 | minimatch "^3.0.4"
1213 | mkdirp "^0.5.1"
1214 | resolve "^1.3.2"
1215 | semver "^5.3.0"
1216 | tslib "^1.8.0"
1217 | tsutils "^2.29.0"
1218 |
1219 | tsutils@^2.29.0:
1220 | version "2.29.0"
1221 | resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-2.29.0.tgz#32b488501467acbedd4b85498673a0812aca0b99"
1222 | integrity sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==
1223 | dependencies:
1224 | tslib "^1.8.1"
1225 |
1226 | tweetnacl@^1.0.3:
1227 | version "1.0.3"
1228 | resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-1.0.3.tgz#ac0af71680458d8a6378d0d0d050ab1407d35596"
1229 | integrity sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw==
1230 |
1231 | type@^1.0.1:
1232 | version "1.2.0"
1233 | resolved "https://registry.yarnpkg.com/type/-/type-1.2.0.tgz#848dd7698dafa3e54a6c479e759c4bc3f18847a0"
1234 | integrity sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==
1235 |
1236 | type@^2.0.0:
1237 | version "2.0.0"
1238 | resolved "https://registry.yarnpkg.com/type/-/type-2.0.0.tgz#5f16ff6ef2eb44f260494dae271033b29c09a9c3"
1239 | integrity sha512-KBt58xCHry4Cejnc2ISQAF7QY+ORngsWfxezO68+12hKV6lQY8P/psIkcbjeHWn7MqcgciWJyCCevFMJdIXpow==
1240 |
1241 | typedarray-to-buffer@^3.1.5:
1242 | version "3.1.5"
1243 | resolved "https://registry.yarnpkg.com/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz#a97ee7a9ff42691b9f783ff1bc5112fe3fca9080"
1244 | integrity sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==
1245 | dependencies:
1246 | is-typedarray "^1.0.0"
1247 |
1248 | typescript@^3.5.1:
1249 | version "3.8.3"
1250 | resolved "https://registry.yarnpkg.com/typescript/-/typescript-3.8.3.tgz#409eb8544ea0335711205869ec458ab109ee1061"
1251 | integrity sha512-MYlEfn5VrLNsgudQTVJeNaQFUAI7DkhnOjdpAp4T+ku1TfQClewlbSuTVHiA+8skNBgaf02TL/kLOvig4y3G8w==
1252 |
1253 | util-deprecate@^1.0.1:
1254 | version "1.0.2"
1255 | resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf"
1256 | integrity sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=
1257 |
1258 | websocket@^1.0.31:
1259 | version "1.0.31"
1260 | resolved "https://registry.yarnpkg.com/websocket/-/websocket-1.0.31.tgz#e5d0f16c3340ed87670e489ecae6144c79358730"
1261 | integrity sha512-VAouplvGKPiKFDTeCCO65vYHsyay8DqoBSlzIO3fayrfOgU94lQN5a1uWVnFrMLceTJw/+fQXR5PGbUVRaHshQ==
1262 | dependencies:
1263 | debug "^2.2.0"
1264 | es5-ext "^0.10.50"
1265 | nan "^2.14.0"
1266 | typedarray-to-buffer "^3.1.5"
1267 | yaeti "^0.0.6"
1268 |
1269 | whatwg-fetch@>=0.10.0:
1270 | version "3.1.0"
1271 | resolved "https://registry.yarnpkg.com/whatwg-fetch/-/whatwg-fetch-3.1.0.tgz#49d630cdfa308dba7f2819d49d09364f540dbcc6"
1272 | integrity sha512-pgmbsVWKpH9GxLXZmtdowDIqtb/rvPyjjQv3z9wLcmgWKFHilKnZD3ldgrOlwJoPGOUluQsRPWd52yVkPfmI1A==
1273 |
1274 | which@^1.2.9:
1275 | version "1.3.1"
1276 | resolved "https://registry.yarnpkg.com/which/-/which-1.3.1.tgz#a45043d54f5805316da8d62f9f50918d3da70b0a"
1277 | integrity sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==
1278 | dependencies:
1279 | isexe "^2.0.0"
1280 |
1281 | wrappy@1:
1282 | version "1.0.2"
1283 | resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f"
1284 | integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=
1285 |
1286 | xxhashjs@^0.2.2:
1287 | version "0.2.2"
1288 | resolved "https://registry.yarnpkg.com/xxhashjs/-/xxhashjs-0.2.2.tgz#8a6251567621a1c46a5ae204da0249c7f8caa9d8"
1289 | integrity sha512-AkTuIuVTET12tpsVIQo+ZU6f/qDmKuRUcjaqR+OIvm+aCBsZ95i7UVY5WJ9TMsSaZ0DA2WxoZ4acu0sPH+OKAw==
1290 | dependencies:
1291 | cuint "^0.2.2"
1292 |
1293 | yaeti@^0.0.6:
1294 | version "0.0.6"
1295 | resolved "https://registry.yarnpkg.com/yaeti/-/yaeti-0.0.6.tgz#f26f484d72684cf42bedfb76970aa1608fbf9577"
1296 | integrity sha1-8m9ITXJoTPQr7ft2lwqhYI+/lXc=
1297 |
--------------------------------------------------------------------------------