From f25a1ec7561c0127a54fbfbeb671ef8581973623 Mon Sep 17 00:00:00 2001 From: "Michael R. Kenworthy" Date: Tue, 4 Dec 2018 01:10:26 -0800 Subject: [PATCH 1/5] Added authorization --- appsscript.json | 12 +++- src/TrelloService.ts | 140 ++++++++++++++++++++++++++++++++++++++++++- 2 files changed, 148 insertions(+), 4 deletions(-) diff --git a/appsscript.json b/appsscript.json index 4002023..bde7770 100644 --- a/appsscript.json +++ b/appsscript.json @@ -2,6 +2,11 @@ { "timeZone": "America/Los_Angeles", "dependencies": { + "libraries": [{ + "userSymbol": "OAuth1", + "libraryId": "1CXDCY5sqT9ph64fFwSzVtXnbjpSfWdRymafDrtIZ7Z_hwysTY7IIhi7s", + "version": "16" + }] }, "exceptionLogging": "STACKDRIVER", @@ -14,6 +19,7 @@ ], "urlFetchWhitelist": [ "https://trello.com/", + "https://api.trello.com/", "https://script.google.com/a/kenworthy.io" ], "gmail": { @@ -24,7 +30,9 @@ "onTriggerFunction": "buildQuickLinkUI" }], "openLinkUrlPrefixes": [ - "https://upload.wikimedia.org/" - ] + "https://upload.wikimedia.org/", + "https://trello.com/" + ], + "authorizationCheckFunction": "testAuthFunc" } } \ No newline at end of file diff --git a/src/TrelloService.ts b/src/TrelloService.ts index 19a6f9f..246bcbd 100644 --- a/src/TrelloService.ts +++ b/src/TrelloService.ts @@ -1,6 +1,6 @@ namespace TrelloService { export function getUrl(request:TrelloRequest):string { - return `${Config.trello.baseUrl}${request.apiPath}${request.params ? "?"+request.params+"&" : "?"}key=${Config.trello.key}&token=${Config.trello.token}` + return `${Config.trello.baseUrl}${request.apiPath}${request.params ? "?"+request.params+"&" : ""}` } export function send(request:TrelloRequest) { @@ -13,7 +13,8 @@ namespace TrelloService { } if(request.method) options['method'] = request.method - return JSON.parse(UrlFetchApp.fetch(getUrl(request), options).getContentText()) + //return JSON.parse(accessProtectedResource(getUrl(request), options)) + return JSON.parse(getOAuthService().fetch(getUrl(request), options).getContentText()) } export interface TrelloRequest { @@ -22,4 +23,139 @@ namespace TrelloService { params:string, payload:object } +} + +function testAuthFunc() { + accessProtectedResource(`https://api.trello.com/1/members/${Config.users.mike}/boards`, {}) +} + +/** + * Attempts to access a non-Google API using a constructed service + * object. + * + * If your add-on needs access to non-Google APIs that require OAuth, + * you need to implement this method. You can use the OAuth1 and + * OAuth2 Apps Script libraries to help implement it. + * + * @param {String} url The URL to access. + * @param {String} method_opt The HTTP method. Defaults to GET. + * @param {Object} headers_opt The HTTP headers. Defaults to an empty + * object. The Authorization field is added + * to the headers in this method. + * @returns {HttpResponse} the result from the UrlFetchApp.fetch() call. + */ +function accessProtectedResource(url, options) { + var service = getOAuthService(); + var maybeAuthorized = service.hasAccess(); + console.log("Checking access") + if (maybeAuthorized) { + // A token is present, but it may be expired or invalid. Make a + // request and check the response code to be sure. + + // Make the UrlFetch request and return the result. + options['muteHttpExceptions'] = true + var resp = getOAuthService().fetch(url, options); + + var code = resp.getResponseCode(); + if (code >= 200 && code < 300) { + console.log("Request was a success") + return resp.getContentText("utf-8"); // Success + } else if (code == 401 || code == 403) { + console.log("Not authorized") + // Not fully authorized for this action. + maybeAuthorized = false; + } else { + // Handle other response codes by logging them and throwing an + // exception. + console.error("Backend server error (%s): %s", code.toString(), + resp.getContentText("utf-8")); + throw ("Backend server error: " + code); + } + } + + if (!maybeAuthorized) { + console.log(`Start authorization flow ${service.authorize()}`) + // Invoke the authorization flow using the default authorization + // prompt card. + CardService.newAuthorizationException() + .setAuthorizationUrl(service.authorize()) + .setResourceDisplayName("Trello needs authorization") + .throwException(); + } +} + +/** + * Create a new OAuth service to facilitate accessing an API. + * This example assumes there is a single service that the add-on needs to + * access. Its name is used when persisting the authorized token, so ensure + * it is unique within the scope of the property store. You must set the + * client secret and client ID, which are obtained when registering your + * add-on with the API. + * + * See the Apps Script OAuth2 Library documentation for more + * information: + * https://github.com/googlesamples/apps-script-oauth2#1-create-the-oauth2-service + * + * @returns A configured OAuth2 service object. + */ +function getOAuthService() { + return OAuth1.createService('trello') + // Set the endpoint URLs. + .setAccessTokenUrl('https://trello.com/1/OAuthGetAccessToken') + .setRequestTokenUrl('https://trello.com/1/OAuthGetRequestToken') + .setAuthorizationUrl('https://trello.com/1/OAuthAuthorizeToken') + // Set the consumer key and secret. + .setConsumerKey(Config.trello.key) + .setConsumerSecret(Config.trello.secret) + // Set the name of the callback function in the script referenced + // above that should be invoked to complete the OAuth flow. + .setCallbackFunction('authCallback') + // Set the property store where authorized tokens should be persisted. + .setPropertyStore(PropertiesService.getUserProperties()); + +} + +/** + * Boilerplate code to determine if a request is authorized and returns + * a corresponding HTML message. When the user completes the OAuth2 flow + * on the service provider's website, this function is invoked from the + * service. In order for authorization to succeed you must make sure that + * the service knows how to call this function by setting the correct + * redirect URL. + * + * The redirect URL to enter is: + * https://script.google.com/macros/d//usercallback + * + * See the Apps Script OAuth2 Library documentation for more + * information: + * https://github.com/googlesamples/apps-script-oauth2#1-create-the-oauth2-service + * + * @param {Object} callbackRequest The request data received from the + * callback function. Pass it to the service's + * handleCallback() method to complete the + * authorization process. + * @returns {HtmlOutput} a success or denied HTML message to display to + * the user. Also sets a timer to close the window + * automatically. + */ +function authCallback(callbackRequest) { + console.log(`We were called back! #{callbackRequest}`) + var authorized = getOAuthService().handleCallback(callbackRequest); + if (authorized) { + console.log("YAY!") + return HtmlService.createHtmlOutput( + 'Success! '); + } else { + console.log("Denied :(") + return HtmlService.createHtmlOutput('Denied'); + } +} + +/** + * Unauthorizes the non-Google service. This is useful for OAuth + * development/testing. Run this method (Run > resetOAuth in the script + * editor) to reset OAuth to re-prompt the user for OAuth. + */ +function resetOAuth() { + getOAuthService().reset(); } \ No newline at end of file From cb6e8a3b05cbdc17f691a029f7fcec9cd5af4f35 Mon Sep 17 00:00:00 2001 From: "Michael R. Kenworthy" Date: Tue, 4 Dec 2018 12:57:47 -0800 Subject: [PATCH 2/5] Code Cleanup --- .claspignore | 4 - appsscript.json | 9 +- example.config.ts | 7 - package-lock.json | 354 +++++++++++++++++++++++++ package.json | 13 + readme.md | 1 + src/Api/Trello/Board.ts | 42 +++ src/Api/Trello/Card.ts | 69 +++++ src/Api/Trello/List.ts | 43 +++ src/Api/Trello/Member.ts | 44 +++ src/Api/Trello/Token.ts | 38 +++ src/Api/Trello/Trello.ts | 91 +++++++ src/Board.ts | 31 --- src/Card.ts | 54 ---- src/List.ts | 33 --- src/Member.ts | 32 --- src/Message.ts | 33 +++ src/QuickLinksAddOn.ts | 171 ------------ src/TrelloService.ts | 161 ----------- src/Ui/MainMenuUi.ts | 21 ++ src/Ui/StateManager.ts | 36 +++ src/Ui/Trello/BoardSelectionWidget.ts | 33 +++ src/Ui/Trello/CardDescriptionWidget.ts | 16 ++ src/Ui/Trello/CardDueDateWidget.ts | 17 ++ src/Ui/Trello/CardNameWidget.ts | 16 ++ src/Ui/Trello/CardSubmitWidget.ts | 32 +++ src/Ui/Trello/ListSelectionWidget.ts | 48 ++++ src/Ui/Trello/TrelloNewCardUi.ts | 55 ++++ src/googleAppsScriptEntryPoints.ts | 25 ++ tslint.json | 31 +++ 30 files changed, 1064 insertions(+), 496 deletions(-) delete mode 100644 .claspignore delete mode 100644 example.config.ts create mode 100644 package-lock.json create mode 100644 package.json create mode 100644 readme.md create mode 100644 src/Api/Trello/Board.ts create mode 100644 src/Api/Trello/Card.ts create mode 100644 src/Api/Trello/List.ts create mode 100644 src/Api/Trello/Member.ts create mode 100644 src/Api/Trello/Token.ts create mode 100644 src/Api/Trello/Trello.ts delete mode 100644 src/Board.ts delete mode 100644 src/Card.ts delete mode 100644 src/List.ts delete mode 100644 src/Member.ts create mode 100644 src/Message.ts delete mode 100644 src/QuickLinksAddOn.ts delete mode 100644 src/TrelloService.ts create mode 100644 src/Ui/MainMenuUi.ts create mode 100644 src/Ui/StateManager.ts create mode 100644 src/Ui/Trello/BoardSelectionWidget.ts create mode 100644 src/Ui/Trello/CardDescriptionWidget.ts create mode 100644 src/Ui/Trello/CardDueDateWidget.ts create mode 100644 src/Ui/Trello/CardNameWidget.ts create mode 100644 src/Ui/Trello/CardSubmitWidget.ts create mode 100644 src/Ui/Trello/ListSelectionWidget.ts create mode 100644 src/Ui/Trello/TrelloNewCardUi.ts create mode 100644 src/googleAppsScriptEntryPoints.ts create mode 100644 tslint.json diff --git a/.claspignore b/.claspignore deleted file mode 100644 index 607c3f9..0000000 --- a/.claspignore +++ /dev/null @@ -1,4 +0,0 @@ -**/** -!**/src/** -!appsscript.json -!config.ts diff --git a/appsscript.json b/appsscript.json index bde7770..31be5dd 100644 --- a/appsscript.json +++ b/appsscript.json @@ -1,11 +1,10 @@ - { "timeZone": "America/Los_Angeles", "dependencies": { "libraries": [{ "userSymbol": "OAuth1", "libraryId": "1CXDCY5sqT9ph64fFwSzVtXnbjpSfWdRymafDrtIZ7Z_hwysTY7IIhi7s", - "version": "16" + "version": "15" }] }, "exceptionLogging": "STACKDRIVER", @@ -31,8 +30,12 @@ }], "openLinkUrlPrefixes": [ "https://upload.wikimedia.org/", + "https://api.trello.com/", "https://trello.com/" ], - "authorizationCheckFunction": "testAuthFunc" + "universalActions": [{ + "text": "Logout", + "runFunction": "logout" + }] } } \ No newline at end of file diff --git a/example.config.ts b/example.config.ts deleted file mode 100644 index 03ff86f..0000000 --- a/example.config.ts +++ /dev/null @@ -1,7 +0,0 @@ -const Config = { - trello: { - "baseUrl": "https://trello.com/1", - "key": "", - "token": "" - } -} diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..548ff3e --- /dev/null +++ b/package-lock.json @@ -0,0 +1,354 @@ +{ + "name": "betterTrello", + "version": "1.0.0", + "lockfileVersion": 1, + "requires": true, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + }, + "argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "requires": { + "sprintf-js": "~1.0.2" + } + }, + "babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", + "dev": true, + "requires": { + "chalk": "^1.1.3", + "esutils": "^2.0.2", + "js-tokens": "^3.0.2" + }, + "dependencies": { + "chalk": { + "version": "1.1.3", + "resolved": "http://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + } + } + } + }, + "balanced-match": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "builtin-modules": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", + "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", + "dev": true + }, + "chalk": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.1.tgz", + "integrity": "sha512-ObN6h1v2fTJSmUXoS3nMQ92LbDK9be4TV+6G+omQlGJFdcUX5heKi1LZ1YnRMIgwTLEj3E24bT6tYni50rlCfQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, + "dependencies": { + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "clasp": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/clasp/-/clasp-0.1.0.tgz", + "integrity": "sha1-zHKsAZEunxMgdckKy8MTSA/C9HA=", + "dev": true, + "requires": { + "pegjs": "0.7.0" + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "requires": { + "color-name": "1.1.3" + } + }, + "color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", + "dev": true + }, + "commander": { + "version": "2.19.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.19.0.tgz", + "integrity": "sha512-6tvAOO+D6OENvRAh524Dh9jcfKTYDQAqvqezbCW82xj5X0pSrcpxtvRKHLG0yBY6SD7PSDrJaj+0AiOcKVd1Xg==", + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", + "dev": true + }, + "diff": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", + "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", + "dev": true + }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", + "dev": true + }, + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true + }, + "esutils": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", + "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=", + "dev": true + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", + "dev": true + }, + "glob": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", + "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", + "dev": true + }, + "js-tokens": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", + "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=", + "dev": true + }, + "js-yaml": { + "version": "3.12.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.12.0.tgz", + "integrity": "sha512-PIt2cnwmPfL4hKNwqeiuz4bKfnzHTBv6HyVgjahA6mPLwPDzjDWrplJBMjHUFxku/N3FlmrbyPclad+I+4mJ3A==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "http://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", + "dev": true + }, + "path-parse": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", + "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", + "dev": true + }, + "pegjs": { + "version": "0.7.0", + "resolved": "http://registry.npmjs.org/pegjs/-/pegjs-0.7.0.tgz", + "integrity": "sha1-qqH4JPnnGX7ETiKlevUi9wDdaJ4=", + "dev": true + }, + "resolve": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.8.1.tgz", + "integrity": "sha512-AicPrAC7Qu1JxPCZ9ZgCZlY35QgFnNqc+0LtbRNxnVw4TXvjQ72wnuL9JQcEBgXkI9JM8MsT9kaQoHcpCRJOYA==", + "dev": true, + "requires": { + "path-parse": "^1.0.5" + } + }, + "semver": { + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.6.0.tgz", + "integrity": "sha512-RS9R6R35NYgQn++fkDWaOmqGoj4Ek9gGs+DPxNUZKuwE183xjJroKvyo1IzVFeXvUrvmALy6FWD5xrdJT25gMg==", + "dev": true + }, + "sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", + "dev": true + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "http://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + }, + "tslib": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.9.3.tgz", + "integrity": "sha512-4krF8scpejhaOgqzBEcGM7yDIEfi0/8+8zDRZhNZZ2kjmHJ4hv3zCbQWxoJGz1iw5U0Jl0nma13xzHXcncMavQ==", + "dev": true + }, + "tslint": { + "version": "5.11.0", + "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.11.0.tgz", + "integrity": "sha1-mPMMAurjzecAYgHkwzywi0hYHu0=", + "dev": true, + "requires": { + "babel-code-frame": "^6.22.0", + "builtin-modules": "^1.1.1", + "chalk": "^2.3.0", + "commander": "^2.12.1", + "diff": "^3.2.0", + "glob": "^7.1.1", + "js-yaml": "^3.7.0", + "minimatch": "^3.0.4", + "resolve": "^1.3.2", + "semver": "^5.3.0", + "tslib": "^1.8.0", + "tsutils": "^2.27.2" + } + }, + "tsutils": { + "version": "2.29.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", + "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", + "dev": true, + "requires": { + "tslib": "^1.8.1" + } + }, + "typescript": { + "version": "3.3.0-dev.20181205", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.3.0-dev.20181205.tgz", + "integrity": "sha512-XYoXPbSZrlKZOA5ykKjzwvGjAtwEhII/hBFXoGzbxfIWaT5uPcg3dISdE8qZ7jb6Xg1vjV2ghtTevToE2+IvCA==", + "dev": true + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", + "dev": true + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..bf8dae5 --- /dev/null +++ b/package.json @@ -0,0 +1,13 @@ +{ + "name": "betterTrello", + "version": "1.0.0", + "description": "Trello's thing didn't have email link attachments so I did this...", + "repository": "https://github.com/mrk3767/betterTrello", + "license": "MIT", + "dependencies": {}, + "devDependencies": { + "clasp": "^0.1.0", + "tslint": "^5.11.0", + "typescript": "^3.3.0-dev.20181205" + } +} diff --git a/readme.md b/readme.md new file mode 100644 index 0000000..1a06df7 --- /dev/null +++ b/readme.md @@ -0,0 +1 @@ +### Setup \ No newline at end of file diff --git a/src/Api/Trello/Board.ts b/src/Api/Trello/Board.ts new file mode 100644 index 0000000..14def95 --- /dev/null +++ b/src/Api/Trello/Board.ts @@ -0,0 +1,42 @@ +import {List} from "./List"; +import URLFetchRequestOptions = GoogleAppsScript.URL_Fetch.URLFetchRequestOptions; + +export class Board { + public static model: string = `/boards`; + + public static getBoard(boardId: string): Board { + const request = { + apiPath: `${this.model}/${boardId}`, + } + + const options = { + method: "get", + } as URLFetchRequestOptions + + return new Board(Trello.getTrello().authorizedRequest(request, options)) + } + + public name: string; + public id: string; + public url: string; + + constructor(obj) { + this.name = obj.name; + this.id = obj.id; + this.url = obj.url; + } + + public getLists(): [List] { + const request = { + apiPath: `${Board.model}/${this.id}/lists`, + } + + const options = { + method: "get", + } as URLFetchRequestOptions + + return Trello.getTrello().authorizedRequest(request, options).map((list) => { + return new List(list); + }); + } +} diff --git a/src/Api/Trello/Card.ts b/src/Api/Trello/Card.ts new file mode 100644 index 0000000..fbbcf9d --- /dev/null +++ b/src/Api/Trello/Card.ts @@ -0,0 +1,69 @@ +import URLFetchRequestOptions = GoogleAppsScript.URL_Fetch.URLFetchRequestOptions; + +export class Card { + public static model: string = `/cards`; + + public static getCard(cardId: string): Card { + const request = { + apiPath: `${Card.model}${cardId}`, + } + + const options = { + method: "get", + } as URLFetchRequestOptions + + return new Card(Trello.getTrello().authorizedRequest(request, options)) + } + + public name: string; + public id: string; + public url: string; + public desc: string; + public due: string; + + constructor(obj) { + this.name = obj.name; + this.id = obj.id; + this.url = obj.url; + this.desc = obj.desc; + this.due = obj.due; + } + + public save(listID: string): Card { + const request = { + apiPath: `${Card.model}`, + payload: { + desc: this.desc, + due: this.due, + idList: listID, + name: this.name, + }, + } + + const options = { + method: "post", + payload: request.payload, + } as URLFetchRequestOptions + + return new Card(Trello.getTrello().authorizedRequest(request, options)) + } + + public addLinkAttachment(url: string): Card { + const request = { + apiPath: `${Card.model}/${this.id}/attachments`, + payload: { + name: "Link to email", + url, + }, + } + + const options = { + method: "post", + payload: request.payload, + } as URLFetchRequestOptions + + Trello.getTrello().authorizedRequest(request, options) + + return this + } +} diff --git a/src/Api/Trello/List.ts b/src/Api/Trello/List.ts new file mode 100644 index 0000000..f7c6d5e --- /dev/null +++ b/src/Api/Trello/List.ts @@ -0,0 +1,43 @@ +import {Card} from "./Card"; +import URLFetchRequestOptions = GoogleAppsScript.URL_Fetch.URLFetchRequestOptions; + +export class List { + public static model: string = `/lists`; + + public static getList(listId: string): List { + const request = { + apiPath: `${List.model}/${listId}`, + } + + const options = { + method: "get", + } as URLFetchRequestOptions + + return new List(Trello.getTrello().authorizedRequest(request, options)) + } + + public name: string; + public id: string; + public url: string; + + constructor(obj) { + this.name = obj.name; + this.id = obj.id; + this.url = obj.url; + } + + public getCards(): [Card] { + const request = { + apiPath: `${List.model}/${this.id}/cards`, + } + + const options = { + method: "get", + } as URLFetchRequestOptions + + return Trello.getTrello().authorizedRequest(request, options).map((list) => { + return new List(list); + }); + } + +} diff --git a/src/Api/Trello/Member.ts b/src/Api/Trello/Member.ts new file mode 100644 index 0000000..2c0a991 --- /dev/null +++ b/src/Api/Trello/Member.ts @@ -0,0 +1,44 @@ +import {Board} from "./Board"; +import URLFetchRequestOptions = GoogleAppsScript.URL_Fetch.URLFetchRequestOptions; +import {Trello} from "./Trello"; + +export class Member { + public static model: string = `/members`; + + public static getMember(memberId: string): Member { + const request = { + apiPath: `${Member.model}/${memberId}`, + } + + const options = { + method: "get", + } as URLFetchRequestOptions + + return new Member(Trello.getTrello().authorizedRequest(request, options)) + } + + public name: string; + public id: string; + public url: string; + + constructor(obj) { + this.name = obj.name; + this.id = obj.id; + this.url = obj.url; + } + + public getBoards(): [Board] { + const request = { + apiPath: `${Member.model}/${this.id}/boards`, + params: "filter=open", + } + + const options = { + method: "get", + } as URLFetchRequestOptions + + return Trello.getTrello().authorizedRequest(request, options).map((board) => { + return new Board(board) + }) + } +} diff --git a/src/Api/Trello/Token.ts b/src/Api/Trello/Token.ts new file mode 100644 index 0000000..a55e872 --- /dev/null +++ b/src/Api/Trello/Token.ts @@ -0,0 +1,38 @@ +import {Member} from "./Member"; +import {Trello} from "./Trello"; + +export class Token { + public static stateKey: string = `user.token` + public static token + + public static getCurrentToken() { + if (this.token) { + return this.token + } else { + return this.token = new Token(StateManager.getStateManager().getState(this.stateKey)) + } + } + private model: string = `/tokens`; + private oauthToken + private public + private oauthTokenSecret + + private constructor(obj) { + this.oauthToken = obj.oauth_token + this.public = obj.public + this.oauthTokenSecret = obj.oauth_token_secret + } + + public getMember(): Member { + const request = { + apiPath: `${this.model}/${this.oauthToken}`, + } + + const options = { + method: `get`, + } + const tokenInfo = Trello.getTrello().authorizedRequest(request, options) + const memberId = tokenInfo.idMember + return Member.getMember(memberId) + } +} diff --git a/src/Api/Trello/Trello.ts b/src/Api/Trello/Trello.ts new file mode 100644 index 0000000..5d2c1c6 --- /dev/null +++ b/src/Api/Trello/Trello.ts @@ -0,0 +1,91 @@ +import {Token} from "./Token"; + +export class Trello { + public static trello + + public static getTrello() { + if (this.trello) { + return this.trello + } else { + return this.trello = new Trello() + } + } + + private static SUCCESS_HTML = "Success!" + private static DENIED_HTML = `Denied` + private static RESOURCE = `Trello account` + private static accessTokenUrl = "https://api.trello.com/1/OAuthGetAccessToken" + private static authorizationUrl = "https://api.trello.com/1/OAuthAuthorizeToken" + private static requestTokenUrl = "https://api.trello.com/1/OAuthGetRequestToken" + private static scopes = "read,write" + private static baseUrl = "https://trello.com/1" + private readonly authService + + private constructor() { + this.authService = this.getAuthService() + } + + public getUrl(props): string { + return `${Trello.baseUrl}${props.apiPath}${props.params ? "?" + props.params : ""}`; + } + + public reset() { + this.authService.reset() + } + + public handleCallback(request) { + const authorized = this.authService.handleCallback(request) + if (authorized) { + StateManager.getStateManager().setState(Token.stateKey, this.authService.getToken_()) + return HtmlService.createHtmlOutput(Trello.SUCCESS_HTML); + } else { + return HtmlService.createHtmlOutput(Trello.DENIED_HTML); + } + } + + public authorizedRequest(request, options) { + const trelloService = this.authService + let maybeAuthorized = trelloService.hasAccess() + if (maybeAuthorized) { + options.muteHttpExceptions = true + const url = this.getUrl(request) + const response = this.authService.fetch(url, options) + const code = response.getResponseCode() + if (code >= 200 && code < 300) { + return JSON.parse(response.getContentText("utf-8")) + } else if (code === 401 || code === 403) { + maybeAuthorized = false + } else { + throw new Error(`Error`) + } + } + if (!maybeAuthorized) { + CardService.newAuthorizationException() + .setAuthorizationUrl(trelloService.authorize()) + .setResourceDisplayName(Trello.RESOURCE) + .throwException() + } + } + + public buildUrl_(url, params) { + const paramString = Object.keys(params).map((key) => { + return encodeURIComponent(key) + "=" + encodeURIComponent(params[key]); + }).join("&"); + return url + (url.indexOf("?") >= 0 ? "&" : "?") + paramString; + } + + private getAuthService() { + const oAuthParams = `name=${APP_NAME}&scope=${Trello.scopes}` + const scriptProperties = PropertiesService.getScriptProperties() + // @ts-ignore + return OAuth1.createService(APP_NAME) + .setAccessTokenUrl(Trello.accessTokenUrl) + .setRequestTokenUrl(`${Trello.requestTokenUrl}?${oAuthParams}`) + .setAuthorizationUrl(`${Trello.authorizationUrl}?${oAuthParams}`) + .setConsumerKey(scriptProperties.getProperty(`trello.key`)) + .setConsumerSecret(scriptProperties.getProperty(`trello.secret`)) + .setCallbackFunction("oAuthCallback") + .setCache(CacheService.getUserCache()) + .setPropertyStore(PropertiesService.getUserProperties()) + } +} diff --git a/src/Board.ts b/src/Board.ts deleted file mode 100644 index a5dead9..0000000 --- a/src/Board.ts +++ /dev/null @@ -1,31 +0,0 @@ -import TrelloRequest = TrelloService.TrelloRequest; -import {List} from "./List"; - -export class Board { - static model:string = `/boards/` - public name:string - public id:string - public url:string - - constructor(obj) { - this.name = obj.name - this.id = obj.id - this.url = obj.url - } - - static getBoard(boardId:string):Board { - let request:TrelloRequest = { - apiPath: `/boards/${boardId}` - } as TrelloRequest - return new Board(TrelloService.send(request)) - } - - getLists():[List] { - let request:TrelloRequest = { - apiPath: `${Board.model}${this.id}/lists` - } as TrelloRequest - return TrelloService.send(request).map((list) => { - return new List(list) - }) - } -} \ No newline at end of file diff --git a/src/Card.ts b/src/Card.ts deleted file mode 100644 index 7644bf9..0000000 --- a/src/Card.ts +++ /dev/null @@ -1,54 +0,0 @@ -import TrelloRequest = TrelloService.TrelloRequest; - -export class Card { - static model:string = `/cards/` - public name:string - public id:string - public url:string - public desc:string - public due:string - - constructor(obj) { - this.name = obj.name - this.id = obj.id - this.url = obj.url - this.desc = obj.desc - } - - save(listID:string):Card { - let request:TrelloRequest = { - apiPath: `${Card.model}`, - method: 'post', - payload: { - name: this.name, - desc: this.desc, - due: this.due, - idList: listID - } - } as TrelloRequest - - return new Card(TrelloService.send(request)) - } - - addLinkAttachment(url:string):Card { - let request:TrelloRequest = { - apiPath: `${Card.model}${this.id}/attachments`, - method: 'post', - payload: { - name: "Link to email", - url: url - } - } as TrelloRequest - TrelloService.send(request) - return this - } - - static getCard(cardId:string):Card { - let request:TrelloRequest = { - apiPath: `${Card.model}${cardId}` - } as TrelloRequest - - return new Card(TrelloService.send(request)) - } - -} \ No newline at end of file diff --git a/src/List.ts b/src/List.ts deleted file mode 100644 index 49733a9..0000000 --- a/src/List.ts +++ /dev/null @@ -1,33 +0,0 @@ -import TrelloRequest = TrelloService.TrelloRequest; -import {Card} from "./Card"; - -export class List { - static model:string = `/lists/` - public name:string - public id:string - public url:string - - constructor(obj) { - this.name = obj.name - this.id = obj.id - this.url = obj.url - } - - static getList(listId:string):List { - let request:TrelloRequest = { - apiPath: `${List.model}${listId}` - } as TrelloRequest - - return new List(TrelloService.send(request)) - } - - getCards():[Card] { - let request:TrelloRequest = { - apiPath: `${List.model}${this.id}/cards` - } as TrelloRequest - return TrelloService.send(request).map((list) => { - return new List(list) - }) - } - -} \ No newline at end of file diff --git a/src/Member.ts b/src/Member.ts deleted file mode 100644 index da9b9a0..0000000 --- a/src/Member.ts +++ /dev/null @@ -1,32 +0,0 @@ -import {Board} from "./Board"; -import TrelloRequest = TrelloService.TrelloRequest; - -export class Member { - static model:string = `/members/` - public name:string - public id:string - public url:string - - constructor(obj) { - this.name = obj.name - this.id = obj.id - this.url = obj.url - } - - static getMember(memberId:string):Member { - let request:TrelloRequest = { - apiPath: `${Member.model}${memberId}` - } as TrelloRequest - return new Member(TrelloService.send(request)) - } - - getBoards():[Board] { - let request:TrelloRequest = { - apiPath: `${Member.model}${this.id}/boards`, - params: "filter=open" - } as TrelloRequest - return TrelloService.send(request).map((board) => { - return new Board(board) - }) - } -} \ No newline at end of file diff --git a/src/Message.ts b/src/Message.ts new file mode 100644 index 0000000..26e86fe --- /dev/null +++ b/src/Message.ts @@ -0,0 +1,33 @@ +class Message { + public accessToken: string; + public messageId: any; + public message: GoogleAppsScript.Gmail.GmailMessage; + public thread: GoogleAppsScript.Gmail.GmailThread; + public threadLink: string; + public subjectLine: string; + constructor(e) { + this.accessToken = e.messageMetadata.accessToken; + GmailApp.setCurrentMessageAccessToken(this.accessToken); + + this.messageId = e.messageMetadata.messageId; + this.message = GmailApp.getMessageById(this.messageId); + + this.thread = this.message.getThread(); + this.threadLink = this.thread.getPermalink(); + + this.subjectLine = this.message.getSubject() + } +} + +function getMessageDetails(e) { + const accessToken = e.messageMetadata.accessToken; + GmailApp.setCurrentMessageAccessToken(accessToken); + + const messageId = e.messageMetadata.messageId; + const message = GmailApp.getMessageById(messageId); + + const thread = message.getThread(); + const threadLink = thread.getPermalink(); + + const subjectLine = message.getSubject() +} diff --git a/src/QuickLinksAddOn.ts b/src/QuickLinksAddOn.ts deleted file mode 100644 index d813931..0000000 --- a/src/QuickLinksAddOn.ts +++ /dev/null @@ -1,171 +0,0 @@ -import {Member} from "./Member"; -import {Board} from "./Board"; -import {List} from "./List"; -import {Card} from "./Card"; - -const userProperties = PropertiesService.getUserProperties() -const state = JSON.parse(userProperties.getProperty("state")) || { - email:{ - subjectLine: undefined - }, - selectedBoard: undefined, - boards:[], - selectedList: undefined, -} - -function updateState() { - userProperties.setProperty("state", JSON.stringify(state)) -} - -function buildQuickLinkUI(e):GoogleAppsScript.Card.Card[] { - return new Array(getCardMenu({})) -} - -function getCardMenu(props):GoogleAppsScript.Card.Card { - return CardService.newCardBuilder() - .addSection(CardService.newCardSection() - .addWidget(CardService.newTextButton() - .setText("Create new card") - .setOnClickAction(CardService.newAction() - .setFunctionName("navigateToTrelloNewCard") - .setLoadIndicator(CardService.LoadIndicator.SPINNER)))) - .build() -} - -function navigateToTrelloNewCard(e) { - const accessToken = e.messageMetadata.accessToken - GmailApp.setCurrentMessageAccessToken(accessToken) - - const messageId = e.messageMetadata.messageId; - const message = GmailApp.getMessageById(messageId); - - const thread = message.getThread() - const threadLink = thread.getPermalink() - - state.email.subjectLine = message.getSubject() - const mike = Member.getMember(Config.users.mike) - const boards = mike.getBoards() - state.boards = boards - - updateState() - - return CardService.newNavigation() - .popToRoot() - .pushCard(getCardTrelloNewCard(state)) -} - -function getCardTrelloNewCard(props):GoogleAppsScript.Card.Card { - var lists = []; - if(props.selectedBoard) - lists = Board.getBoard(props.selectedBoard).getLists() - console.log(`${props.selectedBoard}:${props.selectedBoard} lists:${lists}`) - return CardService.newCardBuilder() - .addSection(CardService.newCardSection() - .addWidget(getWidgetBoardSelection({boards: props.boards, selectedBoard: props.selectedBoard})) - .addWidget(getWidgetListsSelection({lists: lists, selectedList: props.selectedList, selectedBoard: props.selectedBoard})) - .addWidget(getWidgetCardName({subjectLine:props.email.subjectLine})) - .addWidget(getWidgetDescription()) - .addWidget(getWidgetDueDate()) - .addWidget(getWidgetSubmitCard())) - .setHeader(CardService.newCardHeader().setTitle("Create a new card")) - .build() -} - -function getWidgetBoardSelection(props):GoogleAppsScript.Card.Widget { - let selectionInput = CardService.newSelectionInput() - .setFieldName("selectedBoard") - .setTitle("Board") - .setType(CardService.SelectionInputType.DROPDOWN) - .setOnChangeAction(CardService.newAction() - .setFunctionName("handleBoardSelectionChange") - .setLoadIndicator(CardService.LoadIndicator.SPINNER)) - if(!props.selectedBoard) - selectionInput.addItem("No board selected", "", true) - props.boards.forEach((board) => { - selectionInput.addItem(board.name, board.id, board.id == props.selectedBoard) - }) - return selectionInput -} - -function handleBoardSelectionChange(e) { - state.selectedBoard = e.formInput.selectedBoard - updateState() - return CardService.newNavigation() - .updateCard(getCardTrelloNewCard(state)) -} - -function getWidgetListsSelection(props):GoogleAppsScript.Card.Widget { - let selectionInput = CardService.newSelectionInput() - .setFieldName("selectedList") - .setTitle("List") - .setType(CardService.SelectionInputType.DROPDOWN) - .setOnChangeAction(CardService.newAction() - .setFunctionName("handleListSelectionChange") - .setLoadIndicator(CardService.LoadIndicator.SPINNER)) - if(!props.selectedBoard) - selectionInput.addItem("No board selected", "", true) - if(!props.selectedList) - selectionInput.addItem("No list selected", "", true) - props.lists.forEach((list:List) => { - selectionInput.addItem(list.name, list.id, props.selectedList == list.id) - }) - if(props.lists.length == 0) - selectionInput.addItem("No lists on board!", "err", true) - return selectionInput -} - -function handleListSelectionChange(e) { - state.selectedList = e.formInput.selectedList - updateState() - return CardService.newNavigation() - .updateCard(getCardTrelloNewCard(state)) -} - -function getWidgetCardName(props):GoogleAppsScript.Card.Widget { - return CardService.newTextInput() - .setTitle("Name") - .setFieldName("name") - .setValue(props.subjectLine) -} - -function getWidgetDescription():GoogleAppsScript.Card.Widget { - return CardService.newTextInput() - .setMultiline(true) - .setFieldName("description") - .setTitle("Description") -} - -function getWidgetDueDate():GoogleAppsScript.Card.Widget { - const now = new Date() - return CardService.newTextInput() - .setFieldName("dueDate") - .setTitle("Due Date") - .setHint(`${now.toLocaleDateString()} ${now.toLocaleTimeString()}`) -} - -function getWidgetSubmitCard():GoogleAppsScript.Card.Widget { - return CardService.newTextButton() - .setText("ADD") - .setOnClickAction(CardService.newAction() - .setFunctionName("handleTrelloNewCardSubmit") - .setLoadIndicator(CardService.LoadIndicator.SPINNER)) -} - -function handleTrelloNewCardSubmit(e) { - const accessToken = e.messageMetadata.accessToken - GmailApp.setCurrentMessageAccessToken(accessToken) - - const messageId = e.messageMetadata.messageId; - const message = GmailApp.getMessageById(messageId); - - const thread = message.getThread() - const threadLink = thread.getPermalink() - - new Card({ - name: e.formInput.name, - desc: e.formInput.description, - due: e.formInput.dueDate - }).save(e.formInput.selectedList).addLinkAttachment(threadLink) - return CardService.newNavigation() - .popToRoot() -} \ No newline at end of file diff --git a/src/TrelloService.ts b/src/TrelloService.ts deleted file mode 100644 index 246bcbd..0000000 --- a/src/TrelloService.ts +++ /dev/null @@ -1,161 +0,0 @@ -namespace TrelloService { - export function getUrl(request:TrelloRequest):string { - return `${Config.trello.baseUrl}${request.apiPath}${request.params ? "?"+request.params+"&" : ""}` - } - - export function send(request:TrelloRequest) { - let options:object = {} - if(request.payload){ - options['method'] = 'put' - options['payload'] = request.payload - } else { - options['method'] = 'get' - } - if(request.method) - options['method'] = request.method - //return JSON.parse(accessProtectedResource(getUrl(request), options)) - return JSON.parse(getOAuthService().fetch(getUrl(request), options).getContentText()) - } - - export interface TrelloRequest { - method: string; - apiPath:string, - params:string, - payload:object - } -} - -function testAuthFunc() { - accessProtectedResource(`https://api.trello.com/1/members/${Config.users.mike}/boards`, {}) -} - -/** - * Attempts to access a non-Google API using a constructed service - * object. - * - * If your add-on needs access to non-Google APIs that require OAuth, - * you need to implement this method. You can use the OAuth1 and - * OAuth2 Apps Script libraries to help implement it. - * - * @param {String} url The URL to access. - * @param {String} method_opt The HTTP method. Defaults to GET. - * @param {Object} headers_opt The HTTP headers. Defaults to an empty - * object. The Authorization field is added - * to the headers in this method. - * @returns {HttpResponse} the result from the UrlFetchApp.fetch() call. - */ -function accessProtectedResource(url, options) { - var service = getOAuthService(); - var maybeAuthorized = service.hasAccess(); - console.log("Checking access") - if (maybeAuthorized) { - // A token is present, but it may be expired or invalid. Make a - // request and check the response code to be sure. - - // Make the UrlFetch request and return the result. - options['muteHttpExceptions'] = true - var resp = getOAuthService().fetch(url, options); - - var code = resp.getResponseCode(); - if (code >= 200 && code < 300) { - console.log("Request was a success") - return resp.getContentText("utf-8"); // Success - } else if (code == 401 || code == 403) { - console.log("Not authorized") - // Not fully authorized for this action. - maybeAuthorized = false; - } else { - // Handle other response codes by logging them and throwing an - // exception. - console.error("Backend server error (%s): %s", code.toString(), - resp.getContentText("utf-8")); - throw ("Backend server error: " + code); - } - } - - if (!maybeAuthorized) { - console.log(`Start authorization flow ${service.authorize()}`) - // Invoke the authorization flow using the default authorization - // prompt card. - CardService.newAuthorizationException() - .setAuthorizationUrl(service.authorize()) - .setResourceDisplayName("Trello needs authorization") - .throwException(); - } -} - -/** - * Create a new OAuth service to facilitate accessing an API. - * This example assumes there is a single service that the add-on needs to - * access. Its name is used when persisting the authorized token, so ensure - * it is unique within the scope of the property store. You must set the - * client secret and client ID, which are obtained when registering your - * add-on with the API. - * - * See the Apps Script OAuth2 Library documentation for more - * information: - * https://github.com/googlesamples/apps-script-oauth2#1-create-the-oauth2-service - * - * @returns A configured OAuth2 service object. - */ -function getOAuthService() { - return OAuth1.createService('trello') - // Set the endpoint URLs. - .setAccessTokenUrl('https://trello.com/1/OAuthGetAccessToken') - .setRequestTokenUrl('https://trello.com/1/OAuthGetRequestToken') - .setAuthorizationUrl('https://trello.com/1/OAuthAuthorizeToken') - // Set the consumer key and secret. - .setConsumerKey(Config.trello.key) - .setConsumerSecret(Config.trello.secret) - // Set the name of the callback function in the script referenced - // above that should be invoked to complete the OAuth flow. - .setCallbackFunction('authCallback') - // Set the property store where authorized tokens should be persisted. - .setPropertyStore(PropertiesService.getUserProperties()); - -} - -/** - * Boilerplate code to determine if a request is authorized and returns - * a corresponding HTML message. When the user completes the OAuth2 flow - * on the service provider's website, this function is invoked from the - * service. In order for authorization to succeed you must make sure that - * the service knows how to call this function by setting the correct - * redirect URL. - * - * The redirect URL to enter is: - * https://script.google.com/macros/d//usercallback - * - * See the Apps Script OAuth2 Library documentation for more - * information: - * https://github.com/googlesamples/apps-script-oauth2#1-create-the-oauth2-service - * - * @param {Object} callbackRequest The request data received from the - * callback function. Pass it to the service's - * handleCallback() method to complete the - * authorization process. - * @returns {HtmlOutput} a success or denied HTML message to display to - * the user. Also sets a timer to close the window - * automatically. - */ -function authCallback(callbackRequest) { - console.log(`We were called back! #{callbackRequest}`) - var authorized = getOAuthService().handleCallback(callbackRequest); - if (authorized) { - console.log("YAY!") - return HtmlService.createHtmlOutput( - 'Success! '); - } else { - console.log("Denied :(") - return HtmlService.createHtmlOutput('Denied'); - } -} - -/** - * Unauthorizes the non-Google service. This is useful for OAuth - * development/testing. Run this method (Run > resetOAuth in the script - * editor) to reset OAuth to re-prompt the user for OAuth. - */ -function resetOAuth() { - getOAuthService().reset(); -} \ No newline at end of file diff --git a/src/Ui/MainMenuUi.ts b/src/Ui/MainMenuUi.ts new file mode 100644 index 0000000..0afd4af --- /dev/null +++ b/src/Ui/MainMenuUi.ts @@ -0,0 +1,21 @@ +export class MainMenuUi { + private state + private props + + constructor(props?) { + this.props = props + this.state = StateManager.getStateManager().getState(this.constructor.name) + } + + public render(): GoogleAppsScript.Card.Card { + return CardService.newCardBuilder() + .addSection(CardService.newCardSection() + .addWidget(CardService.newTextButton() + .setText("Create new card") + .setOnClickAction(CardService.newAction() + .setFunctionName("handleUiUpdate") + .setParameters({class: "TrelloNewCardUi", handler: "handleClickIntoUi"}) + .setLoadIndicator(CardService.LoadIndicator.SPINNER)))) + .build(); + } +} diff --git a/src/Ui/StateManager.ts b/src/Ui/StateManager.ts new file mode 100644 index 0000000..9376667 --- /dev/null +++ b/src/Ui/StateManager.ts @@ -0,0 +1,36 @@ +const userProperties = PropertiesService.getUserProperties(); + +class StateManager { + public static stateManager + + public static getStateManager() { + if (this.stateManager) { + return this.stateManager + } + return this.stateManager = new StateManager() + } + private state + + constructor() { + this.state = JSON.parse(userProperties.getProperty("state")) || {} + } + + public getState(stateKey) { + return this.state[stateKey] || {} + } + + public setState(stateKey, state) { + const currentState = this.state[stateKey] + this.state[stateKey] = {...currentState, ...state} + userProperties.setProperty("state", JSON.stringify(this.state)) + return this + } + + public reset() { + userProperties.deleteAllProperties() + } + + public toString = (): string => { + return JSON.stringify(this.state) + } +} diff --git a/src/Ui/Trello/BoardSelectionWidget.ts b/src/Ui/Trello/BoardSelectionWidget.ts new file mode 100644 index 0000000..fedf40e --- /dev/null +++ b/src/Ui/Trello/BoardSelectionWidget.ts @@ -0,0 +1,33 @@ +import {Token} from "../../Api/Trello/Token"; + +export class BoardSelectionWidget { + private state + private props + + private boards + + constructor(props?) { + this.props = props + this.state = StateManager.getStateManager().getState(this.constructor.name) + const user = Token.getCurrentToken().getMember() + this.boards = user.getBoards() + } + + public render(): GoogleAppsScript.Card.Widget { + const selectionInput = CardService.newSelectionInput() + .setFieldName("selectedBoard") + .setTitle("Board") + .setType(CardService.SelectionInputType.DROPDOWN) + .setOnChangeAction(CardService.newAction() + .setFunctionName("handleUiUpdate") + .setParameters({class: "TrelloNewCardUi", handler: "handleBoardSelectionChange"}) + .setLoadIndicator(CardService.LoadIndicator.SPINNER)); + if (!this.props.selectedBoard) { + selectionInput.addItem("No board selected", "", true); + } + this.boards.forEach((board) => { + selectionInput.addItem(board.name, board.id, board.id === this.props.selectedBoard); + }); + return selectionInput; + } +} diff --git a/src/Ui/Trello/CardDescriptionWidget.ts b/src/Ui/Trello/CardDescriptionWidget.ts new file mode 100644 index 0000000..389b82a --- /dev/null +++ b/src/Ui/Trello/CardDescriptionWidget.ts @@ -0,0 +1,16 @@ +export class CardDescriptionWidget { + protected state + protected props + + constructor(props?) { + this.props = props + this.state = StateManager.getStateManager().getState(this.constructor.name) + } + + public render(): GoogleAppsScript.Card.Widget { + return CardService.newTextInput() + .setMultiline(true) + .setFieldName("description") + .setTitle("Description"); + } +} diff --git a/src/Ui/Trello/CardDueDateWidget.ts b/src/Ui/Trello/CardDueDateWidget.ts new file mode 100644 index 0000000..e02911f --- /dev/null +++ b/src/Ui/Trello/CardDueDateWidget.ts @@ -0,0 +1,17 @@ +export class CardDueDateWidget { + protected state + protected props + + constructor(props?) { + this.props = props + this.state = StateManager.getStateManager().getState(this.constructor.name) + } + + public render(): GoogleAppsScript.Card.Widget { + const now = new Date(); + return CardService.newTextInput() + .setFieldName("due") + .setTitle("Due Date") + .setHint(`${now.toLocaleDateString()} ${now.toLocaleTimeString()}`); + } +} diff --git a/src/Ui/Trello/CardNameWidget.ts b/src/Ui/Trello/CardNameWidget.ts new file mode 100644 index 0000000..d8b3cfa --- /dev/null +++ b/src/Ui/Trello/CardNameWidget.ts @@ -0,0 +1,16 @@ +export class CardNameWidget { + protected state + protected props + + constructor(props?) { + this.props = props + this.state = StateManager.getStateManager().getState(this.constructor.name) + } + + public render() { + return CardService.newTextInput() + .setTitle("Name") + .setFieldName("name") + .setValue(this.props.subjectLine); + } +} diff --git a/src/Ui/Trello/CardSubmitWidget.ts b/src/Ui/Trello/CardSubmitWidget.ts new file mode 100644 index 0000000..656e353 --- /dev/null +++ b/src/Ui/Trello/CardSubmitWidget.ts @@ -0,0 +1,32 @@ +import {Card} from "../../Api/Trello/Card"; + +export class CardSubmitWidget { + + public static handleSubmit(e) { + const message = new Message(e) + + new Card({ + desc: e.formInput.description, + due: e.formInput.due, + name: e.formInput.name, + }).save(e.formInput.selectedList).addLinkAttachment(message.threadLink); + return CardService.newNavigation() + .popToRoot(); + } + protected state + protected props + + constructor(props?) { + this.props = props + this.state = StateManager.getStateManager().getState(this.constructor.name) + } + + public render(): GoogleAppsScript.Card.Widget { + return CardService.newTextButton() + .setText("ADD") + .setOnClickAction(CardService.newAction() + .setFunctionName("handleUiUpdate") + .setParameters({class: "CardSubmitWidget", handler: "handleSubmit"}) + .setLoadIndicator(CardService.LoadIndicator.SPINNER)); + } +} diff --git a/src/Ui/Trello/ListSelectionWidget.ts b/src/Ui/Trello/ListSelectionWidget.ts new file mode 100644 index 0000000..fb23be2 --- /dev/null +++ b/src/Ui/Trello/ListSelectionWidget.ts @@ -0,0 +1,48 @@ +import {Board} from "../../Api/Trello/Board"; +import {TrelloNewCardUi} from "./TrelloNewCardUi"; + +export class ListSelectionWidget { + + public static handleListSelectionChange(e) { + StateManager.getStateManager().setState(this.name, {selectedList: e.formInput.selectedList}) + return CardService.newNavigation() + .updateCard(new TrelloNewCardUi().render()) // TODO: Find UI parent instead of hardcoding + } + protected state + protected props + + private lists = [] + + constructor(props) { + this.props = props + this.state = StateManager.getStateManager().getState(this.constructor.name) + if (props.selectedBoard) { + const board = Board.getBoard(this.props.selectedBoard) + this.lists = board.getLists() + } + } + + public render(): GoogleAppsScript.Card.Widget { + const selectionInput = CardService.newSelectionInput() + .setFieldName("selectedList") + .setTitle("List") + .setType(CardService.SelectionInputType.DROPDOWN) + .setOnChangeAction(CardService.newAction() + .setFunctionName("handleUiUpdate") + .setParameters({class: "ListSelectionWidget", handler: "handleListSelectionChange"}) + .setLoadIndicator(CardService.LoadIndicator.SPINNER)); + if (!this.props.selectedBoard) { + selectionInput.addItem("No board selected", "", true); + } + if (!this.state.selectedList) { + selectionInput.addItem("No list selected", "", true); + } + this.lists.forEach((list) => { + selectionInput.addItem(list.name, list.id, this.state.selectedList === list.id); + }); + if (this.lists.length === 0) { + selectionInput.addItem("No lists on board!", "err", true); + } + return selectionInput; + } +} diff --git a/src/Ui/Trello/TrelloNewCardUi.ts b/src/Ui/Trello/TrelloNewCardUi.ts new file mode 100644 index 0000000..fa4a0f4 --- /dev/null +++ b/src/Ui/Trello/TrelloNewCardUi.ts @@ -0,0 +1,55 @@ +import {BoardSelectionWidget} from "./BoardSelectionWidget"; +import {CardDescriptionWidget} from "./CardDescriptionWidget"; +import {CardDueDateWidget} from "./CardDueDateWidget"; +import {CardNameWidget} from "./CardNameWidget"; +import {CardSubmitWidget} from "./CardSubmitWidget"; +import {ListSelectionWidget} from "./ListSelectionWidget"; + +export class TrelloNewCardUi { + + public static handleClickIntoUi(e) { + const message = new Message(e) + + StateManager.getStateManager().setState(this.name, {subjectLine: message.subjectLine}) + + return CardService.newNavigation() + .popToRoot() + .pushCard(new TrelloNewCardUi().render()) + } + + public static handleBoardSelectionChange(e) { + StateManager.getStateManager().setState(this.name, {selectedBoard: e.formInput.selectedBoard}) + return CardService.newNavigation() + .updateCard(new TrelloNewCardUi().render()) // TODO: Find UI parent instead of hardcoding + + } + protected state + protected props + + constructor(props?) { + this.props = props + this.state = StateManager.getStateManager().getState(this.constructor.name) + } + + public render(): GoogleAppsScript.Card.Card { + return CardService.newCardBuilder() + .addSection(CardService.newCardSection() + .addWidget(new BoardSelectionWidget({selectedBoard: this.state.selectedBoard}).render()) + .addWidget(new ListSelectionWidget({selectedBoard: this.state.selectedBoard}).render()) + .addWidget(new CardNameWidget({subjectLine: this.state.subjectLine}).render()) + .addWidget(new CardDescriptionWidget().render()) + .addWidget(new CardDueDateWidget().render()) + .addWidget(new CardSubmitWidget().render())) + .setHeader(CardService.newCardHeader().setTitle("Create a new card")) + .build(); + } +} + + + + + + + + + diff --git a/src/googleAppsScriptEntryPoints.ts b/src/googleAppsScriptEntryPoints.ts new file mode 100644 index 0000000..bd8493b --- /dev/null +++ b/src/googleAppsScriptEntryPoints.ts @@ -0,0 +1,25 @@ +import {Trello} from "./Api/Trello/Trello"; +import {MainMenuUi} from "./Ui/MainMenuUi"; + +const APP_NAME = `quickLink` + +function handleUiUpdate(e) { + // TODO: Create a whitelist for classes/methods that can be called + // TODO: Use decorators to add handlers to the whitelist + // TODO: Ensure `e` is handled correctly + // tslint:disable-next-line no-eval + return eval(`${e.parameters.class}.${e.parameters.handler}(JSON.parse('${JSON.stringify(e)}'))`) +} + +function buildQuickLinkUI(e): GoogleAppsScript.Card.Card[] { + return new Array(new MainMenuUi().render()); +} + +function oAuthCallback(request) { + return Trello.getTrello().handleCallback(request) +} + +function logout() { + StateManager.getStateManager().reset() + Trello.getTrello().reset() +} diff --git a/tslint.json b/tslint.json new file mode 100644 index 0000000..65dc293 --- /dev/null +++ b/tslint.json @@ -0,0 +1,31 @@ +{ + "extends": "tslint:recommended", + "rules": { + "max-line-length": { + "options": [120] + }, + "new-parens": true, + "no-arg": true, + "no-bitwise": true, + "no-conditional-assignment": true, + "no-consecutive-blank-lines": false, + "no-console": { + "severity": "warning", + "options": [ + "debug", + "info", + "log", + "time", + "timeEnd", + "trace" + ] + }, + "no-namespace": false, + "semicolon": false + }, + "jsRules": { + "max-line-length": { + "options": [120] + } + } +} \ No newline at end of file From cea52f04df0e80e0e9229a3b24f750a94b11e904 Mon Sep 17 00:00:00 2001 From: "Michael R. Kenworthy" Date: Sat, 8 Dec 2018 00:40:55 -0800 Subject: [PATCH 3/5] Updated readme --- .claspignore | 4 + appsscript.json | 2 +- package-lock.json | 2517 ++++++++++++++++++++++++++-- package.json | 7 +- readme.md | 29 +- src/Ui/StateManager.ts | 2 +- src/googleAppsScriptEntryPoints.ts | 3 +- 7 files changed, 2404 insertions(+), 160 deletions(-) create mode 100644 .claspignore diff --git a/.claspignore b/.claspignore new file mode 100644 index 0000000..5f85f5c --- /dev/null +++ b/.claspignore @@ -0,0 +1,4 @@ +**/node_modules/** +tslint.json +package.json +package-lock.json \ No newline at end of file diff --git a/appsscript.json b/appsscript.json index 31be5dd..cc7009f 100644 --- a/appsscript.json +++ b/appsscript.json @@ -22,7 +22,7 @@ "https://script.google.com/a/kenworthy.io" ], "gmail": { - "name": "QuickLinks", + "name": "TrelloTools", "logoUrl": "https://upload.wikimedia.org/wikipedia/commons/thumb/2/25/External.svg/100px-External.svg.png", "contextualTriggers": [{ "unconditional": {}, diff --git a/package-lock.json b/package-lock.json index 548ff3e..ff0ac25 100644 --- a/package-lock.json +++ b/package-lock.json @@ -4,6 +4,84 @@ "lockfileVersion": 1, "requires": true, "dependencies": { + "@google/clasp": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/@google/clasp/-/clasp-1.7.0.tgz", + "integrity": "sha512-dW96fqC4F2wJvvBHgQt8aZW7z6CKgxdfYgSBDhl4MupndHHBcJnkYIxg+XJBM4HDt2ofT4Ue8JoeiTbhYOX9IA==", + "dev": true, + "requires": { + "anymatch": "^1.3.2", + "axios": "^0.18.0", + "chalk": "^2.4.1", + "cli-spinner": "^0.2.8", + "commander": "^2.15.1", + "connect": "^3.6.6", + "del": "^3.0.0", + "dotf": "^1.2.0", + "ellipsize": "^0.1.0", + "find-parent-dir": "^0.3.0", + "fs": "^0.0.1-security", + "fuzzy": "^0.1.3", + "google-auth-library": "^2.0.0", + "googleapis": "^34.0.0", + "http-shutdown": "^1.2.0", + "inquirer": "^5.2.0", + "inquirer-autocomplete-prompt": "1.0.1", + "is-online": "^7.0.0", + "mkdirp": "^0.5.1", + "multimatch": "^2.1.0", + "opn": "^5.3.0", + "path": "^0.12.7", + "pluralize": "^7.0.0", + "read-file": "^0.2.0", + "read-multiple-files": "^1.1.1", + "readline": "^1.3.0", + "recursive-readdir": "^2.2.2", + "split-lines": "^1.1.0", + "string.prototype.padend": "^3.0.0", + "ts2gas": "^1.3.0", + "ucfirst": "^1.0.0", + "url": "^0.11.0", + "watch": "^1.0.2" + } + }, + "@sindresorhus/is": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.7.0.tgz", + "integrity": "sha512-ONhaKPIufzzrlNbqtWFFd+jlnemX6lJAgq9ZeiZtS7I1PIf/la7CW4m83rTXRnVnsMbW2k56pGYu7AUFJD9Pow==", + "dev": true + }, + "@types/google-apps-script": { + "version": "0.0.32", + "resolved": "https://registry.npmjs.org/@types/google-apps-script/-/google-apps-script-0.0.32.tgz", + "integrity": "sha512-za4/vCLMvDo2pG8S7vLQXwX7KtOkNk8vT/jPnNQTHFl03Y7aHd0jnL+eOHK8lYN2XIEtfhkv/5D1HgNbTCoaGg==", + "dev": true + }, + "agent-base": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.2.1.tgz", + "integrity": "sha512-JVwXMr9nHYTUXsBFKUqhJwvlcYU/blreOEUkhNR2eXZIvwd+c+o5V4MgDPKWnMS/56awN3TRzIP+KoPn+roQtg==", + "dev": true, + "requires": { + "es6-promisify": "^5.0.0" + } + }, + "aggregate-error": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-1.0.0.tgz", + "integrity": "sha1-iINE2tAiCnLjr1CQYRf0h3GSX6w=", + "dev": true, + "requires": { + "clean-stack": "^1.0.0", + "indent-string": "^3.0.0" + } + }, + "ansi-escapes": { + "version": "3.1.0", + "resolved": "http://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.1.0.tgz", + "integrity": "sha512-UgAb8H9D41AQnu/PbWlCofQVcnV4Gs2bBJi9eZPxfU/hgglFh3SMDMENRIqdr7H6XFnXdoknctFByVsCOotTVw==", + "dev": true + }, "ansi-regex": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", @@ -16,6 +94,16 @@ "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", "dev": true }, + "anymatch": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-1.3.2.tgz", + "integrity": "sha512-0XNayC8lTHQ2OI8aljNCN3sSx6hsr/1+rlcDAotXJR7C1oZZHCNsfpbKwMjRA3Uqb5tF1Rae2oloTr4xpq+WjA==", + "dev": true, + "requires": { + "micromatch": "^2.1.5", + "normalize-path": "^2.0.0" + } + }, "argparse": { "version": "1.0.10", "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", @@ -25,6 +113,64 @@ "sprintf-js": "~1.0.2" } }, + "arr-diff": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-2.0.0.tgz", + "integrity": "sha1-jzuCf5Vai9ZpaX5KQlasPOrjVs8=", + "dev": true, + "requires": { + "arr-flatten": "^1.0.1" + } + }, + "arr-flatten": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", + "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", + "dev": true + }, + "array-differ": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/array-differ/-/array-differ-1.0.0.tgz", + "integrity": "sha1-7/UuN1gknTO+QCuLuOVkuytdQDE=", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "array-uniq": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", + "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=", + "dev": true + }, + "array-unique": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.2.1.tgz", + "integrity": "sha1-odl8yvy8JiXMcPrc6zalDFiwGlM=", + "dev": true + }, + "arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", + "dev": true + }, + "axios": { + "version": "0.18.0", + "resolved": "http://registry.npmjs.org/axios/-/axios-0.18.0.tgz", + "integrity": "sha1-MtU+SFHv3AoRmTts0AB4nXDAUQI=", + "dev": true, + "requires": { + "follow-redirects": "^1.3.0", + "is-buffer": "^1.1.5" + } + }, "babel-code-frame": { "version": "6.26.0", "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", @@ -67,12 +213,58 @@ "concat-map": "0.0.1" } }, + "braces": { + "version": "1.8.5", + "resolved": "https://registry.npmjs.org/braces/-/braces-1.8.5.tgz", + "integrity": "sha1-uneWLhLf+WnWt2cR6RS3N4V79qc=", + "dev": true, + "requires": { + "expand-range": "^1.8.1", + "preserve": "^0.2.0", + "repeat-element": "^1.1.2" + } + }, + "buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=", + "dev": true + }, "builtin-modules": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", "dev": true }, + "cacheable-request": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-2.1.4.tgz", + "integrity": "sha1-DYCIAbY0KtM8kd+dC0TcCbkeXD0=", + "dev": true, + "requires": { + "clone-response": "1.0.2", + "get-stream": "3.0.0", + "http-cache-semantics": "3.8.1", + "keyv": "3.0.0", + "lowercase-keys": "1.0.0", + "normalize-url": "2.0.1", + "responselike": "1.0.2" + }, + "dependencies": { + "lowercase-keys": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.0.tgz", + "integrity": "sha1-TjNms55/VFfjXxMkvfb4jQv8cwY=", + "dev": true + } + } + }, + "capture-stack-trace": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/capture-stack-trace/-/capture-stack-trace-1.0.1.tgz", + "integrity": "sha512-mYQLZnx5Qt1JgB1WEiMCf2647plpGeQ2NMR/5L0HNZzGQo4fuSPnK+wjfPnKZV0aiJDgzmWqqkV/g7JD+DW0qw==", + "dev": true + }, "chalk": { "version": "2.4.1", "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.1.tgz", @@ -104,13 +296,46 @@ } } }, - "clasp": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/clasp/-/clasp-0.1.0.tgz", - "integrity": "sha1-zHKsAZEunxMgdckKy8MTSA/C9HA=", + "chardet": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.4.2.tgz", + "integrity": "sha1-tUc7M9yXxCTl2Y3IfVXU2KKci/I=", + "dev": true + }, + "clean-stack": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-1.3.0.tgz", + "integrity": "sha1-noIVAa6XmYbEax1m0tQy2y/UrjE=", + "dev": true + }, + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, + "requires": { + "restore-cursor": "^2.0.0" + } + }, + "cli-spinner": { + "version": "0.2.8", + "resolved": "https://registry.npmjs.org/cli-spinner/-/cli-spinner-0.2.8.tgz", + "integrity": "sha512-Z4l0jljucEUsW/5GNaVYOpGgVmnB/Sq4l2vG9k7RbJYRZ97hVncP+BqRMemTZNBHZ4aYUPJPU/wl6zt/ZSqzPQ==", + "dev": true + }, + "cli-width": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.0.tgz", + "integrity": "sha1-/xnt6Kml5XkyQUewwR8PvLq+1jk=", + "dev": true + }, + "clone-response": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", + "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", "dev": true, "requires": { - "pegjs": "0.7.0" + "mimic-response": "^1.0.0" } }, "color-convert": { @@ -140,215 +365,2201 @@ "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", "dev": true }, - "diff": { - "version": "3.5.0", - "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", - "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", - "dev": true - }, - "escape-string-regexp": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", - "dev": true - }, - "esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true - }, - "esutils": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", - "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=", - "dev": true + "connect": { + "version": "3.6.6", + "resolved": "https://registry.npmjs.org/connect/-/connect-3.6.6.tgz", + "integrity": "sha1-Ce/2xVr3I24TcTWnJXSFi2eG9SQ=", + "dev": true, + "requires": { + "debug": "2.6.9", + "finalhandler": "1.1.0", + "parseurl": "~1.3.2", + "utils-merge": "1.0.1" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + } + } }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", + "core-util-is": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", + "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", "dev": true }, - "glob": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", - "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", + "create-error-class": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/create-error-class/-/create-error-class-3.0.2.tgz", + "integrity": "sha1-Br56vvlHo/FKMP1hBnHUAbyot7Y=", "dev": true, "requires": { - "fs.realpath": "^1.0.0", - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "^3.0.4", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" + "capture-stack-trace": "^1.0.0" } }, - "has-ansi": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", - "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "cwd": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/cwd/-/cwd-0.10.0.tgz", + "integrity": "sha1-FyQAaUBXwioTsM8WFix+S3p/5Wc=", "dev": true, "requires": { - "ansi-regex": "^2.0.0" + "find-pkg": "^0.1.2", + "fs-exists-sync": "^0.1.0" } }, - "has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", - "dev": true - }, - "inflight": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", "dev": true, "requires": { - "once": "^1.3.0", - "wrappy": "1" + "ms": "2.0.0" } }, - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "dev": true - }, - "js-tokens": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", - "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=", + "decode-uri-component": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", + "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", "dev": true }, - "js-yaml": { - "version": "3.12.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.12.0.tgz", - "integrity": "sha512-PIt2cnwmPfL4hKNwqeiuz4bKfnzHTBv6HyVgjahA6mPLwPDzjDWrplJBMjHUFxku/N3FlmrbyPclad+I+4mJ3A==", + "decompress-response": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", + "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", "dev": true, "requires": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" + "mimic-response": "^1.0.0" } }, - "minimatch": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", - "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "define-properties": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", + "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", "dev": true, "requires": { - "brace-expansion": "^1.1.7" + "object-keys": "^1.0.12" } }, - "once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "del": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/del/-/del-3.0.0.tgz", + "integrity": "sha1-U+z2mf/LyzljdpGrE7rxYIGXZuU=", "dev": true, "requires": { - "wrappy": "1" + "globby": "^6.1.0", + "is-path-cwd": "^1.0.0", + "is-path-in-cwd": "^1.0.0", + "p-map": "^1.1.1", + "pify": "^3.0.0", + "rimraf": "^2.2.8" } }, - "path-is-absolute": { - "version": "1.0.1", - "resolved": "http://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", - "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", + "depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=", "dev": true }, - "path-parse": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", - "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", + "diff": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", + "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", "dev": true }, - "pegjs": { - "version": "0.7.0", - "resolved": "http://registry.npmjs.org/pegjs/-/pegjs-0.7.0.tgz", - "integrity": "sha1-qqH4JPnnGX7ETiKlevUi9wDdaJ4=", - "dev": true + "dns-packet": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/dns-packet/-/dns-packet-1.3.1.tgz", + "integrity": "sha512-0UxfQkMhYAUaZI+xrNZOz/as5KgDU0M/fQ9b6SpkyLbk3GEswDi6PADJVaYJradtRVsRIlF1zLyOodbcTCDzUg==", + "dev": true, + "requires": { + "ip": "^1.1.0", + "safe-buffer": "^5.0.1" + } }, - "resolve": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.8.1.tgz", - "integrity": "sha512-AicPrAC7Qu1JxPCZ9ZgCZlY35QgFnNqc+0LtbRNxnVw4TXvjQ72wnuL9JQcEBgXkI9JM8MsT9kaQoHcpCRJOYA==", + "dns-socket": { + "version": "1.6.3", + "resolved": "https://registry.npmjs.org/dns-socket/-/dns-socket-1.6.3.tgz", + "integrity": "sha512-/mUy3VGqIP69dAZjh2xxHXcpK9wk2Len1Dxz8mWAdrIgFC8tnR/aQAyU4a+UTXzOcTvEvGBdp1zFiwnpWKaXng==", "dev": true, "requires": { - "path-parse": "^1.0.5" + "dns-packet": "^1.1.0" } }, - "semver": { - "version": "5.6.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.6.0.tgz", - "integrity": "sha512-RS9R6R35NYgQn++fkDWaOmqGoj4Ek9gGs+DPxNUZKuwE183xjJroKvyo1IzVFeXvUrvmALy6FWD5xrdJT25gMg==", - "dev": true + "dotf": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/dotf/-/dotf-1.2.0.tgz", + "integrity": "sha512-InO+tJ/NBPlxrCiwQpTt7eOxgA3B8IoZboYGmt94MaWT5mmTfr7mHK1RT7OFNvCJx7UFoJ3g0/rq0c+QQWk+8w==", + "dev": true, + "requires": { + "jsonfile": "^3.0.1", + "resolve-file": "^0.3.0", + "resolve-path": "^1.3.3" + } }, - "sprintf-js": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", - "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", + "duplexer3": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", + "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", "dev": true }, - "strip-ansi": { - "version": "3.0.1", - "resolved": "http://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", - "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "ecdsa-sig-formatter": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.10.tgz", + "integrity": "sha1-HFlQAPBKiJffuFAAiSoPTDOvhsM=", "dev": true, "requires": { - "ansi-regex": "^2.0.0" + "safe-buffer": "^5.0.1" } }, - "supports-color": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", - "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=", "dev": true }, - "tslib": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.9.3.tgz", - "integrity": "sha512-4krF8scpejhaOgqzBEcGM7yDIEfi0/8+8zDRZhNZZ2kjmHJ4hv3zCbQWxoJGz1iw5U0Jl0nma13xzHXcncMavQ==", + "ellipsize": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/ellipsize/-/ellipsize-0.1.0.tgz", + "integrity": "sha1-nUNoLUS5GtFuvYQmisEDFwplU/g=", "dev": true }, - "tslint": { - "version": "5.11.0", - "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.11.0.tgz", - "integrity": "sha1-mPMMAurjzecAYgHkwzywi0hYHu0=", + "encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", + "dev": true + }, + "es-abstract": { + "version": "1.12.0", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.12.0.tgz", + "integrity": "sha512-C8Fx/0jFmV5IPoMOFPA9P9G5NtqW+4cOPit3MIuvR2t7Ag2K15EJTpxnHAYTzL+aYQJIESYeXZmDBfOBE1HcpA==", "dev": true, "requires": { - "babel-code-frame": "^6.22.0", - "builtin-modules": "^1.1.1", - "chalk": "^2.3.0", - "commander": "^2.12.1", - "diff": "^3.2.0", - "glob": "^7.1.1", - "js-yaml": "^3.7.0", - "minimatch": "^3.0.4", - "resolve": "^1.3.2", - "semver": "^5.3.0", - "tslib": "^1.8.0", - "tsutils": "^2.27.2" + "es-to-primitive": "^1.1.1", + "function-bind": "^1.1.1", + "has": "^1.0.1", + "is-callable": "^1.1.3", + "is-regex": "^1.0.4" } }, - "tsutils": { - "version": "2.29.0", - "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", - "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", + "es-to-primitive": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.0.tgz", + "integrity": "sha512-qZryBOJjV//LaxLTV6UC//WewneB3LcXOL9NP++ozKVXsIIIpm/2c13UDiD9Jp2eThsecw9m3jPqDwTyobcdbg==", "dev": true, "requires": { - "tslib": "^1.8.1" + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" } }, - "typescript": { - "version": "3.3.0-dev.20181205", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.3.0-dev.20181205.tgz", - "integrity": "sha512-XYoXPbSZrlKZOA5ykKjzwvGjAtwEhII/hBFXoGzbxfIWaT5uPcg3dISdE8qZ7jb6Xg1vjV2ghtTevToE2+IvCA==", + "es6-promise": { + "version": "4.2.5", + "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.5.tgz", + "integrity": "sha512-n6wvpdE43VFtJq+lUDYDBFUwV8TZbuGXLV4D6wKafg13ldznKsyEvatubnmUe31zcvelSzOHF+XbaT+Bl9ObDg==", "dev": true }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "es6-promisify": { + "version": "5.0.0", + "resolved": "http://registry.npmjs.org/es6-promisify/-/es6-promisify-5.0.0.tgz", + "integrity": "sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM=", + "dev": true, + "requires": { + "es6-promise": "^4.0.3" + } + }, + "escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=", + "dev": true + }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", + "dev": true + }, + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true + }, + "esutils": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", + "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=", + "dev": true + }, + "exec-sh": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/exec-sh/-/exec-sh-0.2.2.tgz", + "integrity": "sha512-FIUCJz1RbuS0FKTdaAafAByGS0CPvU3R0MeHxgtl+djzCc//F8HakL8GzmVNZanasTbTAY/3DRFA0KpVqj/eAw==", + "dev": true, + "requires": { + "merge": "^1.2.0" + } + }, + "expand-brackets": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-0.1.5.tgz", + "integrity": "sha1-3wcoTjQqgHzXM6xa9yQR5YHRF3s=", + "dev": true, + "requires": { + "is-posix-bracket": "^0.1.0" + } + }, + "expand-range": { + "version": "1.8.2", + "resolved": "http://registry.npmjs.org/expand-range/-/expand-range-1.8.2.tgz", + "integrity": "sha1-opnv/TNf4nIeuujiV+x5ZE/IUzc=", + "dev": true, + "requires": { + "fill-range": "^2.1.0" + } + }, + "expand-tilde": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/expand-tilde/-/expand-tilde-2.0.2.tgz", + "integrity": "sha1-l+gBqgUt8CRU3kawK/YhZCzchQI=", + "dev": true, + "requires": { + "homedir-polyfill": "^1.0.1" + } + }, + "extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", + "dev": true + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "external-editor": { + "version": "2.2.0", + "resolved": "http://registry.npmjs.org/external-editor/-/external-editor-2.2.0.tgz", + "integrity": "sha512-bSn6gvGxKt+b7+6TKEv1ZycHleA7aHhRHyAqJyp5pbUFuYYNIzpZnQDk7AsYckyWdEnTeAnay0aCy2aV6iTk9A==", + "dev": true, + "requires": { + "chardet": "^0.4.0", + "iconv-lite": "^0.4.17", + "tmp": "^0.0.33" + } + }, + "extglob": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-0.3.2.tgz", + "integrity": "sha1-Lhj/PS9JqydlzskCPwEdqo2DSaE=", + "dev": true, + "requires": { + "is-extglob": "^1.0.0" + } + }, + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "filename-regex": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/filename-regex/-/filename-regex-2.0.1.tgz", + "integrity": "sha1-wcS5vuPglyXdsQa3XB4wH+LxiyY=", + "dev": true + }, + "fill-range": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-2.2.4.tgz", + "integrity": "sha512-cnrcCbj01+j2gTG921VZPnHbjmdAf8oQV/iGeV2kZxGSyfYjjTyY79ErsK1WJWMpw6DaApEX72binqJE+/d+5Q==", + "dev": true, + "requires": { + "is-number": "^2.1.0", + "isobject": "^2.0.0", + "randomatic": "^3.0.0", + "repeat-element": "^1.1.2", + "repeat-string": "^1.5.2" + } + }, + "finalhandler": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.0.tgz", + "integrity": "sha1-zgtoVbRYU+eRsvzGgARtiCU91/U=", + "dev": true, + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.1", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.2", + "statuses": "~1.3.1", + "unpipe": "~1.0.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + } + } + }, + "find-file-up": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/find-file-up/-/find-file-up-0.1.3.tgz", + "integrity": "sha1-z2gJG8+fMApA2kEbN9pczlovvqA=", + "dev": true, + "requires": { + "fs-exists-sync": "^0.1.0", + "resolve-dir": "^0.1.0" + } + }, + "find-parent-dir": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/find-parent-dir/-/find-parent-dir-0.3.0.tgz", + "integrity": "sha1-M8RLQpqysvBkYpnF+fcY83b/jVQ=", + "dev": true + }, + "find-pkg": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/find-pkg/-/find-pkg-0.1.2.tgz", + "integrity": "sha1-G9wiwG42NlUy4qJIBGhUuXiNpVc=", + "dev": true, + "requires": { + "find-file-up": "^0.1.2" + } + }, + "follow-redirects": { + "version": "1.5.10", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.5.10.tgz", + "integrity": "sha512-0V5l4Cizzvqt5D44aTXbFZz+FtyXV1vrDN6qrelxtfYQKW0KO0W2T/hkE8xvGa/540LkZlkaUjO4ailYTFtHVQ==", + "dev": true, + "requires": { + "debug": "=3.1.0" + } + }, + "for-in": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", + "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", + "dev": true + }, + "for-own": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/for-own/-/for-own-0.1.5.tgz", + "integrity": "sha1-UmXGgaTylNq78XyVCbZ2OqhFEM4=", + "dev": true, + "requires": { + "for-in": "^1.0.1" + } + }, + "from2": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", + "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "readable-stream": "^2.0.0" + } + }, + "fs": { + "version": "0.0.1-security", + "resolved": "https://registry.npmjs.org/fs/-/fs-0.0.1-security.tgz", + "integrity": "sha1-invTcYa23d84E/I4WLV+yq9eQdQ=", + "dev": true + }, + "fs-exists-sync": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/fs-exists-sync/-/fs-exists-sync-0.1.0.tgz", + "integrity": "sha1-mC1ok6+RjnLQjeyehnP/K1qNat0=", + "dev": true + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", + "dev": true + }, + "function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", + "dev": true + }, + "fuzzy": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/fuzzy/-/fuzzy-0.1.3.tgz", + "integrity": "sha1-THbsL/CsGjap3M+aAN+GIweNTtg=", + "dev": true + }, + "gcp-metadata": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/gcp-metadata/-/gcp-metadata-0.7.0.tgz", + "integrity": "sha512-ffjC09amcDWjh3VZdkDngIo7WoluyC5Ag9PAYxZbmQLOLNI8lvPtoKTSCyU54j2gwy5roZh6sSMTfkY2ct7K3g==", + "dev": true, + "requires": { + "axios": "^0.18.0", + "extend": "^3.0.1", + "retry-axios": "0.3.2" + } + }, + "get-stream": { + "version": "3.0.0", + "resolved": "http://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", + "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=", + "dev": true + }, + "glob": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", + "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-base": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/glob-base/-/glob-base-0.3.0.tgz", + "integrity": "sha1-27Fk9iIbHAscz4Kuoyi0l98Oo8Q=", + "dev": true, + "requires": { + "glob-parent": "^2.0.0", + "is-glob": "^2.0.0" + } + }, + "glob-parent": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-2.0.0.tgz", + "integrity": "sha1-gTg9ctsFT8zPUzbaqQLxgvbtuyg=", + "dev": true, + "requires": { + "is-glob": "^2.0.0" + } + }, + "global-modules": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-0.2.3.tgz", + "integrity": "sha1-6lo77ULG1s6ZWk+KEmm12uIjgo0=", + "dev": true, + "requires": { + "global-prefix": "^0.1.4", + "is-windows": "^0.2.0" + } + }, + "global-prefix": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-0.1.5.tgz", + "integrity": "sha1-jTvGuNo8qBEqFg2NSW/wRiv+948=", + "dev": true, + "requires": { + "homedir-polyfill": "^1.0.0", + "ini": "^1.3.4", + "is-windows": "^0.2.0", + "which": "^1.2.12" + } + }, + "globby": { + "version": "6.1.0", + "resolved": "http://registry.npmjs.org/globby/-/globby-6.1.0.tgz", + "integrity": "sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=", + "dev": true, + "requires": { + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "dependencies": { + "pify": { + "version": "2.3.0", + "resolved": "http://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true + } + } + }, + "google-auth-library": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/google-auth-library/-/google-auth-library-2.0.1.tgz", + "integrity": "sha512-CWLKZxqYw4SE+fE3GWbVT9r/10h75w8lB3cdmmLpLtCfccFDcsI84qI5rx7npemlrHtKJh3C2HUz4s6SihCeIQ==", + "dev": true, + "requires": { + "axios": "^0.18.0", + "gcp-metadata": "^0.7.0", + "gtoken": "^2.3.0", + "https-proxy-agent": "^2.2.1", + "jws": "^3.1.5", + "lodash.isstring": "^4.0.1", + "lru-cache": "^4.1.3", + "semver": "^5.5.0" + } + }, + "google-p12-pem": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/google-p12-pem/-/google-p12-pem-1.0.3.tgz", + "integrity": "sha512-KGnAiMMWaJp4j4tYVvAjfP3wCKZRLv9M1Nir2wRRNWUYO7j1aX8O9Qgz+a8/EQ5rAvuo4SIu79n6SIdkNl7Msg==", + "dev": true, + "requires": { + "node-forge": "^0.7.5", + "pify": "^4.0.0" + }, + "dependencies": { + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + } + } + }, + "googleapis": { + "version": "34.0.0", + "resolved": "https://registry.npmjs.org/googleapis/-/googleapis-34.0.0.tgz", + "integrity": "sha512-nGfTSrlQF77HDNOHDy0ii3ET1h8Yap6QXxkfMZsre+7hBg91g4RsgrA50BgrOXpbNlQCBOGXWhUsa267kVeA/Q==", + "dev": true, + "requires": { + "google-auth-library": "^2.0.0", + "googleapis-common": "^0.3.0" + } + }, + "googleapis-common": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/googleapis-common/-/googleapis-common-0.3.0.tgz", + "integrity": "sha512-OqQ2iskzjPHLoM+AXk7e/TmEsdgxyAk8PVbMg0S8v2wPhgMu2wTawEL7zH9QG236u/RqQ1Ak120oSWsamPnWGg==", + "dev": true, + "requires": { + "axios": "^0.18.0", + "google-auth-library": "^2.0.0", + "pify": "^3.0.0", + "qs": "^6.5.2", + "url-template": "^2.0.8", + "uuid": "^3.2.1" + } + }, + "got": { + "version": "6.7.1", + "resolved": "http://registry.npmjs.org/got/-/got-6.7.1.tgz", + "integrity": "sha1-JAzQV4WpoY5WHcG0S0HHY+8ejbA=", + "dev": true, + "requires": { + "create-error-class": "^3.0.0", + "duplexer3": "^0.1.4", + "get-stream": "^3.0.0", + "is-redirect": "^1.0.0", + "is-retry-allowed": "^1.0.0", + "is-stream": "^1.0.0", + "lowercase-keys": "^1.0.0", + "safe-buffer": "^5.0.1", + "timed-out": "^4.0.0", + "unzip-response": "^2.0.1", + "url-parse-lax": "^1.0.0" + } + }, + "graceful-fs": { + "version": "4.1.15", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.15.tgz", + "integrity": "sha512-6uHUhOPEBgQ24HM+r6b/QwWfZq+yiFcipKFrOFiBEnWdy5sdzYoi+pJeQaPI5qOLRFqWmAXUPQNsielzdLoecA==", + "dev": true + }, + "gtoken": { + "version": "2.3.0", + "resolved": "http://registry.npmjs.org/gtoken/-/gtoken-2.3.0.tgz", + "integrity": "sha512-Jc9/8mV630cZE9FC5tIlJCZNdUjwunvlwOtCz6IDlaiB4Sz68ki29a1+q97sWTnTYroiuF9B135rod9zrQdHLw==", + "dev": true, + "requires": { + "axios": "^0.18.0", + "google-p12-pem": "^1.0.0", + "jws": "^3.1.4", + "mime": "^2.2.0", + "pify": "^3.0.0" + } + }, + "has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dev": true, + "requires": { + "function-bind": "^1.1.1" + } + }, + "has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", + "dev": true + }, + "has-symbol-support-x": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/has-symbol-support-x/-/has-symbol-support-x-1.4.2.tgz", + "integrity": "sha512-3ToOva++HaW+eCpgqZrCfN51IPB+7bJNVT6CUATzueB5Heb8o6Nam0V3HG5dlDvZU1Gn5QLcbahiKw/XVk5JJw==", + "dev": true + }, + "has-symbols": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.0.tgz", + "integrity": "sha1-uhqPGvKg/DllD1yFA2dwQSIGO0Q=", + "dev": true + }, + "has-to-string-tag-x": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/has-to-string-tag-x/-/has-to-string-tag-x-1.4.1.tgz", + "integrity": "sha512-vdbKfmw+3LoOYVr+mtxHaX5a96+0f3DljYd8JOqvOLsf5mw2Otda2qCDT9qRqLAhrjyQ0h7ual5nOiASpsGNFw==", + "dev": true, + "requires": { + "has-symbol-support-x": "^1.4.1" + } + }, + "homedir-polyfill": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/homedir-polyfill/-/homedir-polyfill-1.0.1.tgz", + "integrity": "sha1-TCu8inWJmP7r9e1oWA921GdotLw=", + "dev": true, + "requires": { + "parse-passwd": "^1.0.0" + } + }, + "http-cache-semantics": { + "version": "3.8.1", + "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-3.8.1.tgz", + "integrity": "sha512-5ai2iksyV8ZXmnZhHH4rWPoxxistEexSi5936zIQ1bnNTW5VnA85B6P/VpXiRM017IgRvb2kKo1a//y+0wSp3w==", + "dev": true + }, + "http-errors": { + "version": "1.6.3", + "resolved": "http://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", + "dev": true, + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.0", + "statuses": ">= 1.4.0 < 2" + }, + "dependencies": { + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", + "dev": true + } + } + }, + "http-shutdown": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/http-shutdown/-/http-shutdown-1.2.0.tgz", + "integrity": "sha1-3y2AZ6iFbpnRHp3OshYJWR4d9RQ=", + "dev": true + }, + "https-proxy-agent": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.1.tgz", + "integrity": "sha512-HPCTS1LW51bcyMYbxUIOO4HEOlQ1/1qRaFWcyxvwaqUS9TY88aoEuHUY33kuAh1YhVVaDQhLZsnPd+XNARWZlQ==", + "dev": true, + "requires": { + "agent-base": "^4.1.0", + "debug": "^3.1.0" + } + }, + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "indent-string": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-3.2.0.tgz", + "integrity": "sha1-Sl/W0nzDMvN+VBmlBNu4NxBckok=", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", + "dev": true + }, + "ini": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", + "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", + "dev": true + }, + "inquirer": { + "version": "5.2.0", + "resolved": "http://registry.npmjs.org/inquirer/-/inquirer-5.2.0.tgz", + "integrity": "sha512-E9BmnJbAKLPGonz0HeWHtbKf+EeSP93paWO3ZYoUpq/aowXvYGjjCSuashhXPpzbArIjBbji39THkxTz9ZeEUQ==", + "dev": true, + "requires": { + "ansi-escapes": "^3.0.0", + "chalk": "^2.0.0", + "cli-cursor": "^2.1.0", + "cli-width": "^2.0.0", + "external-editor": "^2.1.0", + "figures": "^2.0.0", + "lodash": "^4.3.0", + "mute-stream": "0.0.7", + "run-async": "^2.2.0", + "rxjs": "^5.5.2", + "string-width": "^2.1.0", + "strip-ansi": "^4.0.0", + "through": "^2.3.6" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, + "inquirer-autocomplete-prompt": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/inquirer-autocomplete-prompt/-/inquirer-autocomplete-prompt-1.0.1.tgz", + "integrity": "sha512-Y4V6ifAu9LNrNjcEtYq8YUKhrgmmufUn5fsDQqeWgHY8rEO6ZAQkNUiZtBm2kw2uUQlC9HdgrRCHDhTPPguH5A==", + "dev": true, + "requires": { + "ansi-escapes": "^3.0.0", + "chalk": "^2.0.0", + "figures": "^2.0.0", + "run-async": "^2.3.0" + } + }, + "into-stream": { + "version": "3.1.0", + "resolved": "http://registry.npmjs.org/into-stream/-/into-stream-3.1.0.tgz", + "integrity": "sha1-lvsKk2wSur1v8XUqF9BWFqvQlMY=", + "dev": true, + "requires": { + "from2": "^2.1.1", + "p-is-promise": "^1.1.0" + } + }, + "ip": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/ip/-/ip-1.1.5.tgz", + "integrity": "sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo=", + "dev": true + }, + "ip-regex": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/ip-regex/-/ip-regex-2.1.0.tgz", + "integrity": "sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk=", + "dev": true + }, + "is-buffer": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", + "dev": true + }, + "is-callable": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.4.tgz", + "integrity": "sha512-r5p9sxJjYnArLjObpjA4xu5EKI3CuKHkJXMhT7kwbpUyIFD1n5PMAsoPvWnvtZiNz7LjkYDRZhd7FlI0eMijEA==", + "dev": true + }, + "is-date-object": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.1.tgz", + "integrity": "sha1-mqIOtq7rv/d/vTPnTKAbM1gdOhY=", + "dev": true + }, + "is-dotfile": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-dotfile/-/is-dotfile-1.0.3.tgz", + "integrity": "sha1-pqLzL/0t+wT1yiXs0Pa4PPeYoeE=", + "dev": true + }, + "is-equal-shallow": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/is-equal-shallow/-/is-equal-shallow-0.1.3.tgz", + "integrity": "sha1-IjgJj8Ih3gvPpdnqxMRdY4qhxTQ=", + "dev": true, + "requires": { + "is-primitive": "^2.0.0" + } + }, + "is-extendable": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", + "dev": true + }, + "is-extglob": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz", + "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "is-glob": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz", + "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=", + "dev": true, + "requires": { + "is-extglob": "^1.0.0" + } + }, + "is-ip": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-ip/-/is-ip-2.0.0.tgz", + "integrity": "sha1-aO6gfooKCpTC0IDdZ0xzGrKkYas=", + "dev": true, + "requires": { + "ip-regex": "^2.0.0" + } + }, + "is-number": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-2.1.0.tgz", + "integrity": "sha1-Afy7s5NGOlSPL0ZszhbezknbkI8=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "is-object": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-object/-/is-object-1.0.1.tgz", + "integrity": "sha1-iVJojF7C/9awPsyF52ngKQMINHA=", + "dev": true + }, + "is-online": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-online/-/is-online-7.0.0.tgz", + "integrity": "sha1-fiQIwK4efje6jVC9sjcmDTK/2W4=", + "dev": true, + "requires": { + "got": "^6.7.1", + "p-any": "^1.0.0", + "p-timeout": "^1.0.0", + "public-ip": "^2.3.0" + } + }, + "is-path-cwd": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz", + "integrity": "sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0=", + "dev": true + }, + "is-path-in-cwd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-1.0.1.tgz", + "integrity": "sha512-FjV1RTW48E7CWM7eE/J2NJvAEEVektecDBVBE5Hh3nM1Jd0kvhHtX68Pr3xsDf857xt3Y4AkwVULK1Vku62aaQ==", + "dev": true, + "requires": { + "is-path-inside": "^1.0.0" + } + }, + "is-path-inside": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz", + "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=", + "dev": true, + "requires": { + "path-is-inside": "^1.0.1" + } + }, + "is-plain-obj": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", + "integrity": "sha1-caUMhCnfync8kqOQpKA7OfzVHT4=", + "dev": true + }, + "is-posix-bracket": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-posix-bracket/-/is-posix-bracket-0.1.1.tgz", + "integrity": "sha1-MzTceXdDaOkvAW5vvAqI9c1ua8Q=", + "dev": true + }, + "is-primitive": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-primitive/-/is-primitive-2.0.0.tgz", + "integrity": "sha1-IHurkWOEmcB7Kt8kCkGochADRXU=", + "dev": true + }, + "is-promise": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", + "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=", + "dev": true + }, + "is-redirect": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-redirect/-/is-redirect-1.0.0.tgz", + "integrity": "sha1-HQPd7VO9jbDzDCbk+V02/HyH3CQ=", + "dev": true + }, + "is-regex": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.4.tgz", + "integrity": "sha1-VRdIm1RwkbCTDglWVM7SXul+lJE=", + "dev": true, + "requires": { + "has": "^1.0.1" + } + }, + "is-retry-allowed": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-retry-allowed/-/is-retry-allowed-1.1.0.tgz", + "integrity": "sha1-EaBgVotnM5REAz0BJaYaINVk+zQ=", + "dev": true + }, + "is-stream": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", + "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", + "dev": true + }, + "is-symbol": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.2.tgz", + "integrity": "sha512-HS8bZ9ox60yCJLH9snBpIwv9pYUAkcuLhSA1oero1UB5y9aiQpRA8y2ex945AOtCZL1lJDeIk3G5LthswI46Lw==", + "dev": true, + "requires": { + "has-symbols": "^1.0.0" + } + }, + "is-utf8": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz", + "integrity": "sha1-Sw2hRCEE0bM2NA6AeX6GXPOffXI=", + "dev": true + }, + "is-windows": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-0.2.0.tgz", + "integrity": "sha1-3hqm1j6indJIc3tp8f+LgALSEIw=", + "dev": true + }, + "is-wsl": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", + "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", + "dev": true + }, + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", + "dev": true + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", + "dev": true + }, + "isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + "dev": true, + "requires": { + "isarray": "1.0.0" + } + }, + "isurl": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isurl/-/isurl-1.0.0.tgz", + "integrity": "sha512-1P/yWsxPlDtn7QeRD+ULKQPaIaN6yF368GZ2vDfv0AL0NwpStafjWCDDdn0k8wgFMWpVAqG7oJhxHnlud42i9w==", + "dev": true, + "requires": { + "has-to-string-tag-x": "^1.2.0", + "is-object": "^1.0.1" + } + }, + "js-tokens": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", + "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=", + "dev": true + }, + "js-yaml": { + "version": "3.12.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.12.0.tgz", + "integrity": "sha512-PIt2cnwmPfL4hKNwqeiuz4bKfnzHTBv6HyVgjahA6mPLwPDzjDWrplJBMjHUFxku/N3FlmrbyPclad+I+4mJ3A==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, + "json-buffer": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", + "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", + "dev": true + }, + "jsonfile": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-3.0.1.tgz", + "integrity": "sha1-pezG9l9T9mLEQVx2daAzHQmS7GY=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6" + } + }, + "jwa": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.1.6.tgz", + "integrity": "sha512-tBO/cf++BUsJkYql/kBbJroKOgHWEigTKBAjjBEmrMGYd1QMBC74Hr4Wo2zCZw6ZrVhlJPvoMrkcOnlWR/DJfw==", + "dev": true, + "requires": { + "buffer-equal-constant-time": "1.0.1", + "ecdsa-sig-formatter": "1.0.10", + "safe-buffer": "^5.0.1" + } + }, + "jws": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.1.5.tgz", + "integrity": "sha512-GsCSexFADNQUr8T5HPJvayTjvPIfoyJPtLQBwn5a4WZQchcrPMPMAWcC1AzJVRDKyD6ZPROPAxgv6rfHViO4uQ==", + "dev": true, + "requires": { + "jwa": "^1.1.5", + "safe-buffer": "^5.0.1" + } + }, + "keyv": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.0.0.tgz", + "integrity": "sha512-eguHnq22OE3uVoSYG0LVWNP+4ppamWr9+zWBe1bsNcovIMy6huUJFPgy4mGwCd/rnl3vOLGW1MTlu4c57CT1xA==", + "dev": true, + "requires": { + "json-buffer": "3.0.0" + } + }, + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + }, + "lazy-cache": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-2.0.2.tgz", + "integrity": "sha1-uRkKT5EzVGlIQIWfio9whNiCImQ=", + "dev": true, + "requires": { + "set-getter": "^0.1.0" + } + }, + "lodash": { + "version": "4.17.11", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz", + "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==", + "dev": true + }, + "lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=", + "dev": true + }, + "lowercase-keys": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", + "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", + "dev": true + }, + "lru-cache": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.5.tgz", + "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "dev": true, + "requires": { + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" + } + }, + "math-random": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/math-random/-/math-random-1.0.1.tgz", + "integrity": "sha1-izqsWIuKZuSXXjzepn97sylgH6w=", + "dev": true + }, + "merge": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/merge/-/merge-1.2.1.tgz", + "integrity": "sha512-VjFo4P5Whtj4vsLzsYBu5ayHhoHJ0UqNm7ibvShmbmoz7tGi0vXaoJbGdB+GmDMLUdg8DpQXEIeVDAe8MaABvQ==", + "dev": true + }, + "micromatch": { + "version": "2.3.11", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-2.3.11.tgz", + "integrity": "sha1-hmd8l9FyCzY0MdBNDRUpO9OMFWU=", + "dev": true, + "requires": { + "arr-diff": "^2.0.0", + "array-unique": "^0.2.1", + "braces": "^1.8.2", + "expand-brackets": "^0.1.4", + "extglob": "^0.3.1", + "filename-regex": "^2.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.1", + "kind-of": "^3.0.2", + "normalize-path": "^2.0.1", + "object.omit": "^2.0.0", + "parse-glob": "^3.0.4", + "regex-cache": "^0.4.2" + } + }, + "mime": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.0.tgz", + "integrity": "sha512-ikBcWwyqXQSHKtciCcctu9YfPbFYZ4+gbHEmE0Q8jzcTYQg5dHCr3g2wwAZjPoJfQVXZq6KXAjpXOTf5/cjT7w==", + "dev": true + }, + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true + }, + "mimic-response": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", + "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", + "dev": true + }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "minimist": { + "version": "0.0.8", + "resolved": "http://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", + "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", + "dev": true + }, + "mkdirp": { + "version": "0.5.1", + "resolved": "http://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", + "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", + "dev": true, + "requires": { + "minimist": "0.0.8" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "multimatch": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/multimatch/-/multimatch-2.1.0.tgz", + "integrity": "sha1-nHkGoi+0wCkZ4vX3UWG0zb1LKis=", + "dev": true, + "requires": { + "array-differ": "^1.0.0", + "array-union": "^1.0.1", + "arrify": "^1.0.0", + "minimatch": "^3.0.0" + } + }, + "mute-stream": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", + "dev": true + }, + "node-forge": { + "version": "0.7.6", + "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-0.7.6.tgz", + "integrity": "sha512-sol30LUpz1jQFBjOKwbjxijiE3b6pjd74YwfD0fJOKPjF+fONKb2Yg8rYgS6+bK6VDl+/wfr4IYpC7jDzLUIfw==", + "dev": true + }, + "normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", + "dev": true, + "requires": { + "remove-trailing-separator": "^1.0.1" + } + }, + "normalize-url": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-2.0.1.tgz", + "integrity": "sha512-D6MUW4K/VzoJ4rJ01JFKxDrtY1v9wrgzCX5f2qj/lzH1m/lW6MhUZFKerVsnyjOhOsYzI9Kqqak+10l4LvLpMw==", + "dev": true, + "requires": { + "prepend-http": "^2.0.0", + "query-string": "^5.0.1", + "sort-keys": "^2.0.0" + }, + "dependencies": { + "prepend-http": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", + "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", + "dev": true + } + } + }, + "object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", + "dev": true + }, + "object-keys": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.0.12.tgz", + "integrity": "sha512-FTMyFUm2wBcGHnH2eXmz7tC6IwlqQZ6mVZ+6dm6vZ4IQIHjs6FdNsQBuKGPuUUUY6NfJw2PshC08Tn6LzLDOag==", + "dev": true + }, + "object.omit": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/object.omit/-/object.omit-2.0.1.tgz", + "integrity": "sha1-Gpx0SCnznbuFjHbKNXmuKlTr0fo=", + "dev": true, + "requires": { + "for-own": "^0.1.4", + "is-extendable": "^0.1.1" + } + }, + "on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", + "dev": true, + "requires": { + "ee-first": "1.1.1" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, + "requires": { + "mimic-fn": "^1.0.0" + } + }, + "opn": { + "version": "5.4.0", + "resolved": "https://registry.npmjs.org/opn/-/opn-5.4.0.tgz", + "integrity": "sha512-YF9MNdVy/0qvJvDtunAOzFw9iasOQHpVthTCvGzxt61Il64AYSGdK+rYwld7NAfk9qJ7dt+hymBNSc9LNYS+Sw==", + "dev": true, + "requires": { + "is-wsl": "^1.1.0" + } + }, + "os-homedir": { + "version": "1.0.2", + "resolved": "http://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", + "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=", + "dev": true + }, + "os-tmpdir": { + "version": "1.0.2", + "resolved": "http://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", + "dev": true + }, + "p-any": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/p-any/-/p-any-1.1.0.tgz", + "integrity": "sha512-Ef0tVa4CZ5pTAmKn+Cg3w8ABBXh+hHO1aV8281dKOoUHfX+3tjG2EaFcC+aZyagg9b4EYGsHEjz21DnEE8Og2g==", + "dev": true, + "requires": { + "p-some": "^2.0.0" + } + }, + "p-cancelable": { + "version": "0.4.1", + "resolved": "http://registry.npmjs.org/p-cancelable/-/p-cancelable-0.4.1.tgz", + "integrity": "sha512-HNa1A8LvB1kie7cERyy21VNeHb2CWJJYqyyC2o3klWFfMGlFmWv2Z7sFgZH8ZiaYL95ydToKTFVXgMV/Os0bBQ==", + "dev": true + }, + "p-finally": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", + "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", + "dev": true + }, + "p-is-promise": { + "version": "1.1.0", + "resolved": "http://registry.npmjs.org/p-is-promise/-/p-is-promise-1.1.0.tgz", + "integrity": "sha1-nJRWmJ6fZYgBewQ01WCXZ1w9oF4=", + "dev": true + }, + "p-map": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-1.2.0.tgz", + "integrity": "sha512-r6zKACMNhjPJMTl8KcFH4li//gkrXWfbD6feV8l6doRHlzljFWGJ2AP6iKaCJXyZmAUMOPtvbW7EXkbWO/pLEA==", + "dev": true + }, + "p-some": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/p-some/-/p-some-2.0.1.tgz", + "integrity": "sha1-Zdh8ixVO289SIdFnd4ttLhUPbwY=", + "dev": true, + "requires": { + "aggregate-error": "^1.0.0" + } + }, + "p-timeout": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/p-timeout/-/p-timeout-1.2.1.tgz", + "integrity": "sha1-XrOzU7f86Z8QGhA4iAuwVOu+o4Y=", + "dev": true, + "requires": { + "p-finally": "^1.0.0" + } + }, + "parse-glob": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/parse-glob/-/parse-glob-3.0.4.tgz", + "integrity": "sha1-ssN2z7EfNVE7rdFz7wu246OIORw=", + "dev": true, + "requires": { + "glob-base": "^0.3.0", + "is-dotfile": "^1.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.0" + } + }, + "parse-passwd": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/parse-passwd/-/parse-passwd-1.0.0.tgz", + "integrity": "sha1-bVuTSkVpk7I9N/QKOC1vFmao5cY=", + "dev": true + }, + "parseurl": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz", + "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M=", + "dev": true + }, + "path": { + "version": "0.12.7", + "resolved": "https://registry.npmjs.org/path/-/path-0.12.7.tgz", + "integrity": "sha1-1NwqUGxM4hl+tIHr/NWzbAFAsQ8=", + "dev": true, + "requires": { + "process": "^0.11.1", + "util": "^0.10.3" + } + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "http://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", + "dev": true + }, + "path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", + "dev": true + }, + "path-parse": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", + "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", + "dev": true + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "pinkie": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", + "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", + "dev": true + }, + "pinkie-promise": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", + "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", + "dev": true, + "requires": { + "pinkie": "^2.0.0" + } + }, + "pluralize": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-7.0.0.tgz", + "integrity": "sha512-ARhBOdzS3e41FbkW/XWrTEtukqqLoK5+Z/4UeDaLuSW+39JPeFgs4gCGqsrJHVZX0fUrx//4OF0K1CUGwlIFow==", + "dev": true + }, + "prepend-http": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", + "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=", + "dev": true + }, + "preserve": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/preserve/-/preserve-0.2.0.tgz", + "integrity": "sha1-gV7R9uvGWSb4ZbMQwHE7yzMVzks=", + "dev": true + }, + "process": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", + "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=", + "dev": true + }, + "process-nextick-args": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.0.tgz", + "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw==", + "dev": true + }, + "pseudomap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", + "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=", + "dev": true + }, + "public-ip": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/public-ip/-/public-ip-2.4.0.tgz", + "integrity": "sha512-74cIy+T2cDmt+Z71AfVipH2q6qqZITPyNGszKV86OGDYIRvti1m8zg4GOaiTPCLgEIWnToKYXbhEnMiZWHPEUA==", + "dev": true, + "requires": { + "dns-socket": "^1.6.2", + "got": "^8.0.0", + "is-ip": "^2.0.0", + "pify": "^3.0.0" + }, + "dependencies": { + "got": { + "version": "8.3.2", + "resolved": "https://registry.npmjs.org/got/-/got-8.3.2.tgz", + "integrity": "sha512-qjUJ5U/hawxosMryILofZCkm3C84PLJS/0grRIpjAwu+Lkxxj5cxeCU25BG0/3mDSpXKTyZr8oh8wIgLaH0QCw==", + "dev": true, + "requires": { + "@sindresorhus/is": "^0.7.0", + "cacheable-request": "^2.1.1", + "decompress-response": "^3.3.0", + "duplexer3": "^0.1.4", + "get-stream": "^3.0.0", + "into-stream": "^3.1.0", + "is-retry-allowed": "^1.1.0", + "isurl": "^1.0.0-alpha5", + "lowercase-keys": "^1.0.0", + "mimic-response": "^1.0.0", + "p-cancelable": "^0.4.0", + "p-timeout": "^2.0.1", + "pify": "^3.0.0", + "safe-buffer": "^5.1.1", + "timed-out": "^4.0.1", + "url-parse-lax": "^3.0.0", + "url-to-options": "^1.0.1" + } + }, + "p-timeout": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/p-timeout/-/p-timeout-2.0.1.tgz", + "integrity": "sha512-88em58dDVB/KzPEx1X0N3LwFfYZPyDc4B6eF38M1rk9VTZMbxXXgjugz8mmwpS9Ox4BDZ+t6t3QP5+/gazweIA==", + "dev": true, + "requires": { + "p-finally": "^1.0.0" + } + }, + "prepend-http": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", + "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", + "dev": true + }, + "url-parse-lax": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", + "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", + "dev": true, + "requires": { + "prepend-http": "^2.0.0" + } + } + } + }, + "punycode": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", + "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=", + "dev": true + }, + "qs": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.6.0.tgz", + "integrity": "sha512-KIJqT9jQJDQx5h5uAVPimw6yVg2SekOKu959OCtktD3FjzbpvaPr8i4zzg07DOMz+igA4W/aNM7OV8H37pFYfA==", + "dev": true + }, + "query-string": { + "version": "5.1.1", + "resolved": "http://registry.npmjs.org/query-string/-/query-string-5.1.1.tgz", + "integrity": "sha512-gjWOsm2SoGlgLEdAGt7a6slVOk9mGiXmPFMqrEhLQ68rhQuBnpfs3+EmlvqKyxnCo9/PPlF+9MtY02S1aFg+Jw==", + "dev": true, + "requires": { + "decode-uri-component": "^0.2.0", + "object-assign": "^4.1.0", + "strict-uri-encode": "^1.0.0" + } + }, + "querystring": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", + "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=", + "dev": true + }, + "randomatic": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/randomatic/-/randomatic-3.1.1.tgz", + "integrity": "sha512-TuDE5KxZ0J461RVjrJZCJc+J+zCkTb1MbH9AQUq68sMhOMcy9jLcb3BrZKgp9q9Ncltdg4QVqWrH02W2EFFVYw==", + "dev": true, + "requires": { + "is-number": "^4.0.0", + "kind-of": "^6.0.0", + "math-random": "^1.0.1" + }, + "dependencies": { + "is-number": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-4.0.0.tgz", + "integrity": "sha512-rSklcAIlf1OmFdyAqbnWTLVelsQ58uvZ66S/ZyawjWqIviTWCjg2PzVGw8WUA+nNuPTqb4wgA+NszrJ+08LlgQ==", + "dev": true + }, + "kind-of": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==", + "dev": true + } + } + }, + "read-file": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/read-file/-/read-file-0.2.0.tgz", + "integrity": "sha1-cMa6+IQux9FUD5gf0Oau1Mgb1UU=", + "dev": true + }, + "read-multiple-files": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/read-multiple-files/-/read-multiple-files-1.1.1.tgz", + "integrity": "sha1-TA55+hp90d05w+j2yZ4ZxmcC5n4=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "run-parallel": "^1.1.2", + "strip-bom": "^2.0.0" + } + }, + "readable-stream": { + "version": "2.3.6", + "resolved": "http://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", + "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "readline": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/readline/-/readline-1.3.0.tgz", + "integrity": "sha1-xYDXfvLPyHUrEySYBg3JeTp6wBw=", + "dev": true + }, + "recursive-readdir": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/recursive-readdir/-/recursive-readdir-2.2.2.tgz", + "integrity": "sha512-nRCcW9Sj7NuZwa2XvH9co8NPeXUBhZP7CRKJtU+cS6PW9FpCIFoI5ib0NT1ZrbNuPoRy0ylyCaUL8Gih4LSyFg==", + "dev": true, + "requires": { + "minimatch": "3.0.4" + } + }, + "regex-cache": { + "version": "0.4.4", + "resolved": "https://registry.npmjs.org/regex-cache/-/regex-cache-0.4.4.tgz", + "integrity": "sha512-nVIZwtCjkC9YgvWkpM55B5rBhBYRZhAaJbgcFYXXsHnbZ9UZI9nnVWYZpBlCqv9ho2eZryPnWrZGsOdPwVWXWQ==", + "dev": true, + "requires": { + "is-equal-shallow": "^0.1.3" + } + }, + "remove-trailing-separator": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", + "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", + "dev": true + }, + "repeat-element": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", + "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", + "dev": true + }, + "repeat-string": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", + "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", + "dev": true + }, + "resolve": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.8.1.tgz", + "integrity": "sha512-AicPrAC7Qu1JxPCZ9ZgCZlY35QgFnNqc+0LtbRNxnVw4TXvjQ72wnuL9JQcEBgXkI9JM8MsT9kaQoHcpCRJOYA==", + "dev": true, + "requires": { + "path-parse": "^1.0.5" + } + }, + "resolve-dir": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/resolve-dir/-/resolve-dir-0.1.1.tgz", + "integrity": "sha1-shklmlYC+sXFxJatiUpujMQwJh4=", + "dev": true, + "requires": { + "expand-tilde": "^1.2.2", + "global-modules": "^0.2.3" + }, + "dependencies": { + "expand-tilde": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/expand-tilde/-/expand-tilde-1.2.2.tgz", + "integrity": "sha1-C4HrqJflo9MdHD0QL48BRB5VlEk=", + "dev": true, + "requires": { + "os-homedir": "^1.0.1" + } + } + } + }, + "resolve-file": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/resolve-file/-/resolve-file-0.3.0.tgz", + "integrity": "sha1-EeH7RkVm06fFAMt+lIHo8LAKFO8=", + "dev": true, + "requires": { + "cwd": "^0.10.0", + "expand-tilde": "^2.0.2", + "extend-shallow": "^2.0.1", + "fs-exists-sync": "^0.1.0", + "homedir-polyfill": "^1.0.1", + "lazy-cache": "^2.0.2", + "resolve": "^1.2.0" + } + }, + "resolve-path": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/resolve-path/-/resolve-path-1.4.0.tgz", + "integrity": "sha1-xL2p9e+y/OZSR4c6s2u02DT+Fvc=", + "dev": true, + "requires": { + "http-errors": "~1.6.2", + "path-is-absolute": "1.0.1" + } + }, + "responselike": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", + "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", + "dev": true, + "requires": { + "lowercase-keys": "^1.0.0" + } + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, + "requires": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + } + }, + "retry-axios": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/retry-axios/-/retry-axios-0.3.2.tgz", + "integrity": "sha512-jp4YlI0qyDFfXiXGhkCOliBN1G7fRH03Nqy8YdShzGqbY5/9S2x/IR6C88ls2DFkbWuL3ASkP7QD3pVrNpPgwQ==", + "dev": true + }, + "rimraf": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.2.tgz", + "integrity": "sha512-lreewLK/BlghmxtfH36YYVg1i8IAce4TI7oao75I1g245+6BctqTVQiBP3YUJ9C6DQOXJmkYR9X9fCLtCOJc5w==", + "dev": true, + "requires": { + "glob": "^7.0.5" + } + }, + "run-async": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.3.0.tgz", + "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=", + "dev": true, + "requires": { + "is-promise": "^2.1.0" + } + }, + "run-parallel": { + "version": "1.1.9", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.1.9.tgz", + "integrity": "sha512-DEqnSRTDw/Tc3FXf49zedI638Z9onwUotBMiUFKmrO2sdFKIbXamXGQ3Axd4qgphxKB4kw/qP1w5kTxnfU1B9Q==", + "dev": true + }, + "rxjs": { + "version": "5.5.12", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-5.5.12.tgz", + "integrity": "sha512-xx2itnL5sBbqeeiVgNPVuQQ1nC8Jp2WfNJhXWHmElW9YmrpS9UVnNzhP3EH3HFqexO5Tlp8GhYY+WEcqcVMvGw==", + "dev": true, + "requires": { + "symbol-observable": "1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "dev": true + }, + "semver": { + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.6.0.tgz", + "integrity": "sha512-RS9R6R35NYgQn++fkDWaOmqGoj4Ek9gGs+DPxNUZKuwE183xjJroKvyo1IzVFeXvUrvmALy6FWD5xrdJT25gMg==", + "dev": true + }, + "set-getter": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/set-getter/-/set-getter-0.1.0.tgz", + "integrity": "sha1-12nBgsnVpR9AkUXy+6guXoboA3Y=", + "dev": true, + "requires": { + "to-object-path": "^0.3.0" + } + }, + "setprototypeof": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", + "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==", + "dev": true + }, + "signal-exit": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", + "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", + "dev": true + }, + "sort-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/sort-keys/-/sort-keys-2.0.0.tgz", + "integrity": "sha1-ZYU1WEhh7JfXMNbPQYIuH1ZoQSg=", + "dev": true, + "requires": { + "is-plain-obj": "^1.0.0" + } + }, + "split-lines": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/split-lines/-/split-lines-1.1.0.tgz", + "integrity": "sha1-Oruo9ZhhQUL5240nq2q4dWYqHgk=", + "dev": true + }, + "sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", + "dev": true + }, + "statuses": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.3.1.tgz", + "integrity": "sha1-+vUbnrdKrvOzrPStX2Gr8ky3uT4=", + "dev": true + }, + "strict-uri-encode": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz", + "integrity": "sha1-J5siXfHVgrH1TmWt3UNS4Y+qBxM=", + "dev": true + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, + "string.prototype.padend": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/string.prototype.padend/-/string.prototype.padend-3.0.0.tgz", + "integrity": "sha1-86rvfBcZ8XDF6rHDK/eA2W4h8vA=", + "dev": true, + "requires": { + "define-properties": "^1.1.2", + "es-abstract": "^1.4.3", + "function-bind": "^1.0.2" + } + }, + "string_decoder": { + "version": "1.1.1", + "resolved": "http://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dev": true, + "requires": { + "safe-buffer": "~5.1.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "http://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "strip-bom": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz", + "integrity": "sha1-YhmoVhZSBJHzV4i9vxRHqZx+aw4=", + "dev": true, + "requires": { + "is-utf8": "^0.2.0" + } + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + }, + "symbol-observable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.0.1.tgz", + "integrity": "sha1-g0D8RwLDEi310iKI+IKD9RPT/dQ=", + "dev": true + }, + "through": { + "version": "2.3.8", + "resolved": "http://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", + "dev": true + }, + "timed-out": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz", + "integrity": "sha1-8y6srFoXW+ol1/q1Zas+2HQe9W8=", + "dev": true + }, + "tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "requires": { + "os-tmpdir": "~1.0.2" + } + }, + "to-object-path": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", + "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "ts2gas": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/ts2gas/-/ts2gas-1.3.0.tgz", + "integrity": "sha512-zp+oY1fc/i9ZM7n1MhyXZ7VlRH/oTSS6YArW5ebk8oznZstRWLNdAfm1VAlkiOt/Dpey1bOfHFGV3ACa86OpRw==", + "dev": true, + "requires": { + "typescript": "^2.9.2" + }, + "dependencies": { + "typescript": { + "version": "2.9.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-2.9.2.tgz", + "integrity": "sha512-Gr4p6nFNaoufRIY4NMdpQRNmgxVIGMs4Fcu/ujdYk3nAZqk7supzBE9idmvfZIlH/Cuj//dvi+019qEue9lV0w==", + "dev": true + } + } + }, + "tslib": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.9.3.tgz", + "integrity": "sha512-4krF8scpejhaOgqzBEcGM7yDIEfi0/8+8zDRZhNZZ2kjmHJ4hv3zCbQWxoJGz1iw5U0Jl0nma13xzHXcncMavQ==", + "dev": true + }, + "tslint": { + "version": "5.11.0", + "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.11.0.tgz", + "integrity": "sha1-mPMMAurjzecAYgHkwzywi0hYHu0=", + "dev": true, + "requires": { + "babel-code-frame": "^6.22.0", + "builtin-modules": "^1.1.1", + "chalk": "^2.3.0", + "commander": "^2.12.1", + "diff": "^3.2.0", + "glob": "^7.1.1", + "js-yaml": "^3.7.0", + "minimatch": "^3.0.4", + "resolve": "^1.3.2", + "semver": "^5.3.0", + "tslib": "^1.8.0", + "tsutils": "^2.27.2" + } + }, + "tsutils": { + "version": "2.29.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", + "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", + "dev": true, + "requires": { + "tslib": "^1.8.1" + } + }, + "typescript": { + "version": "3.3.0-dev.20181205", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.3.0-dev.20181205.tgz", + "integrity": "sha512-XYoXPbSZrlKZOA5ykKjzwvGjAtwEhII/hBFXoGzbxfIWaT5uPcg3dISdE8qZ7jb6Xg1vjV2ghtTevToE2+IvCA==", + "dev": true + }, + "ucfirst": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/ucfirst/-/ucfirst-1.0.0.tgz", + "integrity": "sha1-ThBbZEjQXiZOzsQ14LkZNjxfLy8=", + "dev": true + }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", + "dev": true + }, + "unzip-response": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/unzip-response/-/unzip-response-2.0.1.tgz", + "integrity": "sha1-0vD3N9FrBhXnKmk17QQhRXLVb5c=", + "dev": true + }, + "url": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", + "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", + "dev": true, + "requires": { + "punycode": "1.3.2", + "querystring": "0.2.0" + } + }, + "url-parse-lax": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-1.0.0.tgz", + "integrity": "sha1-evjzA2Rem9eaJy56FKxovAYJ2nM=", + "dev": true, + "requires": { + "prepend-http": "^1.0.1" + } + }, + "url-template": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/url-template/-/url-template-2.0.8.tgz", + "integrity": "sha1-/FZaPMy/93MMd19WQflVV5FDnyE=", + "dev": true + }, + "url-to-options": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/url-to-options/-/url-to-options-1.0.1.tgz", + "integrity": "sha1-FQWgOiiaSMvXpDTvuu7FBV9WM6k=", + "dev": true + }, + "util": { + "version": "0.10.4", + "resolved": "https://registry.npmjs.org/util/-/util-0.10.4.tgz", + "integrity": "sha512-0Pm9hTQ3se5ll1XihRic3FDIku70C+iHUdT/W926rSgHV5QgXsYbKZN8MSC3tJtSkhuROzvsQjAaFENRXr+19A==", + "dev": true, + "requires": { + "inherits": "2.0.3" + } + }, + "util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", + "dev": true + }, + "utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=", + "dev": true + }, + "uuid": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.2.tgz", + "integrity": "sha512-yXJmeNaw3DnnKAOKJE51sL/ZaYfWJRl1pK9dr19YFCu0ObS231AB1/LbqTKRAQ5kw8A90rA6fr4riOUpTZvQZA==", + "dev": true + }, + "watch": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/watch/-/watch-1.0.2.tgz", + "integrity": "sha1-NApxe952Vyb6CqB9ch4BR6VR3ww=", + "dev": true, + "requires": { + "exec-sh": "^0.2.0", + "minimist": "^1.2.0" + }, + "dependencies": { + "minimist": { + "version": "1.2.0", + "resolved": "http://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", + "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", + "dev": true + } + } + }, + "which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", "dev": true + }, + "yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=", + "dev": true } } } diff --git a/package.json b/package.json index bf8dae5..a44fe98 100644 --- a/package.json +++ b/package.json @@ -1,12 +1,13 @@ { - "name": "betterTrello", + "name": "TrelloTools", "version": "1.0.0", "description": "Trello's thing didn't have email link attachments so I did this...", - "repository": "https://github.com/mrk3767/betterTrello", + "repository": "https://github.com/mrk3767/TrelloTools", "license": "MIT", "dependencies": {}, "devDependencies": { - "clasp": "^0.1.0", + "@google/clasp": "^1.7.0", + "@types/google-apps-script": "0.0.32", "tslint": "^5.11.0", "typescript": "^3.3.0-dev.20181205" } diff --git a/readme.md b/readme.md index 1a06df7..42e0811 100644 --- a/readme.md +++ b/readme.md @@ -1 +1,28 @@ -### Setup \ No newline at end of file +### Setup for development + +1. Clone repo +1. `npm install` +1. `npx clasp login` +1. Navigate to the [G Suite Developer Hub](https://script.google.com/home/usersettings) +1. Enable the `Google Apps Script API` +1. `npx clasp create` +1. Select `standalone` +1. `npm clasp push` +1. Get your key and secret (at the bottom of the page) from [Trello](https://trello.com/app-key) +1. `npx clasp open` +1. Navigate to `File`->`Project Properties`->`Script Properties` +1. Add two rows + 1. `trello.key`:`` + 1. `trello.secret`:`` +1. Save +1. Navigate to `Publish`->`Deploy from manifest...` +1. Click `Get ID` +1. Copy the deployment ID +1. Open Gmail +1. Navigate to `Settings`->`Add-ons` +1. Check `Enable developer add-ons for my account` +1. Enter deployment ID +1. Click install + +### Stable deployments +* `AKfycbzVf5oSzTU0uKvWv3Az5LV5E2NTAj1Z-waBEeSE_AIGZXG1yj1kvU9B5EFz3Em9GRUjnA` \ No newline at end of file diff --git a/src/Ui/StateManager.ts b/src/Ui/StateManager.ts index 9376667..24196cc 100644 --- a/src/Ui/StateManager.ts +++ b/src/Ui/StateManager.ts @@ -1,6 +1,6 @@ const userProperties = PropertiesService.getUserProperties(); -class StateManager { +export class StateManager { public static stateManager public static getStateManager() { diff --git a/src/googleAppsScriptEntryPoints.ts b/src/googleAppsScriptEntryPoints.ts index bd8493b..c78820f 100644 --- a/src/googleAppsScriptEntryPoints.ts +++ b/src/googleAppsScriptEntryPoints.ts @@ -1,7 +1,8 @@ import {Trello} from "./Api/Trello/Trello"; import {MainMenuUi} from "./Ui/MainMenuUi"; +import {StateManager} from "./Ui/StateManager" -const APP_NAME = `quickLink` +const APP_NAME = `TrelloTools` function handleUiUpdate(e) { // TODO: Create a whitelist for classes/methods that can be called From 10f2c878ee81b047953d80c0a0e0a44b61d1c2bb Mon Sep 17 00:00:00 2001 From: Mike K Date: Sat, 8 Dec 2018 15:49:34 -0800 Subject: [PATCH 4/5] Update readme.md --- readme.md | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/readme.md b/readme.md index 42e0811..5fdc695 100644 --- a/readme.md +++ b/readme.md @@ -1,3 +1,10 @@ +### Install add-on in Gmail +1. Open Gmail +1. Navigate to `Settings`->`Add-ons` +1. Check `Enable developer add-ons for my account` +1. Enter deployment ID (obtain from dev setup or stable deployments below) +1. Click install + ### Setup for development 1. Clone repo @@ -18,11 +25,6 @@ 1. Navigate to `Publish`->`Deploy from manifest...` 1. Click `Get ID` 1. Copy the deployment ID -1. Open Gmail -1. Navigate to `Settings`->`Add-ons` -1. Check `Enable developer add-ons for my account` -1. Enter deployment ID -1. Click install ### Stable deployments -* `AKfycbzVf5oSzTU0uKvWv3Az5LV5E2NTAj1Z-waBEeSE_AIGZXG1yj1kvU9B5EFz3Em9GRUjnA` \ No newline at end of file +* `AKfycbzVf5oSzTU0uKvWv3Az5LV5E2NTAj1Z-waBEeSE_AIGZXG1yj1kvU9B5EFz3Em9GRUjnA` From 6762bf06f5b71b6aaa16327adefa9a9640fc4577 Mon Sep 17 00:00:00 2001 From: Mike K Date: Sat, 8 Dec 2018 15:52:23 -0800 Subject: [PATCH 5/5] Update readme.md --- readme.md | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/readme.md b/readme.md index 5fdc695..c4e7c0d 100644 --- a/readme.md +++ b/readme.md @@ -28,3 +28,9 @@ ### Stable deployments * `AKfycbzVf5oSzTU0uKvWv3Az5LV5E2NTAj1Z-waBEeSE_AIGZXG1yj1kvU9B5EFz3Em9GRUjnA` + +### Thanks +* [Google Apps Script](https://developers.google.com/apps-script/overview) +* [Trello API](https://developers.trello.com) +* [Google Clasp](https://github.com/google/clasp) +* [Microsoft TypeScript](https://github.com/Microsoft/TypeScript)