├── .gitignore ├── .npmignore ├── LICENSE ├── README.md ├── package.json ├── src ├── data │ ├── cloud_firestore_clients.ts │ ├── cloud_firestore_data_handler.ts │ ├── cloud_firestore_data_handler_factory.ts │ ├── cloud_firestore_scopes.ts │ └── index.ts ├── endpoint │ ├── authentication.ts │ ├── authorize.ts │ ├── index.ts │ ├── protectedresources │ │ ├── abstract_protected_resource_endpoint.ts │ │ ├── index.ts │ │ └── userinfo.ts │ ├── token.ts │ ├── tokeninfo.ts │ └── views │ │ ├── consent_view_template.ts │ │ └── default_consent_view_template.ts ├── granttype │ ├── custom_grant_handler_provider.ts │ └── index.ts ├── index.ts ├── models │ ├── index.ts │ ├── request_map.ts │ └── request_wrapper.ts └── utils │ ├── configuration.ts │ ├── crypto.ts │ ├── index.ts │ └── navigation.ts ├── tsconfig.json ├── tslint.json ├── views ├── authentication.ejs └── consent.ejs └── yarn.lock /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | dist 3 | -------------------------------------------------------------------------------- /.npmignore: -------------------------------------------------------------------------------- 1 | _test 2 | tslint.json 3 | tsconfig.json 4 | *.log 5 | *.tgz 6 | *.DS_Store 7 | .idea 8 | *.iml 9 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Apache License 2 | Version 2.0, January 2004 3 | http://www.apache.org/licenses/ 4 | 5 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 6 | 7 | 1. Definitions. 8 | 9 | "License" shall mean the terms and conditions for use, reproduction, and 10 | distribution as defined by Sections 1 through 9 of this document. 11 | 12 | "Licensor" shall mean the copyright owner or entity authorized by the copyright 13 | owner that is granting the License. 14 | 15 | "Legal Entity" shall mean the union of the acting entity and all other entities 16 | that control, are controlled by, or are under common control with that entity. 17 | For the purposes of this definition, "control" means (i) the power, direct or 18 | indirect, to cause the direction or management of such entity, whether by 19 | contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the 20 | outstanding shares, or (iii) beneficial ownership of such entity. 21 | 22 | "You" (or "Your") shall mean an individual or Legal Entity exercising 23 | permissions granted by this License. 24 | 25 | "Source" form shall mean the preferred form for making modifications, including 26 | but not limited to software source code, documentation source, and configuration 27 | files. 28 | 29 | "Object" form shall mean any form resulting from mechanical transformation or 30 | translation of a Source form, including but not limited to compiled object code, 31 | generated documentation, and conversions to other media types. 32 | 33 | "Work" shall mean the work of authorship, whether in Source or Object form, made 34 | available under the License, as indicated by a copyright notice that is included 35 | in or attached to the work (an example is provided in the Appendix below). 36 | 37 | "Derivative Works" shall mean any work, whether in Source or Object form, that 38 | is based on (or derived from) the Work and for which the editorial revisions, 39 | annotations, elaborations, or other modifications represent, as a whole, an 40 | original work of authorship. For the purposes of this License, Derivative Works 41 | shall not include works that remain separable from, or merely link (or bind by 42 | name) to the interfaces of, the Work and Derivative Works thereof. 43 | 44 | "Contribution" shall mean any work of authorship, including the original version 45 | of the Work and any modifications or additions to that Work or Derivative Works 46 | thereof, that is intentionally submitted to Licensor for inclusion in the Work 47 | by the copyright owner or by an individual or Legal Entity authorized to submit 48 | on behalf of the copyright owner. For the purposes of this definition, 49 | "submitted" means any form of electronic, verbal, or written communication sent 50 | to the Licensor or its representatives, including but not limited to 51 | communication on electronic mailing lists, source code control systems, and 52 | issue tracking systems that are managed by, or on behalf of, the Licensor for 53 | the purpose of discussing and improving the Work, but excluding communication 54 | that is conspicuously marked or otherwise designated in writing by the copyright 55 | owner as "Not a Contribution." 56 | 57 | "Contributor" shall mean Licensor and any individual or Legal Entity on behalf 58 | of whom a Contribution has been received by Licensor and subsequently 59 | incorporated within the Work. 60 | 61 | 2. Grant of Copyright License. 62 | 63 | Subject to the terms and conditions of this License, each Contributor hereby 64 | grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, 65 | irrevocable copyright license to reproduce, prepare Derivative Works of, 66 | publicly display, publicly perform, sublicense, and distribute the Work and such 67 | Derivative Works in Source or Object form. 68 | 69 | 3. Grant of Patent License. 70 | 71 | Subject to the terms and conditions of this License, each Contributor hereby 72 | grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, 73 | irrevocable (except as stated in this section) patent license to make, have 74 | made, use, offer to sell, sell, import, and otherwise transfer the Work, where 75 | such license applies only to those patent claims licensable by such Contributor 76 | that are necessarily infringed by their Contribution(s) alone or by combination 77 | of their Contribution(s) with the Work to which such Contribution(s) was 78 | submitted. If You institute patent litigation against any entity (including a 79 | cross-claim or counterclaim in a lawsuit) alleging that the Work or a 80 | Contribution incorporated within the Work constitutes direct or contributory 81 | patent infringement, then any patent licenses granted to You under this License 82 | for that Work shall terminate as of the date such litigation is filed. 83 | 84 | 4. Redistribution. 85 | 86 | You may reproduce and distribute copies of the Work or Derivative Works thereof 87 | in any medium, with or without modifications, and in Source or Object form, 88 | provided that You meet the following conditions: 89 | 90 | You must give any other recipients of the Work or Derivative Works a copy of 91 | this License; and 92 | You must cause any modified files to carry prominent notices stating that You 93 | changed the files; and 94 | You must retain, in the Source form of any Derivative Works that You distribute, 95 | all copyright, patent, trademark, and attribution notices from the Source form 96 | of the Work, excluding those notices that do not pertain to any part of the 97 | Derivative Works; and 98 | If the Work includes a "NOTICE" text file as part of its distribution, then any 99 | Derivative Works that You distribute must include a readable copy of the 100 | attribution notices contained within such NOTICE file, excluding those notices 101 | that do not pertain to any part of the Derivative Works, in at least one of the 102 | following places: within a NOTICE text file distributed as part of the 103 | Derivative Works; within the Source form or documentation, if provided along 104 | with the Derivative Works; or, within a display generated by the Derivative 105 | Works, if and wherever such third-party notices normally appear. The contents of 106 | the NOTICE file are for informational purposes only and do not modify the 107 | License. You may add Your own attribution notices within Derivative Works that 108 | You distribute, alongside or as an addendum to the NOTICE text from the Work, 109 | provided that such additional attribution notices cannot be construed as 110 | modifying the License. 111 | You may add Your own copyright statement to Your modifications and may provide 112 | additional or different license terms and conditions for use, reproduction, or 113 | distribution of Your modifications, or for any such Derivative Works as a whole, 114 | provided Your use, reproduction, and distribution of the Work otherwise complies 115 | with the conditions stated in this License. 116 | 117 | 5. Submission of Contributions. 118 | 119 | Unless You explicitly state otherwise, any Contribution intentionally submitted 120 | for inclusion in the Work by You to the Licensor shall be under the terms and 121 | conditions of this License, without any additional terms or conditions. 122 | Notwithstanding the above, nothing herein shall supersede or modify the terms of 123 | any separate license agreement you may have executed with Licensor regarding 124 | such Contributions. 125 | 126 | 6. Trademarks. 127 | 128 | This License does not grant permission to use the trade names, trademarks, 129 | service marks, or product names of the Licensor, except as required for 130 | reasonable and customary use in describing the origin of the Work and 131 | reproducing the content of the NOTICE file. 132 | 133 | 7. Disclaimer of Warranty. 134 | 135 | Unless required by applicable law or agreed to in writing, Licensor provides the 136 | Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, 137 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, 138 | including, without limitation, any warranties or conditions of TITLE, 139 | NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are 140 | solely responsible for determining the appropriateness of using or 141 | redistributing the Work and assume any risks associated with Your exercise of 142 | permissions under this License. 143 | 144 | 8. Limitation of Liability. 145 | 146 | In no event and under no legal theory, whether in tort (including negligence), 147 | contract, or otherwise, unless required by applicable law (such as deliberate 148 | and grossly negligent acts) or agreed to in writing, shall any Contributor be 149 | liable to You for damages, including any direct, indirect, special, incidental, 150 | or consequential damages of any character arising as a result of this License or 151 | out of the use or inability to use the Work (including but not limited to 152 | damages for loss of goodwill, work stoppage, computer failure or malfunction, or 153 | any and all other commercial damages or losses), even if such Contributor has 154 | been advised of the possibility of such damages. 155 | 156 | 9. Accepting Warranty or Additional Liability. 157 | 158 | While redistributing the Work or Derivative Works thereof, You may choose to 159 | offer, and charge a fee for, acceptance of support, warranty, indemnity, or 160 | other liability obligations and/or rights consistent with this License. However, 161 | in accepting such obligations, You may act only on Your own behalf and on Your 162 | sole responsibility, not on behalf of any other Contributor, and only if You 163 | agree to indemnify, defend, and hold each Contributor harmless for any liability 164 | incurred by, or claims asserted against, such Contributor by reason of your 165 | accepting any such warranty or additional liability. 166 | 167 | END OF TERMS AND CONDITIONS 168 | 169 | APPENDIX: How to apply the Apache License to your work 170 | 171 | To apply the Apache License to your work, attach the following boilerplate 172 | notice, with the fields enclosed by brackets "[]" replaced with your own 173 | identifying information. (Don't include the brackets!) The text should be 174 | enclosed in the appropriate comment syntax for the file format. We also 175 | recommend that a file or class name and description of purpose be included on 176 | the same "printed page" as the copyright notice for easier identification within 177 | third-party archives. 178 | 179 | Copyright [yyyy] [name of copyright owner] 180 | 181 | Licensed under the Apache License, Version 2.0 (the "License"); 182 | you may not use this file except in compliance with the License. 183 | You may obtain a copy of the License at 184 | 185 | http://www.apache.org/licenses/LICENSE-2.0 186 | 187 | Unless required by applicable law or agreed to in writing, software 188 | distributed under the License is distributed on an "AS IS" BASIS, 189 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 190 | See the License for the specific language governing permissions and 191 | limitations under the License. 192 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # oauth2-firebase 2 | 3 | This library provides OAuth2 server implementation for Firebase. The points are: 4 | 5 | * Supporting Google Sign-In, GitHub Login and Facebook Login to authenticate users as Federation ID provider using Firebase Authentication. 6 | * Providing each endpoint for Cloud Functions. 7 | * Storing information into Cloud Firestore. 8 | * Supporting Authorization Code Grant, Implicit Grant and Client Credentials grant of OAuth 2.0. 9 | 10 | [![NPM Version](https://img.shields.io/npm/v/oauth2-firebase.svg)](https://www.npmjs.org/package/oauth2-firebase) 11 | 12 | # How to install 13 | 14 | This section describes how to use this library. 15 | 16 | ## Prerequisite 17 | 18 | You must already have some Firebase project which enables Cloud Functions, Cloud Firestore and Firebase Authentication. 19 | Especially, it is necessary to enable the Google Sign-In or Facebook Login for Federation ID provider on the Firebase Authentication. 20 | 21 | ## Install this library 22 | 23 | This library has been providing as JavaScript library on the npm repository. You can install this library 24 | with the `npm` command. We represent your project directory `${PROJECT_HOME}`. 25 | 26 | ```bash 27 | $ cd ${PROJECT_HOME} 28 | $ cd functions 29 | $ npm install oauth2-firebase --save 30 | ``` 31 | 32 | ## Define endpoints as Cloud Functions 33 | 34 | This library provides some endpoints for OAuth 2.0. Each endpoint is a handler function for the express. 35 | 36 | If you use the TypeScript to write your functions, add the following code to your `functions/index.ts` file. 37 | 38 | ``` 39 | $ vi index.ts 40 | ``` 41 | 42 | The code you need to write is the following: 43 | 44 | **Google Sign-In** 45 | 46 | ```javascript 47 | import * as functions from "firebase-functions"; 48 | import {authorize, Configuration, googleAccountAuthentication, token} from "oauth2-firebase"; 49 | 50 | Configuration.init({ 51 | crypto_auth_token_secret_key_32: functions.config().crypto.auth_token_secret_key_32, 52 | project_api_key: functions.config().project.api_key 53 | }); 54 | 55 | exports.token = token(); 56 | exports.authorize = authorize(); 57 | exports.authentication = googleAccountAuthentication(); 58 | 59 | ... 60 | ``` 61 | 62 | **Facebook Login** 63 | 64 | ```javascript 65 | import * as functions from "firebase-functions"; 66 | import {authorize, Configuration, facebookAccountAuthentication, token} from "oauth2-firebase"; 67 | 68 | Configuration.init({ 69 | crypto_auth_token_secret_key_32: functions.config().crypto.auth_token_secret_key_32, 70 | project_api_key: functions.config().project.api_key 71 | }); 72 | 73 | exports.token = token(); 74 | exports.authorize = authorize(); 75 | exports.authentication = facebookAccountAuthentication(); 76 | 77 | ... 78 | ``` 79 | 80 | **GitHub Login** 81 | 82 | ```javascript 83 | import * as functions from "firebase-functions"; 84 | import {authorize, Configuration, githubAccountAuthentication, token} from "oauth2-firebase"; 85 | 86 | Configuration.init({ 87 | crypto_auth_token_secret_key_32: functions.config().crypto.auth_token_secret_key_32, 88 | project_api_key: functions.config().project.api_key 89 | }); 90 | 91 | exports.token = token(); 92 | exports.authorize = authorize(); 93 | exports.authentication = githubAccountAuthentication(); 94 | 95 | ... 96 | ``` 97 | 98 | By the code above, the following endpoints are defined: 99 | 100 | * `https://.../token` - Token endpoint. 101 | * `https://.../authorize` - Authorization endpoint. 102 | * `https://.../authentication` - Login page for Google Sign-In. 103 | 104 | ## Generate a shared key 105 | 106 | This library uses a shared key for navigating pages. You need to generate a random string for the shared key. 107 | The string must be 32 length. For example: 108 | 109 | ```bash 110 | $ cat /dev/urandom | base64 | fold -w 32 | head -n 1 111 | ``` 112 | 113 | ## Set a configuration value to your project 114 | 115 | After generating the random string, you need to set the string as the shared key with the following `firebase` command. 116 | 117 | ```bash 118 | firebase functions:config:set crypto.auth_token_secret_key_32= 119 | ``` 120 | 121 | In addition, you need to set the API Key value of your Firebase project. You can retrieve the API Key value by the 122 | following steps: 123 | 124 | 1. Go to the setting page of your Firebase project: `https://console.firebase.google.com/project//settings/general/` 125 | 1. Get the string of the field labeled `Web API Key`. 126 | 127 | Then, execute the following command to register the configuration: 128 | 129 | ```bash 130 | firebase functions:config:set project.api_key= 131 | ``` 132 | 133 | ## Deploy your project 134 | 135 | After writing the code and setting the configuration, deploy your project to the Firebase. 136 | 137 | ```bash 138 | $ firebase deploy --only functions 139 | ``` 140 | 141 | # Operations 142 | 143 | You need to setup the database to operate OAuth2.0 server as like the following: 144 | 145 | * Register your client 146 | * Set a description for each scope 147 | 148 | ## Register your client 149 | 150 | In OAuth2.0, each client must be registered in advance. This library uses the Cloud Firestore as the storage 151 | for the client definitions. In the current version, you need to register client definitions with the Firebase Console 152 | manually. To register a client definition, add a new doc in a "clients" collection as like the following: 153 | 154 | * Collection: `clients` 155 | * Doc ID: Auto-generated. This will be used as a Client ID value. 156 | * Fields: 157 | * `user_id` - The user ID which represents this client as a user. 158 | * `provider_name` - The provider name who this client provides. 159 | * `client_secret` - The client secret string. You need to generate this string as the shared key, and need to share the provider. 160 | * `redirect_uri` - If this client supports Authorization Code grant and Implicit grant, you need to set this redirect_uri string. 161 | * `grant_type` - This is an object. Each key represents a grant type, and each value is boolean whether the grant type is supported or not. You need to set these entries: `authorization_code`, `password`, `client_credentials` and `refresh_token`. 162 | * `response_type` - This is an object. Each key represents a response type, and each value is boolean whether the response type is supported or not. You need to set these entries: `code` and `token`. 163 | * `scope` - This is an object. Each key represents a scope, and each value is boolean whether the scope is supported or not. You need to set the entry: `profile`. 164 | 165 | The following is a sample JSON string which represents the values above: 166 | 167 | ```json 168 | { 169 | "user_id": "client@123", 170 | "provider_name": "Google, Inc.", 171 | "client_secret": "foobar123456", 172 | "redirect_uri": "https://foobar.com/foo/bar/baz", 173 | "grant_type": { 174 | "authorization_code": true, 175 | "password": false, 176 | "client_credentials": true, 177 | "refresh_token": true 178 | }, 179 | "response_type": { 180 | "code": true, 181 | "token": true 182 | }, 183 | "scope": { 184 | "profile": true 185 | } 186 | } 187 | ``` 188 | 189 | ## Set a description for each scope 190 | 191 | This library shows a consent page to ask whether they allow or deny scopes. You need to register descriptions for each scope with the Firebase Console manually. To register a scope description, add a new doc in a "scopes" collection as like the following: 192 | 193 | * Collection: `scopes` 194 | * Doc ID: Auto-generated. 195 | * Fields: 196 | * `name` - Scope name (ex. "profile"). 197 | * `description` - Scope description (ex. "User profile information (User ID and Nickname)"). 198 | 199 | The following is a sample JSON string which represents the values above: 200 | 201 | ```json 202 | { 203 | "name": "profile", 204 | "description": "User profile information (User ID and Nickname)" 205 | } 206 | ``` 207 | 208 | # Use Additional Endpoints 209 | 210 | This library provides some additional endpoints: 211 | 212 | * userinfo - Userinfo API endpoint. 213 | * tokeninfo - Tokeninfo API endpoint. 214 | 215 | ## Userinfo API endpoint 216 | 217 | In OpenID Connect specification, the userinfo endpoint is defined. It provides the authenticated user's information. 218 | You can provide the userinfo API endpoint easily by writing the following code: 219 | 220 | ```javascript 221 | import {userinfo} from "oauth2-firebase"; 222 | ... 223 | exports.userinfo = userinfo(); 224 | ``` 225 | 226 | This userinfo endpoint works as a protected resource endpoint. That is, the access token is necessary to use this endpoint. 227 | For example: 228 | 229 | ```bash 230 | $ curl -X POST -H "Authorization: Bearer " https://.../userinfo 231 | ``` 232 | 233 | If the access token is valid, you will retrieve the following result: 234 | 235 | ```json 236 | { 237 | "sub": "", 238 | "name": "" 239 | } 240 | ``` 241 | 242 | ## Tokeninfo API endpoint 243 | 244 | The tokeninfo API endpoint provides the information of the passed access token. By this endpoint, you can confirm 245 | whether the passed access token is issued for your client or not. You can provide the tokeninfo API endpoint easily by 246 | writing the following code: 247 | 248 | ```javascript 249 | import {tokeninfo} from "oauth2-firebase"; 250 | ... 251 | exports.tokeninfo = tokeninfo(); 252 | ``` 253 | 254 | The tokeninfo API endpoint accepts an access token as a query parameter called "access_token". For example: 255 | 256 | ```bash 257 | curl https://.../tokeninfo?access_token= 258 | ``` 259 | 260 | If the access token is valid, you will retrieve the following result: 261 | 262 | ```json 263 | { 264 | "aud": "", 265 | "sub": "", 266 | "expires_in": "", 267 | "scope": "" 268 | } 269 | ``` 270 | 271 | You can check whether the access token is for your client or not by comparing the `aud` value. 272 | 273 | # Configurations 274 | 275 | You can configure each behavior of this library. 276 | 277 | ## Set expires_in values to access tokens 278 | 279 | You can set each expires_in values (unit: sec) for access tokens per grant types. For example: 280 | 281 | ```javascript 282 | const expiresInMap = new Map(); 283 | expiresInMap.set("authorization_code", 2678400); 284 | expiresInMap.set("implicit", 86400); 285 | expiresInMap.set("password", 86400); 286 | expiresInMap.set("client_credentials", 2678400); 287 | expiresInMap.set("refresh_token", 2678400); 288 | Configuration.init({ 289 | ... 290 | tokens_expires_in: expiresInMap 291 | }); 292 | ``` 293 | 294 | In this library, the default values are: 295 | 296 | * Authorization Code Grant: 86400 297 | * Implicit Grant: 3600 298 | * Password: 86400 299 | * Client Credentials: 86400 300 | * Refresh Token: 86400 301 | 302 | ## Customize the consent page design 303 | 304 | This library provides a very simple design of the consent page. But, you can customize the design. For instance, you 305 | can provide your own template string for the consent page from your code. 306 | 307 | To customize the page design, you need to create a new class which implements the `ConsentViewTemplate` interface. 308 | For example, the class code will be like the following: 309 | 310 | ```javascript 311 | import {ConsentViewTemplate} from "oauth2-firebase/dist/endpoint/views/consent_view_template"; 312 | 313 | export class MyConsentViewTemplate implements ConsentViewTemplate { 314 | 315 | provide(): Promise { 316 | return new Promise((resolve, reject) => { 317 | resolve(` 318 | 319 | 320 | 321 | 322 | 323 | Authorization page 324 | 325 | 326 |

<%= providerName %> requests the following permissions:

327 |
    328 | <% for (const key of scope.split(" ")) { %> 329 |
  • <%= scopes.get(key) %>
  • 330 | <% } %> 331 |
332 |

Could you allow them?

333 |
334 | 335 | 336 | 337 | 338 |
339 | 340 | 341 | `) 342 | }) 343 | } 344 | 345 | } 346 | ``` 347 | 348 | The template string is written as the "ejs" template. This library binds the following values to the template at rendering. 349 | 350 | * `providerName: string` - The provider name of the client. 351 | * `scope: string` - The scope string devided by space the client code specifies. 352 | * `scopes: Map` - The map object which has a set of the scope name and its description. 353 | * `encryptedAuthToken: string` - The encrypted auth token. You need to set this as the hidden parameter. 354 | * `encryptedUserId: string` - The encrypted user ID. You need to set this as the hidden parameter. 355 | 356 | And, you need to set the instance to the Configuration class instance as like the following: 357 | 358 | ```javascript 359 | import * as functions from "firebase-functions"; 360 | import {authorize, Configuration, googleAccountAuthentication, token, userinfo} from "oauth2-firebase"; 361 | import {MyConsentViewTemplate} from "./my_consent_view_template" 362 | 363 | Configuration.init({ 364 | crypto_auth_token_secret_key_32: functions.config().crypto.auth_token_secret_key_32, 365 | project_api_key: functions.config().project.api_key, 366 | views_consent_template: new MyConsentViewTemplate() 367 | }); 368 | 369 | exports.token = token(); 370 | exports.authorize = authorize(); 371 | exports.authentication = googleAccountAuthentication(); 372 | exports.userinfo = userinfo(); 373 | 374 | ... 375 | ``` 376 | 377 | ## Add Your Protected Resource Endpoint 378 | 379 | In this library, the `userinfo` protected resource endpoint is provided as default. But, you can add your own protected 380 | resource endpoint. Each protected resource receives the request including the access token issued for users/clients, 381 | checks whether the access token is valid or not against using the protected resource, and actually returns the resources 382 | and/or creates some resource or does something. This library provides a convenience abstract class. You can define your 383 | endpoint by creating a new class which extends the abstract class and implements the following two methods: 384 | 385 | * `validateScope()` - Check whether the passed scopes are valid to call this endpoint. 386 | * `handleRequest()` - The code body to access to target resources. 387 | 388 | To publish your endpoint on the Cloud Functions, you need to retrieve the endpoint function by the `endpoint` property. 389 | As the result, your code will be like the following: 390 | 391 | ```javascript 392 | import * as express from "express"; 393 | import {AbstractProtectedResourceEndpoint} from "oauth2-firebase"; 394 | import {ProtectedResourceEndpointResponse} from "oauth2-nodejs"; 395 | 396 | class FriendsEndpoint extends AbstractProtectedResourceEndpoint { 397 | 398 | protected validateScope(scopes: string[]): boolean { 399 | return scopes.indexOf("frields") !== -1; 400 | } 401 | 402 | protected handleRequest(req: express.Request, endpointInfo: ProtectedResourceResponse): Promise { 403 | return new Promise((resolve, reject) => { 404 | fetchFrields(endpointInfo.userId).then(friends => { 405 | resolve(JSON.stringify(friends)); 406 | }).catch(e => { 407 | reject(e); 408 | }) 409 | }); 410 | } 411 | 412 | } 413 | 414 | exports.friends = new FriendsEndpoint().endpoint; 415 | ``` 416 | 417 | If the passed access token is invalid, the `handleRequest()` function will not be called and returns an error response 418 | by the abstract class. 419 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "oauth2-firebase", 3 | "version": "0.1.16", 4 | "description": "This library provides OAuth2 server implementation for Firebase", 5 | "main": "dist/index.js", 6 | "types": "dist/index.d.ts", 7 | "scripts": { 8 | "lint": "tslint -p .", 9 | "clean": "rm -rf dist", 10 | "clean:pack": "rm -f *.tgz", 11 | "build": "tsc", 12 | "build:watch": "tsc -w", 13 | "build:clean": "yarn clean:pack && yarn clean && yarn build", 14 | "package": "yarn build:clean && yarn pack", 15 | "release": "yarn build:clean && yarn publish" 16 | }, 17 | "repository": { 18 | "type": "git", 19 | "url": "git+ssh://git@github.com/yoichiro/oauth2-firebase.git" 20 | }, 21 | "author": "Yoichiro Tanaka (https://github.com/yoichiro)", 22 | "license": "Apache-2.0", 23 | "bugs": { 24 | "url": "https://github.com/yoichiro/oauth2-firebase/issues", 25 | "email": "yoichiro6642@gmail.com" 26 | }, 27 | "homepage": "https://github.com/yoichiro/oauth2-firebase.git", 28 | "dependencies": { 29 | "ejs": "^2.6.1", 30 | "express": "^4.16.3", 31 | "firebase-admin": "~5.13.1", 32 | "firebase-functions": "^2.0.0", 33 | "oauth2-nodejs": "^0.2.0", 34 | "secure-random-string": "^1.1.0" 35 | }, 36 | "devDependencies": { 37 | "@types/ejs": "^2.6.0", 38 | "@types/express": "^4.16.0", 39 | "@types/node": "^10.5.2", 40 | "tslint": "^5.8.0", 41 | "typescript": "^2.5.3" 42 | } 43 | } 44 | -------------------------------------------------------------------------------- /src/data/cloud_firestore_clients.ts: -------------------------------------------------------------------------------- 1 | import * as admin from "firebase-admin"; 2 | 3 | export class Client { 4 | 5 | private _clientId: string 6 | private _providerName: string 7 | private _scopes: string[] 8 | private _responseTypes: string[] 9 | private _grantTypes: string[] 10 | private _userId: string 11 | private _clientSecret: string 12 | 13 | get clientId(): string { 14 | return this._clientId; 15 | } 16 | 17 | set clientId(value: string) { 18 | this._clientId = value; 19 | } 20 | 21 | get providerName(): string { 22 | return this._providerName; 23 | } 24 | 25 | set providerName(value: string) { 26 | this._providerName = value; 27 | } 28 | 29 | get scopes(): string[] { 30 | return this._scopes; 31 | } 32 | 33 | set scopes(value: string[]) { 34 | this._scopes = value; 35 | } 36 | 37 | get responseTypes(): string[] { 38 | return this._responseTypes; 39 | } 40 | 41 | set responseTypes(value: string[]) { 42 | this._responseTypes = value; 43 | } 44 | 45 | get grantTypes(): string[] { 46 | return this._grantTypes; 47 | } 48 | 49 | set grantTypes(value: string[]) { 50 | this._grantTypes = value; 51 | } 52 | 53 | get userId(): string { 54 | return this._userId; 55 | } 56 | 57 | set userId(value: string) { 58 | this._userId = value; 59 | } 60 | 61 | get clientSecret(): string { 62 | return this._clientSecret; 63 | } 64 | 65 | set clientSecret(value: string) { 66 | this._clientSecret = value; 67 | } 68 | 69 | } 70 | 71 | export class CloudFirestoreClients { 72 | 73 | public static async fetch(clientId: string): Promise { 74 | const db = admin.firestore() 75 | const client = await db.collection("clients").doc(clientId).get() 76 | if (client.exists) { 77 | const result = new Client() 78 | result.clientId = client.id 79 | result.clientSecret = client.get("client_secret") 80 | result.grantTypes = Object.keys(client.get("grant_type")).filter((value: string): boolean => {return client.get(`grant_type.${value}`)}) 81 | result.responseTypes = Object.keys(client.get("response_type")).filter((value: string): boolean => {return client.get(`response_type.${value}`)}) 82 | result.scopes = Object.keys(client.get("scope")).filter((value: string): boolean => {return client.get(`scope.${value}`)}) 83 | result.userId = client.get("user_id") 84 | result.providerName = client.get("provider_name") 85 | return result 86 | } else { 87 | return undefined 88 | } 89 | } 90 | 91 | } 92 | -------------------------------------------------------------------------------- /src/data/cloud_firestore_data_handler.ts: -------------------------------------------------------------------------------- 1 | import * as functions from "firebase-functions"; 2 | import * as admin from "firebase-admin"; 3 | import * as firestore from "@google-cloud/firestore"; 4 | import * as url from "url"; 5 | import {AccessToken, AuthInfo, DataHandler, Request} from "oauth2-nodejs"; 6 | import {Configuration} from "../utils"; 7 | const secureRandomString = require("secure-random-string"); 8 | 9 | admin.initializeApp(functions.config().firebase) 10 | 11 | export class CloudFirestoreDataHandler implements DataHandler { 12 | 13 | private _request: Request 14 | 15 | constructor(request: Request) { 16 | this._request = request 17 | } 18 | 19 | getRequest(): Request { 20 | return this._request 21 | } 22 | 23 | async createOrUpdateAccessToken(authInfo: AuthInfo, grantType: string): Promise { 24 | const db = admin.firestore() 25 | const token = secureRandomString({length: 128}) 26 | const expiresIn = Configuration.instance.tokens_expires_in.get(grantType) || 86400 27 | const createdOn = Date.now() 28 | const data = { 29 | auth_info_id: authInfo.id, 30 | token: token, 31 | expires_in: expiresIn, 32 | created_on: createdOn 33 | } 34 | await db.collection("access_tokens").add(data) 35 | const result = new AccessToken() 36 | result.authId = authInfo.id 37 | result.expiresIn = expiresIn 38 | result.createdOn = createdOn 39 | result.token = token 40 | return result 41 | } 42 | 43 | async createOrUpdateAuthInfo(clientId: string, userId: string, scope?: string): Promise { 44 | // TODO: Check the scope 45 | const db = admin.firestore() 46 | let queryRef = db.collection("auth_infos").where("client_id", "==", clientId).where("user_id", "==", userId) 47 | if (scope) { 48 | scope.split(" ").forEach(s => { 49 | queryRef = queryRef.where(`scope.${s}`, "==", true) 50 | }) 51 | } 52 | const snapshot = await queryRef.get() 53 | const code = secureRandomString({length: 64}) 54 | if (snapshot.empty) { 55 | const refreshToken = secureRandomString() 56 | const data = { 57 | user_id: userId, 58 | client_id: clientId, 59 | scope: scope ? scope.split(" ").reduce((a: {[key: string]: boolean}, c: string) => {a[c] = true; return a;}, {}) : {}, 60 | refresh_token: refreshToken, 61 | code: code 62 | } 63 | const authInfoRef = await db.collection("auth_infos").add(data) 64 | const result = new AuthInfo() 65 | result.id = authInfoRef.id 66 | result.userId = userId 67 | result.clientId = clientId 68 | result.refreshToken = refreshToken 69 | result.code = code 70 | if (scope) { 71 | result.scope = scope 72 | } 73 | return result 74 | } else { 75 | // TODO: Check the size of the docs 76 | const authInfo = snapshot.docs[0] 77 | await db.collection("auth_infos").doc(authInfo.id).update({ 78 | code: code 79 | }) 80 | const result = new AuthInfo() 81 | result.id = authInfo.id 82 | result.userId = authInfo.get("user_id") 83 | result.clientId = authInfo.get("client_id") 84 | result.refreshToken = authInfo.get("refresh_token") 85 | result.code = code 86 | const scopes = Object.keys(authInfo.get("scope")) 87 | if (scopes.length > 0) { 88 | result.scope = scopes.join(" ") 89 | } 90 | return result 91 | } 92 | } 93 | 94 | private async findAuthInfo(fieldName: string, fieldValue: string): Promise { 95 | const db = admin.firestore() 96 | const queryRef = db.collection("auth_infos").where(fieldName, "==", fieldValue) 97 | const snapshot = await queryRef.get() 98 | if (snapshot.empty) { 99 | return undefined 100 | } else { 101 | // TODO: Check the size of the docs 102 | const authInfo = snapshot.docs[0] 103 | return this.convertAuthInfo(authInfo) 104 | } 105 | } 106 | 107 | private convertAuthInfo(authInfo: firestore.DocumentSnapshot): AuthInfo { 108 | const result = new AuthInfo() 109 | result.id = authInfo.id 110 | result.userId = authInfo.get("user_id") 111 | result.clientId = authInfo.get("client_id") 112 | result.refreshToken = authInfo.get("refresh_token") 113 | const scopes = Object.keys(authInfo.get("scope")) 114 | if (scopes.length > 0) { 115 | result.scope = scopes.join(" ") 116 | } 117 | return result 118 | } 119 | 120 | async getClientUserId(clientId: string, clientSecret: string): Promise { 121 | const db = admin.firestore() 122 | const client = await db.collection("clients").doc(clientId).get() 123 | if (client.exists) { 124 | return client.get("user_id") 125 | } 126 | return undefined 127 | } 128 | 129 | async validateClient(clientId: string, clientSecret: string, grantType: string): Promise { 130 | const db = admin.firestore() 131 | const client = await db.collection("clients").doc(clientId).get() 132 | if (client.exists) { 133 | // TODO: Check the client status and/or etc. 134 | return client.get(`grant_type.${grantType}`) && client.get("client_secret") === clientSecret 135 | } 136 | return false 137 | } 138 | 139 | async validateClientById(clientId: string): Promise { 140 | const db = admin.firestore() 141 | const client = await db.collection("clients").doc(clientId).get() 142 | if (client.exists) { 143 | // TODO: Check the client status and/or etc. 144 | return true 145 | } 146 | return false 147 | } 148 | 149 | async validateClientForAuthorization(clientId: string, responseType: string): Promise { 150 | const db = admin.firestore() 151 | const client = await db.collection("clients").doc(clientId).get() 152 | if (client.exists) { 153 | return responseType.split(" ").every((value: string): boolean => { 154 | return client.get(`response_type.${value}`) 155 | }) 156 | } 157 | return false 158 | } 159 | 160 | async validateRedirectUri(clientId: string, redirectUri: string): Promise { 161 | const db = admin.firestore() 162 | const client = await db.collection("clients").doc(clientId).get() 163 | if (client.exists) { 164 | const registeredRedirectUri = client.get("redirect_uri") 165 | const validRedirectUrl = url.parse(registeredRedirectUri) 166 | const redirectUrl = url.parse(redirectUri) 167 | return (validRedirectUrl.protocol === redirectUrl.protocol) && 168 | (validRedirectUrl.host === redirectUrl.host) && 169 | (validRedirectUrl.pathname === redirectUrl.pathname) 170 | } 171 | return false 172 | } 173 | 174 | async validateScope(clientId: string, scope?: string): Promise { 175 | if (scope) { 176 | const db = admin.firestore() 177 | const client = await db.collection("clients").doc(clientId).get() 178 | if (client.exists) { 179 | return client.get(`scope.${scope}`) 180 | } 181 | return false 182 | } else { 183 | return false 184 | } 185 | } 186 | 187 | async getAccessToken(token: string): Promise { 188 | const db = admin.firestore() 189 | const queryRef = await db.collection("access_tokens").where("token", "==", token) 190 | const snapshot = await queryRef.get() 191 | if (snapshot.empty) { 192 | return undefined 193 | } else { 194 | // TODO: Check the size of the docs 195 | const accessToken = snapshot.docs[0] 196 | const result = new AccessToken() 197 | result.authId = accessToken.get("auth_info_id") 198 | result.expiresIn = accessToken.get("expires_in") 199 | result.createdOn = accessToken.get("created_on") 200 | result.token = accessToken.get("token") 201 | return result 202 | } 203 | } 204 | 205 | async getAuthInfoByRefreshToken(refreshToken: string): Promise { 206 | return await this.findAuthInfo("refresh_token", refreshToken) 207 | } 208 | 209 | async getAuthInfoByCode(code: string): Promise { 210 | return await this.findAuthInfo("code", code) 211 | } 212 | 213 | async getAuthInfoById(id: string): Promise { 214 | const db = admin.firestore() 215 | const authInfo = await db.collection("auth_infos").doc(id).get() 216 | if (authInfo.exists) { 217 | return this.convertAuthInfo(authInfo) 218 | } else { 219 | return undefined 220 | } 221 | } 222 | 223 | async validateUserById(userId: string): Promise { 224 | const auth = admin.auth() 225 | try { 226 | const user = await auth.getUser(userId) 227 | if (user) { 228 | return true 229 | } 230 | } catch(e) { 231 | console.log("e", e) 232 | } 233 | return false 234 | } 235 | 236 | async getUserId(username: string, password: string): Promise { 237 | throw new Error("Not implemented") 238 | } 239 | 240 | } 241 | -------------------------------------------------------------------------------- /src/data/cloud_firestore_data_handler_factory.ts: -------------------------------------------------------------------------------- 1 | import {DataHandler, DataHandlerFactory, Request} from "oauth2-nodejs"; 2 | import {CloudFirestoreDataHandler} from "./cloud_firestore_data_handler"; 3 | 4 | export class CloudFirestoreDataHandlerFactory implements DataHandlerFactory { 5 | 6 | create(request: Request): DataHandler { 7 | return new CloudFirestoreDataHandler(request) 8 | } 9 | 10 | } 11 | -------------------------------------------------------------------------------- /src/data/cloud_firestore_scopes.ts: -------------------------------------------------------------------------------- 1 | import * as admin from "firebase-admin"; 2 | import * as firestore from "@google-cloud/firestore"; 3 | 4 | export class CloudFirestoreScopes { 5 | 6 | public static async fetch(): Promise> { 7 | const db = admin.firestore() 8 | const snapshot = await db.collection("scopes").get() 9 | const result = new Map() 10 | snapshot.forEach((doc: firestore.QueryDocumentSnapshot): void => { 11 | result.set(doc.get("name"), doc.get("description")) 12 | }) 13 | return result 14 | } 15 | 16 | } 17 | -------------------------------------------------------------------------------- /src/data/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./cloud_firestore_clients" 2 | export * from "./cloud_firestore_data_handler" 3 | export * from "./cloud_firestore_data_handler_factory" 4 | export * from "./cloud_firestore_scopes" 5 | -------------------------------------------------------------------------------- /src/endpoint/authentication.ts: -------------------------------------------------------------------------------- 1 | import * as functions from "firebase-functions"; 2 | import * as admin from "firebase-admin"; 3 | import * as express from "express"; 4 | import * as path from "path"; 5 | import {RequestWrapper} from "../models"; 6 | import {Configuration, Crypto, Navigation} from "../utils"; 7 | 8 | class AuthenticationApp { 9 | 10 | static create(providerName: string): express.Express { 11 | const authenticationApp = express() 12 | 13 | authenticationApp.set("views", path.join(__dirname, "../../views")) 14 | 15 | authenticationApp.get("/", (req, resp) => { 16 | const request = new RequestWrapper(req) 17 | const authToken = request.getParameter("auth_token") 18 | resp.render("authentication.ejs", { 19 | authToken: authToken, 20 | projectId: process.env.GCLOUD_PROJECT, 21 | projectApiKey: Configuration.instance.project_apikey, 22 | providerName: providerName 23 | }) 24 | }) 25 | 26 | authenticationApp.post("/", async (req, resp) => { 27 | const request = new RequestWrapper(req) 28 | const encryptedAuthToken = request.getParameter("auth_token")! 29 | const idTokenString = request.getParameter("id_token")! 30 | const success = request.getParameter("success") 31 | const error = request.getParameter("error") 32 | if (success === "true") { 33 | try { 34 | const idToken = await admin.auth().verifyIdToken(idTokenString) 35 | if (idToken.aud === process.env.GCLOUD_PROJECT) { 36 | const encryptedUserId = Crypto.encrypt(idToken.sub) 37 | Navigation.redirect(resp, "/authorize/consent", {"auth_token": encryptedAuthToken, "user_id": encryptedUserId}) 38 | } 39 | } catch(e) { 40 | console.log("e", e) 41 | } 42 | } else { 43 | console.log("error", error) 44 | } 45 | const authToken = JSON.parse(Crypto.decrypt(request.getParameter("auth_token")!)) 46 | Navigation.redirect(resp, authToken["redirect_uri"], {"error": "access_denied"}) 47 | }) 48 | 49 | return authenticationApp 50 | } 51 | 52 | } 53 | 54 | export function googleAccountAuthentication() { 55 | return functions.https.onRequest(AuthenticationApp.create("Google")) 56 | } 57 | 58 | export function facebookAccountAuthentication() { 59 | return functions.https.onRequest(AuthenticationApp.create("Facebook")) 60 | } 61 | 62 | export function githubAccountAuthentication() { 63 | return functions.https.onRequest(AuthenticationApp.create("Github")) 64 | } 65 | -------------------------------------------------------------------------------- /src/endpoint/authorize.ts: -------------------------------------------------------------------------------- 1 | import * as functions from "firebase-functions"; 2 | import * as express from "express"; 3 | import * as ejs from "ejs"; 4 | import {RequestMap, RequestWrapper} from "../models"; 5 | import {AuthorizationEndpoint} from "oauth2-nodejs"; 6 | import {CloudFirestoreDataHandlerFactory, CloudFirestoreScopes, CloudFirestoreClients} from "../data"; 7 | import {Configuration, Crypto, Navigation} from "../utils"; 8 | 9 | const authorizeApp = express() 10 | 11 | authorizeApp.get("/entry", async (req, resp) => { 12 | const request = new RequestWrapper(req) 13 | const authorizationEndpoint = new AuthorizationEndpoint() 14 | authorizationEndpoint.dataHandlerFactory = new CloudFirestoreDataHandlerFactory() 15 | authorizationEndpoint.allowedResponseTypes = ["code", "token"] 16 | try { 17 | const authorizationEndpointResponse = await authorizationEndpoint.handleRequest(request) 18 | if (authorizationEndpointResponse.isSuccess()) { 19 | 20 | const authToken: {[key: string]: string | number} = { 21 | "client_id": request.getParameter("client_id")!, 22 | "redirect_uri": request.getParameter("redirect_uri")!, 23 | "response_type": request.getParameter("response_type")!, 24 | "scope": request.getParameter("scope")!, 25 | "created_at": Date.now() 26 | } 27 | const state = request.getParameter("state") 28 | if (state) { 29 | authToken["state"] = state 30 | } 31 | const authTokenString = Crypto.encrypt(JSON.stringify(authToken)) 32 | Navigation.redirect(resp, "/authentication/", {"auth_token": authTokenString}) 33 | } else { 34 | const error = authorizationEndpointResponse.error 35 | resp.set("Content-Type", "application/json; charset=UTF-8") 36 | resp.status(error.code).send(error.toJson()) 37 | } 38 | } catch(e) { 39 | console.error(e) 40 | resp.status(500).send(e.toString()) 41 | } 42 | }) 43 | 44 | authorizeApp.get("/consent", async (req, resp) => { 45 | const request = new RequestWrapper(req) 46 | const encryptedAuthToken = request.getParameter("auth_token")! 47 | const authToken = JSON.parse(Crypto.decrypt(encryptedAuthToken)) 48 | const client = await CloudFirestoreClients.fetch(authToken["client_id"]) 49 | const encryptedUserId = request.getParameter("user_id")! 50 | const scopes = await CloudFirestoreScopes.fetch() 51 | const consentViewTemplate = Configuration.instance.view_consent_template 52 | try { 53 | const template = await consentViewTemplate.provide() 54 | const html = ejs.render(template, { 55 | scope: authToken["scope"], 56 | encryptedAuthToken, 57 | encryptedUserId, 58 | scopes, 59 | providerName: client!["providerName"] 60 | }) 61 | resp.status(200).send(html) 62 | } catch(e) { 63 | console.error(e) 64 | resp.status(500).send(e.toString()) 65 | } 66 | }) 67 | 68 | authorizeApp.post("/consent", async (req, resp) => { 69 | const requestWrapper = new RequestWrapper(req) 70 | const encryptedAuthToken = requestWrapper.getParameter("auth_token")! 71 | const authToken = JSON.parse(Crypto.decrypt(encryptedAuthToken)) 72 | const encryptedUserId = requestWrapper.getParameter("user_id")! 73 | const userId = Crypto.decrypt(encryptedUserId) 74 | const requestMap = new RequestMap() 75 | requestMap.setParameter("user_id", userId) 76 | requestMap.setParameter("state", authToken["state"]) 77 | requestMap.setParameter("client_id", authToken["client_id"]) 78 | requestMap.setParameter("redirect_uri", authToken["redirect_uri"]) 79 | requestMap.setParameter("response_type", authToken["response_type"]) 80 | requestMap.setParameter("scope", authToken["scope"]) 81 | const authorizationEndpoint = new AuthorizationEndpoint() 82 | authorizationEndpoint.dataHandlerFactory = new CloudFirestoreDataHandlerFactory() 83 | authorizationEndpoint.allowedResponseTypes = ["code", "token"] 84 | const action = requestWrapper.getParameter("action") 85 | if (action === "allow") { 86 | Navigation.backTo(resp, await authorizationEndpoint.allow(requestMap), authToken["redirect_uri"]) 87 | } else { 88 | Navigation.backTo(resp, await authorizationEndpoint.deny(requestMap), authToken["redirect_uri"]) 89 | } 90 | }) 91 | 92 | export function authorize() { 93 | return functions.https.onRequest(authorizeApp) 94 | } 95 | -------------------------------------------------------------------------------- /src/endpoint/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./authorize" 2 | export * from "./authentication" 3 | export * from "./token" 4 | export * from "./protectedresources" 5 | export * from "./tokeninfo" 6 | -------------------------------------------------------------------------------- /src/endpoint/protectedresources/abstract_protected_resource_endpoint.ts: -------------------------------------------------------------------------------- 1 | import * as express from "express"; 2 | import * as functions from "firebase-functions"; 3 | import {AccessDenied, DefaultAccessTokenFetcherProvider, ProtectedResourceEndpoint, UnknownError, ProtectedResourceEndpointResponse} from "oauth2-nodejs"; 4 | import {CloudFirestoreDataHandlerFactory} from "../../data"; 5 | import {RequestWrapper} from "../../models"; 6 | import {Navigation} from "../../utils"; 7 | 8 | export abstract class AbstractProtectedResourceEndpoint { 9 | 10 | public get endpoint(): functions.HttpsFunction { 11 | return functions.https.onRequest(async (req, resp) => { 12 | const request = new RequestWrapper(req) 13 | const protectedResourceEndpoint = new ProtectedResourceEndpoint() 14 | protectedResourceEndpoint.accessTokenFetcherProvider = new DefaultAccessTokenFetcherProvider() 15 | protectedResourceEndpoint.dataHandlerFactory = new CloudFirestoreDataHandlerFactory() 16 | const result = await protectedResourceEndpoint.handleRequest(request) 17 | if (result.isSuccess()) { 18 | const endpointInfo = result.value 19 | if (this.validateScope(endpointInfo.scope.split(" "))) { 20 | resp.set("Content-Type", "application/json; charset=UTF-8") 21 | try { 22 | const responseBody = await this.handleRequest(req, endpointInfo) 23 | resp.status(200).send(responseBody) 24 | } catch(e) { 25 | Navigation.sendError(resp, new UnknownError(e.toString())) 26 | } 27 | } else { 28 | Navigation.sendError(resp, new AccessDenied("")) 29 | } 30 | } else { 31 | Navigation.sendError(resp, result.error) 32 | } 33 | }) 34 | } 35 | 36 | protected abstract validateScope(scopes: string[]): boolean 37 | 38 | protected abstract handleRequest(req: express.Request, endpointInfo: ProtectedResourceEndpointResponse): Promise 39 | 40 | } -------------------------------------------------------------------------------- /src/endpoint/protectedresources/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./abstract_protected_resource_endpoint" 2 | export * from "./userinfo" 3 | -------------------------------------------------------------------------------- /src/endpoint/protectedresources/userinfo.ts: -------------------------------------------------------------------------------- 1 | import * as admin from "firebase-admin"; 2 | import * as functions from "firebase-functions"; 3 | import * as express from "express"; 4 | import {ProtectedResourceEndpointResponse} from "oauth2-nodejs"; 5 | import {AbstractProtectedResourceEndpoint} from "./abstract_protected_resource_endpoint"; 6 | 7 | export class UserinfoEndpoint extends AbstractProtectedResourceEndpoint { 8 | 9 | protected handleRequest(req: express.Request, endpointInfo: ProtectedResourceEndpointResponse): Promise { 10 | return new Promise((resolve, reject) => { 11 | const auth = admin.auth() 12 | auth.getUser(endpointInfo.userId).then((userRecord) => { 13 | resolve(JSON.stringify({"sub": endpointInfo.userId, "name": userRecord.displayName})) 14 | }).catch(e => { 15 | reject(e) 16 | }) 17 | }) 18 | } 19 | 20 | protected validateScope(scopes: string[]): boolean { 21 | return scopes.indexOf("profile") !== -1 22 | } 23 | 24 | } 25 | 26 | export function userinfo(): functions.HttpsFunction { 27 | return new UserinfoEndpoint().endpoint 28 | } 29 | -------------------------------------------------------------------------------- /src/endpoint/token.ts: -------------------------------------------------------------------------------- 1 | import * as functions from "firebase-functions"; 2 | import {DefaultClientCredentialFetcherProvider, TokenEndpoint} from "oauth2-nodejs"; 3 | import {RequestWrapper} from "../models"; 4 | import {CustomGrantHandlerProvider} from "../granttype"; 5 | import {CloudFirestoreDataHandlerFactory} from "../data"; 6 | 7 | export function token() { 8 | return functions.https.onRequest(async (req, resp) => { 9 | if (req.method === "POST") { 10 | const request = new RequestWrapper(req) 11 | const tokenEndpoint = new TokenEndpoint() 12 | const clientCredentialFetcherProvider = new DefaultClientCredentialFetcherProvider() 13 | tokenEndpoint.grantHandlerProvider = new CustomGrantHandlerProvider(clientCredentialFetcherProvider) 14 | tokenEndpoint.clientCredentialFetcherProvider = clientCredentialFetcherProvider 15 | tokenEndpoint.dataHandlerFactory = new CloudFirestoreDataHandlerFactory() 16 | try { 17 | const tokenEndpointResponse = await tokenEndpoint.handleRequest(request); 18 | resp.set("Content-Type", "application/json; charset=UTF-8") 19 | resp.status(tokenEndpointResponse.code).send(tokenEndpointResponse.body) 20 | } catch(e) { 21 | console.error(e) 22 | resp.status(500).send(e.toString()) 23 | } 24 | } else { 25 | resp.status(405).send("Method not allowed") 26 | } 27 | }) 28 | } 29 | -------------------------------------------------------------------------------- /src/endpoint/tokeninfo.ts: -------------------------------------------------------------------------------- 1 | import * as functions from "firebase-functions"; 2 | import {TokeninfoEndpoint} from "oauth2-nodejs"; 3 | import {RequestWrapper} from "../models"; 4 | import {CloudFirestoreDataHandlerFactory} from "../data"; 5 | 6 | export function tokeninfo() { 7 | return functions.https.onRequest(async (req, resp) => { 8 | if (req.method === "GET") { 9 | const request = new RequestWrapper(req) 10 | const tokeninfoEndpoint = new TokeninfoEndpoint() 11 | tokeninfoEndpoint.dataHandlerFactory = new CloudFirestoreDataHandlerFactory() 12 | try { 13 | const tokeninfoEndpointResponse = await tokeninfoEndpoint.handleRequest(request); 14 | resp.set("Content-Type", "application/json; charset=UTF-8") 15 | resp.status(tokeninfoEndpointResponse.code).send(tokeninfoEndpointResponse.body) 16 | } catch(e) { 17 | console.error(e) 18 | resp.status(500).send(e.toString()) 19 | } 20 | } else { 21 | resp.status(405).send("Method not allowed") 22 | } 23 | }) 24 | } 25 | -------------------------------------------------------------------------------- /src/endpoint/views/consent_view_template.ts: -------------------------------------------------------------------------------- 1 | export interface ConsentViewTemplate { 2 | 3 | provide(): Promise 4 | 5 | } 6 | -------------------------------------------------------------------------------- /src/endpoint/views/default_consent_view_template.ts: -------------------------------------------------------------------------------- 1 | import * as fs from "fs"; 2 | import * as path from "path"; 3 | import {ConsentViewTemplate} from "./consent_view_template"; 4 | 5 | export class DefaultConsentViewTemplate implements ConsentViewTemplate { 6 | 7 | public provide(): Promise { 8 | return new Promise((resolve, reject) => { 9 | fs.readFile(path.join(__dirname, "../../../views/consent.ejs"), "utf8", (err, data) => { 10 | if (err) { 11 | console.error(err) 12 | reject(err) 13 | } else { 14 | resolve(data) 15 | } 16 | }) 17 | }) 18 | } 19 | 20 | } 21 | -------------------------------------------------------------------------------- /src/granttype/custom_grant_handler_provider.ts: -------------------------------------------------------------------------------- 1 | import { 2 | AuthorizationCodeGrantHandler, 3 | ClientCredentialFetcherProvider, ClientCredentialsGrantHandler, 4 | GrantHandler, 5 | GrantHandlerProvider, RefreshTokenGrantHandler 6 | } from "oauth2-nodejs"; 7 | 8 | export class CustomGrantHandlerProvider extends GrantHandlerProvider { 9 | 10 | constructor(clientCredentialFetcherProvider: ClientCredentialFetcherProvider) { 11 | super() 12 | 13 | const handlers = new Map() 14 | 15 | const authorizationCode = new AuthorizationCodeGrantHandler() 16 | authorizationCode.clientCredentialFetcherProvider = clientCredentialFetcherProvider 17 | handlers.set("authorization_code", authorizationCode) 18 | 19 | const refreshToken = new RefreshTokenGrantHandler() 20 | refreshToken.clientCredentialFetcherProvider = clientCredentialFetcherProvider 21 | handlers.set("refresh_token", refreshToken) 22 | 23 | const clientCredentials = new ClientCredentialsGrantHandler() 24 | clientCredentials.clientCredentialFetcherProvider = clientCredentialFetcherProvider 25 | handlers.set("client_credentials", clientCredentials) 26 | 27 | this.handlers = handlers 28 | } 29 | 30 | } 31 | -------------------------------------------------------------------------------- /src/granttype/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./custom_grant_handler_provider" 2 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./data" 2 | export * from "./endpoint" 3 | export * from "./granttype" 4 | export * from "./models" 5 | export * from "./utils" 6 | -------------------------------------------------------------------------------- /src/models/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./request_map" 2 | export * from "./request_wrapper" 3 | -------------------------------------------------------------------------------- /src/models/request_map.ts: -------------------------------------------------------------------------------- 1 | import {Request} from "oauth2-nodejs"; 2 | 3 | export class RequestMap implements Request { 4 | 5 | private _headerMap: Map 6 | private _parameterMap: Map 7 | 8 | constructor() { 9 | this._headerMap = new Map() 10 | this._parameterMap = new Map() 11 | } 12 | 13 | getHeader(name: string): string | undefined { 14 | return this._headerMap.get(name) 15 | } 16 | 17 | setHeader(name: string, value: string): void { 18 | this._headerMap.set(name, value) 19 | } 20 | 21 | getParameter(name: string): string | undefined { 22 | return this._parameterMap.get(name) 23 | } 24 | 25 | setParameter(name: string, value: string): void { 26 | this._parameterMap.set(name, value) 27 | } 28 | 29 | getParameterMap(): Map { 30 | return this._parameterMap 31 | } 32 | 33 | } -------------------------------------------------------------------------------- /src/models/request_wrapper.ts: -------------------------------------------------------------------------------- 1 | import {Request} from "oauth2-nodejs"; 2 | import * as express from "express"; 3 | 4 | export class RequestWrapper implements Request { 5 | 6 | private _original: express.Request 7 | 8 | constructor(original: express.Request) { 9 | this._original = original 10 | } 11 | 12 | getHeader(name: string): string { 13 | return this._original.get(name) || "" 14 | } 15 | 16 | getParameter(name: string): string | undefined { 17 | return this._original.query[name] || this._original.body[name] 18 | } 19 | 20 | getParameterMap(): Map { 21 | const result = new Map() 22 | for (const key in this._original.query) { 23 | if (this._original.body.hasOwnProperty(key)) { 24 | result.set(key, this._original.query[key]) 25 | } 26 | } 27 | for (const key in this._original.body) { 28 | if (this._original.body.hasOwnProperty(key)) { 29 | result.set(key, this._original.body[key]) 30 | } 31 | } 32 | return result 33 | } 34 | 35 | } 36 | -------------------------------------------------------------------------------- /src/utils/configuration.ts: -------------------------------------------------------------------------------- 1 | import {ConsentViewTemplate} from "../endpoint/views/consent_view_template"; 2 | import {DefaultConsentViewTemplate} from "../endpoint/views/default_consent_view_template"; 3 | 4 | export interface ConfigurationParameters { 5 | crypto_auth_token_secret_key_32: string 6 | project_api_key: string 7 | views_authentication_path?: string 8 | views_consent_template?: ConsentViewTemplate 9 | tokens_expires_in?: Map 10 | } 11 | 12 | export class Configuration { 13 | 14 | private static _instance: Configuration 15 | 16 | private _crypto_auth_token_secret_key_32: string | undefined 17 | private _project_apikey: string | undefined 18 | private _view_consent_template: ConsentViewTemplate | undefined 19 | private _tokens_expires_in: Map | undefined 20 | 21 | private constructor() { 22 | } 23 | 24 | public static get instance(): Configuration { 25 | if (this._instance == undefined) { 26 | this._instance = new Configuration() 27 | } 28 | return this._instance 29 | } 30 | 31 | public static init(params: ConfigurationParameters): void { 32 | this.instance._crypto_auth_token_secret_key_32 = params.crypto_auth_token_secret_key_32 33 | this.instance._project_apikey = params.project_api_key 34 | this.instance._view_consent_template = params.views_consent_template 35 | this.instance._tokens_expires_in = params.tokens_expires_in 36 | } 37 | 38 | public get crypto_auth_token_secret_key_32(): string { 39 | if (this._crypto_auth_token_secret_key_32) { 40 | return this._crypto_auth_token_secret_key_32 41 | } else { 42 | throw new Error("crypto_auth_token_secret_key_32 not set") 43 | } 44 | } 45 | 46 | get project_apikey(): string { 47 | if (this._project_apikey) { 48 | return this._project_apikey 49 | } else { 50 | throw new Error("project_api_key not set") 51 | } 52 | } 53 | 54 | get view_consent_template(): ConsentViewTemplate { 55 | if (this._view_consent_template) { 56 | return this._view_consent_template 57 | } else { 58 | return new DefaultConsentViewTemplate() 59 | } 60 | } 61 | 62 | get tokens_expires_in(): Map { 63 | if (this._tokens_expires_in) { 64 | return this._tokens_expires_in 65 | } else { 66 | const result = new Map() 67 | result.set("authorization_code", 86400) 68 | result.set("implicit", 3600) 69 | result.set("password", 86400) 70 | result.set("client_credentials", 86400) 71 | result.set("refresh_token", 86400) 72 | return result 73 | } 74 | } 75 | 76 | } 77 | -------------------------------------------------------------------------------- /src/utils/crypto.ts: -------------------------------------------------------------------------------- 1 | import * as crypto from "crypto"; 2 | import {Configuration} from "./configuration"; 3 | 4 | export class Crypto { 5 | 6 | static encrypt = (text: string): string => { 7 | const iv = crypto.randomBytes(16) 8 | const cipher = crypto.createCipheriv( 9 | "aes-256-cbc", 10 | Buffer.from(Configuration.instance.crypto_auth_token_secret_key_32, "ascii"), 11 | iv) 12 | let encrypted = cipher.update(text) 13 | encrypted = Buffer.concat([encrypted, cipher.final()]) 14 | return `${iv.toString("hex")}:${encrypted.toString("hex")}` 15 | } 16 | 17 | static decrypt = (text: string): string => { 18 | const divided = text.split(":") 19 | const iv = Buffer.from(divided.shift()!, "hex") 20 | const encrypted = Buffer.from(divided.join(":"), "hex") 21 | const decipher = crypto.createDecipheriv( 22 | 'aes-256-cbc', 23 | Buffer.from(Configuration.instance.crypto_auth_token_secret_key_32, "ascii"), 24 | iv); 25 | let decrypted = decipher.update(encrypted); 26 | decrypted = Buffer.concat([decrypted, decipher.final()]); 27 | return decrypted.toString(); 28 | } 29 | 30 | } 31 | -------------------------------------------------------------------------------- /src/utils/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./crypto" 2 | export * from "./navigation" 3 | export * from "./configuration" 4 | -------------------------------------------------------------------------------- /src/utils/navigation.ts: -------------------------------------------------------------------------------- 1 | import * as url from "url"; 2 | import * as querystring from "querystring"; 3 | import * as express from "express"; 4 | import {AuthorizationEndpointResponse, OAuthError, Result} from "oauth2-nodejs"; 5 | 6 | export class Navigation { 7 | 8 | public static redirect(resp: express.Response, uri: string, 9 | parameters?: {[key: string]: string | number}, 10 | fragments?: {[key: string]: string | number}): void { 11 | const targetUrl = url.parse(uri, true) 12 | if (parameters) { 13 | const query = targetUrl.query 14 | Object.keys(parameters).forEach((key: string): void => { 15 | const value: string | number = parameters[key] 16 | query[key] = typeof value === "string" ? value : String(value) 17 | }) 18 | } 19 | if (fragments) { 20 | targetUrl.hash = `#${querystring.stringify(fragments)}` 21 | } 22 | resp.redirect(url.format(targetUrl)) 23 | } 24 | 25 | public static backTo(resp: express.Response, result: Result, redirectUri: string): void { 26 | if (result.isSuccess()) { 27 | const response = result.value 28 | this.redirect(resp, redirectUri, response.query, response.fragment) 29 | } else { 30 | this.redirect(resp, redirectUri, {"error": result.error.getType()}, {}) 31 | } 32 | } 33 | 34 | public static sendError(resp: express.Response, error: OAuthError): void { 35 | resp.set("Content-Type", "application/json; charset=UTF-8") 36 | resp.status(error.code).send(error.toJson()) 37 | } 38 | 39 | } 40 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es6", 4 | "module": "commonjs", 5 | "moduleResolution": "node", 6 | "outDir": "./dist", 7 | "sourceMap": true, 8 | "lib": ["es6"], 9 | "allowSyntheticDefaultImports": false, 10 | "allowUnreachableCode": false, 11 | "allowUnusedLabels": false, 12 | "declaration": true, 13 | "forceConsistentCasingInFileNames": true, 14 | "noFallthroughCasesInSwitch": true, 15 | "noEmitOnError": true, 16 | "noImplicitReturns": true, 17 | "noImplicitAny": true, 18 | "strictNullChecks": true, 19 | "alwaysStrict": true, 20 | "noUnusedLocals": true 21 | }, 22 | "compileOnSave": true, 23 | "include": [ 24 | "src/**/*" 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "rules": { 3 | // -- Strict errors -- 4 | // These lint rules are likely always a good idea. 5 | 6 | // Force function overloads to be declared together. This ensures readers understand APIs. 7 | "adjacent-overload-signatures": true, 8 | 9 | // Do not allow the subtle/obscure comma operator. 10 | "ban-comma-operator": true, 11 | 12 | // Do not allow internal modules or namespaces . These are deprecated in favor of ES6 modules. 13 | "no-namespace": true, 14 | 15 | // Do not allow parameters to be reassigned. To avoid bugs, developers should instead assign new values to new vars. 16 | "no-parameter-reassignment": true, 17 | 18 | // Force the use of ES6-style imports instead of /// imports. 19 | "no-reference": true, 20 | 21 | // Do not allow type assertions that do nothing. This is a big warning that the developer may not understand the 22 | // code currently being edited (they may be incorrectly handling a different type case that does not exist). 23 | "no-unnecessary-type-assertion": true, 24 | 25 | // Disallow nonsensical label usage. 26 | "label-position": true, 27 | 28 | // Disallows the (often typo) syntax if (var1 = var2). Replace with if (var2) { var1 = var2 }. 29 | "no-conditional-assignment": true, 30 | 31 | // Disallows constructors for primitive types (e.g. new Number('123'), though Number('123') is still allowed). 32 | "no-construct": true, 33 | 34 | // Do not allow super() to be called twice in a constructor. 35 | "no-duplicate-super": true, 36 | 37 | // Do not allow the same case to appear more than once in a switch block. 38 | "no-duplicate-switch-case": true, 39 | 40 | // Do not allow a variable to be declared more than once in the same block. Consider function parameters in this 41 | // rule. 42 | "no-duplicate-variable": [true, "check-parameters"], 43 | 44 | // Disallows a variable definition in an inner scope from shadowing a variable in an outer scope. Developers should 45 | // instead use a separate variable name. 46 | "no-shadowed-variable": true, 47 | 48 | // Empty blocks are almost never needed. Allow the one general exception: empty catch blocks. 49 | "no-empty": [true, "allow-empty-catch"], 50 | 51 | // Functions must either be handled directly (e.g. with a catch() handler) or returned to another function. 52 | // This is a major source of errors in Cloud Functions and the team strongly recommends leaving this rule on. 53 | "no-floating-promises": true, 54 | 55 | // Do not allow any imports for modules that are not in package.json. These will almost certainly fail when 56 | // deployed. 57 | "no-implicit-dependencies": false, 58 | 59 | // The 'this' keyword can only be used inside of classes. 60 | "no-invalid-this": true, 61 | 62 | // Do not allow strings to be thrown because they will not include stack traces. Throw Errors instead. 63 | "no-string-throw": true, 64 | 65 | // Disallow control flow statements, such as return, continue, break, and throw in finally blocks. 66 | "no-unsafe-finally": true, 67 | 68 | // Do not allow variables to be used before they are declared. 69 | "no-use-before-declare": true, 70 | 71 | // Expressions must always return a value. Avoids common errors like const myValue = functionReturningVoid(); 72 | "no-void-expression": [true, "ignore-arrow-function-shorthand"], 73 | 74 | // Disallow duplicate imports in the same file. 75 | "no-duplicate-imports": true, 76 | 77 | 78 | // -- Strong Warnings -- 79 | // These rules should almost never be needed, but may be included due to legacy code. 80 | // They are left as a warning to avoid frustration with blocked deploys when the developer 81 | // understand the warning and wants to deploy anyway. 82 | 83 | // Warn when an empty interface is defined. These are generally not useful. 84 | "no-empty-interface": {"severity": "warning"}, 85 | 86 | // Warn when an import will have side effects. 87 | "no-import-side-effect": {"severity": "warning"}, 88 | 89 | // Warn when variables are defined with var. Var has subtle meaning that can lead to bugs. Strongly prefer const for 90 | // most values and let for values that will change. 91 | "no-var-keyword": {"severity": "warning"}, 92 | 93 | // Prefer === and !== over == and !=. The latter operators support overloads that are often accidental. 94 | "triple-equals": {"severity": "warning"}, 95 | 96 | // Warn when using deprecated APIs. 97 | "deprecation": {"severity": "warning"}, 98 | 99 | // -- Light Warnigns -- 100 | // These rules are intended to help developers use better style. Simpler code has fewer bugs. These would be "info" 101 | // if TSLint supported such a level. 102 | 103 | // prefer for( ... of ... ) to an index loop when the index is only used to fetch an object from an array. 104 | // (Even better: check out utils like .map if transforming an array!) 105 | "prefer-for-of": {"severity": "warning"}, 106 | 107 | // Warns if function overloads could be unified into a single function with optional or rest parameters. 108 | "unified-signatures": {"severity": "warning"}, 109 | 110 | // Prefer const for values that will not change. This better documents code. 111 | "prefer-const": {"severity": "warning"}, 112 | 113 | // Multi-line object liiterals and function calls should have a trailing comma. This helps avoid merge conflicts. 114 | "trailing-comma": {"severity": "warning"} 115 | }, 116 | 117 | "defaultSeverity": "error" 118 | } 119 | -------------------------------------------------------------------------------- /views/authentication.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | <%= providerName %> Account Authentication page 8 | 9 | 48 | 49 | 50 |

Now loading...

51 |
52 | 53 | 54 | 55 | 56 |
57 | 58 | 59 | -------------------------------------------------------------------------------- /views/consent.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Authorization page 8 | 9 | 10 |

<%= providerName %> requests the following:

11 |
    12 | <% for (const key of scope.split(" ")) { %> 13 |
  • <%= scopes.get(key) %>
  • 14 | <% } %> 15 |
16 |
17 | 18 | 19 | 20 | 21 |
22 | 23 | 24 | -------------------------------------------------------------------------------- /yarn.lock: -------------------------------------------------------------------------------- 1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. 2 | # yarn lockfile v1 3 | 4 | 5 | "@firebase/app-types@0.3.2": 6 | version "0.3.2" 7 | resolved "https://registry.yarnpkg.com/@firebase/app-types/-/app-types-0.3.2.tgz#a92dc544290e2893bd8c02a81e684dae3d8e7c85" 8 | 9 | "@firebase/app@^0.3.1": 10 | version "0.3.3" 11 | resolved "https://registry.yarnpkg.com/@firebase/app/-/app-0.3.3.tgz#cb8df89495e4409e92ab30c0068b9e0641a6db81" 12 | dependencies: 13 | "@firebase/app-types" "0.3.2" 14 | "@firebase/util" "0.2.1" 15 | dom-storage "2.1.0" 16 | tslib "1.9.0" 17 | xmlhttprequest "1.8.0" 18 | 19 | "@firebase/database-types@0.3.2": 20 | version "0.3.2" 21 | resolved "https://registry.yarnpkg.com/@firebase/database-types/-/database-types-0.3.2.tgz#70611a64dd460e0e253c7427f860d56a1afd86fe" 22 | 23 | "@firebase/database@^0.3.1": 24 | version "0.3.4" 25 | resolved "https://registry.yarnpkg.com/@firebase/database/-/database-0.3.4.tgz#67fd48ed6d8fffc81c6c6f8e68bde70b99ba8ca9" 26 | dependencies: 27 | "@firebase/database-types" "0.3.2" 28 | "@firebase/logger" "0.1.1" 29 | "@firebase/util" "0.2.1" 30 | faye-websocket "0.11.1" 31 | tslib "1.9.0" 32 | 33 | "@firebase/logger@0.1.1": 34 | version "0.1.1" 35 | resolved "https://registry.yarnpkg.com/@firebase/logger/-/logger-0.1.1.tgz#af5df54253286993f4b367c3dabe569c848860d3" 36 | 37 | "@firebase/util@0.2.1": 38 | version "0.2.1" 39 | resolved "https://registry.yarnpkg.com/@firebase/util/-/util-0.2.1.tgz#b59a2fbf14fce21401cbebf776a3e0260b591380" 40 | dependencies: 41 | tslib "1.9.0" 42 | 43 | "@google-cloud/common@^0.17.0": 44 | version "0.17.0" 45 | resolved "https://registry.yarnpkg.com/@google-cloud/common/-/common-0.17.0.tgz#8ef558750db481fc10a13757a49479ab9a1c8c07" 46 | dependencies: 47 | array-uniq "^1.0.3" 48 | arrify "^1.0.1" 49 | concat-stream "^1.6.0" 50 | create-error-class "^3.0.2" 51 | duplexify "^3.5.0" 52 | ent "^2.2.0" 53 | extend "^3.0.1" 54 | google-auto-auth "^0.10.0" 55 | is "^3.2.0" 56 | log-driver "1.2.7" 57 | methmeth "^1.1.0" 58 | modelo "^4.2.0" 59 | request "^2.79.0" 60 | retry-request "^3.0.0" 61 | split-array-stream "^1.0.0" 62 | stream-events "^1.0.1" 63 | string-format-obj "^1.1.0" 64 | through2 "^2.0.3" 65 | 66 | "@google-cloud/common@^0.20.3": 67 | version "0.20.3" 68 | resolved "https://registry.yarnpkg.com/@google-cloud/common/-/common-0.20.3.tgz#639fb9ed07b0e20bdcfa84ebb0838b8cb2068e3b" 69 | dependencies: 70 | "@types/duplexify" "^3.5.0" 71 | "@types/request" "^2.47.0" 72 | arrify "^1.0.1" 73 | axios "^0.18.0" 74 | duplexify "^3.6.0" 75 | ent "^2.2.0" 76 | extend "^3.0.1" 77 | google-auth-library "^1.6.0" 78 | is "^3.2.1" 79 | pify "^3.0.0" 80 | request "^2.87.0" 81 | retry-request "^4.0.0" 82 | split-array-stream "^2.0.0" 83 | stream-events "^1.0.4" 84 | through2 "^2.0.3" 85 | 86 | "@google-cloud/firestore@^0.15.4": 87 | version "0.15.4" 88 | resolved "https://registry.yarnpkg.com/@google-cloud/firestore/-/firestore-0.15.4.tgz#5a23cc7d0c516d0e019e0a8554749b518a0f6723" 89 | dependencies: 90 | "@google-cloud/common" "^0.20.3" 91 | bun "^0.0.12" 92 | deep-equal "^1.0.1" 93 | extend "^3.0.1" 94 | functional-red-black-tree "^1.0.1" 95 | google-gax "^0.17.1" 96 | google-proto-files "^0.16.1" 97 | is "^3.2.1" 98 | lodash.merge "^4.6.1" 99 | pkg-up "^2.0.0" 100 | through2 "^2.0.3" 101 | 102 | "@google-cloud/storage@^1.6.0": 103 | version "1.7.0" 104 | resolved "https://registry.yarnpkg.com/@google-cloud/storage/-/storage-1.7.0.tgz#07bff573d92d5c294db6a04af246688875a8f74b" 105 | dependencies: 106 | "@google-cloud/common" "^0.17.0" 107 | arrify "^1.0.0" 108 | async "^2.0.1" 109 | compressible "^2.0.12" 110 | concat-stream "^1.5.0" 111 | create-error-class "^3.0.2" 112 | duplexify "^3.5.0" 113 | extend "^3.0.0" 114 | gcs-resumable-upload "^0.10.2" 115 | hash-stream-validation "^0.2.1" 116 | is "^3.0.1" 117 | mime "^2.2.0" 118 | mime-types "^2.0.8" 119 | once "^1.3.1" 120 | pumpify "^1.5.1" 121 | request "^2.85.0" 122 | safe-buffer "^5.1.1" 123 | snakeize "^0.1.0" 124 | stream-events "^1.0.1" 125 | through2 "^2.0.0" 126 | xdg-basedir "^3.0.0" 127 | 128 | "@mrmlnc/readdir-enhanced@^2.2.1": 129 | version "2.2.1" 130 | resolved "https://registry.yarnpkg.com/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz#524af240d1a360527b730475ecfa1344aa540dde" 131 | dependencies: 132 | call-me-maybe "^1.0.1" 133 | glob-to-regexp "^0.3.0" 134 | 135 | "@nodelib/fs.stat@^1.0.1": 136 | version "1.1.0" 137 | resolved "https://registry.yarnpkg.com/@nodelib/fs.stat/-/fs.stat-1.1.0.tgz#50c1e2260ac0ed9439a181de3725a0168d59c48a" 138 | 139 | "@protobufjs/aspromise@^1.1.1", "@protobufjs/aspromise@^1.1.2": 140 | version "1.1.2" 141 | resolved "https://registry.yarnpkg.com/@protobufjs/aspromise/-/aspromise-1.1.2.tgz#9b8b0cc663d669a7d8f6f5d0893a14d348f30fbf" 142 | 143 | "@protobufjs/base64@^1.1.2": 144 | version "1.1.2" 145 | resolved "https://registry.yarnpkg.com/@protobufjs/base64/-/base64-1.1.2.tgz#4c85730e59b9a1f1f349047dbf24296034bb2735" 146 | 147 | "@protobufjs/codegen@^2.0.4": 148 | version "2.0.4" 149 | resolved "https://registry.yarnpkg.com/@protobufjs/codegen/-/codegen-2.0.4.tgz#7ef37f0d010fb028ad1ad59722e506d9262815cb" 150 | 151 | "@protobufjs/eventemitter@^1.1.0": 152 | version "1.1.0" 153 | resolved "https://registry.yarnpkg.com/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz#355cbc98bafad5978f9ed095f397621f1d066b70" 154 | 155 | "@protobufjs/fetch@^1.1.0": 156 | version "1.1.0" 157 | resolved "https://registry.yarnpkg.com/@protobufjs/fetch/-/fetch-1.1.0.tgz#ba99fb598614af65700c1619ff06d454b0d84c45" 158 | dependencies: 159 | "@protobufjs/aspromise" "^1.1.1" 160 | "@protobufjs/inquire" "^1.1.0" 161 | 162 | "@protobufjs/float@^1.0.2": 163 | version "1.0.2" 164 | resolved "https://registry.yarnpkg.com/@protobufjs/float/-/float-1.0.2.tgz#5e9e1abdcb73fc0a7cb8b291df78c8cbd97b87d1" 165 | 166 | "@protobufjs/inquire@^1.1.0": 167 | version "1.1.0" 168 | resolved "https://registry.yarnpkg.com/@protobufjs/inquire/-/inquire-1.1.0.tgz#ff200e3e7cf2429e2dcafc1140828e8cc638f089" 169 | 170 | "@protobufjs/path@^1.1.2": 171 | version "1.1.2" 172 | resolved "https://registry.yarnpkg.com/@protobufjs/path/-/path-1.1.2.tgz#6cc2b20c5c9ad6ad0dccfd21ca7673d8d7fbf68d" 173 | 174 | "@protobufjs/pool@^1.1.0": 175 | version "1.1.0" 176 | resolved "https://registry.yarnpkg.com/@protobufjs/pool/-/pool-1.1.0.tgz#09fd15f2d6d3abfa9b65bc366506d6ad7846ff54" 177 | 178 | "@protobufjs/utf8@^1.1.0": 179 | version "1.1.0" 180 | resolved "https://registry.yarnpkg.com/@protobufjs/utf8/-/utf8-1.1.0.tgz#a777360b5b39a1a2e5106f8e858f2fd2d060c570" 181 | 182 | "@types/body-parser@*": 183 | version "1.17.0" 184 | resolved "https://registry.yarnpkg.com/@types/body-parser/-/body-parser-1.17.0.tgz#9f5c9d9bd04bb54be32d5eb9fc0d8c974e6cf58c" 185 | dependencies: 186 | "@types/connect" "*" 187 | "@types/node" "*" 188 | 189 | "@types/caseless@*": 190 | version "0.12.1" 191 | resolved "https://registry.yarnpkg.com/@types/caseless/-/caseless-0.12.1.tgz#9794c69c8385d0192acc471a540d1f8e0d16218a" 192 | 193 | "@types/connect@*": 194 | version "3.4.32" 195 | resolved "https://registry.yarnpkg.com/@types/connect/-/connect-3.4.32.tgz#aa0e9616b9435ccad02bc52b5b454ffc2c70ba28" 196 | dependencies: 197 | "@types/node" "*" 198 | 199 | "@types/cors@^2.8.1": 200 | version "2.8.4" 201 | resolved "https://registry.yarnpkg.com/@types/cors/-/cors-2.8.4.tgz#50991a759a29c0b89492751008c6af7a7c8267b0" 202 | dependencies: 203 | "@types/express" "*" 204 | 205 | "@types/duplexify@^3.5.0": 206 | version "3.5.0" 207 | resolved "https://registry.yarnpkg.com/@types/duplexify/-/duplexify-3.5.0.tgz#c1e8a2c4e05f2a5545c61c31283b76f92d48b007" 208 | dependencies: 209 | "@types/node" "*" 210 | 211 | "@types/ejs@^2.6.0": 212 | version "2.6.0" 213 | resolved "https://registry.yarnpkg.com/@types/ejs/-/ejs-2.6.0.tgz#56502bca6e2e1b4cf9351918ca76cdaa93fe3b6c" 214 | 215 | "@types/events@*": 216 | version "1.2.0" 217 | resolved "https://registry.yarnpkg.com/@types/events/-/events-1.2.0.tgz#81a6731ce4df43619e5c8c945383b3e62a89ea86" 218 | 219 | "@types/express-serve-static-core@*": 220 | version "4.16.0" 221 | resolved "https://registry.yarnpkg.com/@types/express-serve-static-core/-/express-serve-static-core-4.16.0.tgz#fdfe777594ddc1fe8eb8eccce52e261b496e43e7" 222 | dependencies: 223 | "@types/events" "*" 224 | "@types/node" "*" 225 | "@types/range-parser" "*" 226 | 227 | "@types/express@*", "@types/express@^4.11.1", "@types/express@^4.16.0": 228 | version "4.16.0" 229 | resolved "https://registry.yarnpkg.com/@types/express/-/express-4.16.0.tgz#6d8bc42ccaa6f35cf29a2b7c3333cb47b5a32a19" 230 | dependencies: 231 | "@types/body-parser" "*" 232 | "@types/express-serve-static-core" "*" 233 | "@types/serve-static" "*" 234 | 235 | "@types/form-data@*": 236 | version "2.2.1" 237 | resolved "https://registry.yarnpkg.com/@types/form-data/-/form-data-2.2.1.tgz#ee2b3b8eaa11c0938289953606b745b738c54b1e" 238 | dependencies: 239 | "@types/node" "*" 240 | 241 | "@types/google-cloud__storage@^1.1.7": 242 | version "1.7.0" 243 | resolved "https://registry.yarnpkg.com/@types/google-cloud__storage/-/google-cloud__storage-1.7.0.tgz#60abb5669f78ed07abb625f556d1f396ee84724e" 244 | dependencies: 245 | "@types/node" "*" 246 | "@types/request" "*" 247 | 248 | "@types/jsonwebtoken@^7.2.6": 249 | version "7.2.8" 250 | resolved "https://registry.yarnpkg.com/@types/jsonwebtoken/-/jsonwebtoken-7.2.8.tgz#8d199dab4ddb5bba3234f8311b804d2027af2b3a" 251 | dependencies: 252 | "@types/node" "*" 253 | 254 | "@types/lodash@^4.14.34": 255 | version "4.14.113" 256 | resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.113.tgz#1d1cb063f17fec4cc46f1a90d978ebf441113061" 257 | 258 | "@types/long@^4.0.0": 259 | version "4.0.0" 260 | resolved "https://registry.yarnpkg.com/@types/long/-/long-4.0.0.tgz#719551d2352d301ac8b81db732acb6bdc28dbdef" 261 | 262 | "@types/mime@*": 263 | version "2.0.0" 264 | resolved "https://registry.yarnpkg.com/@types/mime/-/mime-2.0.0.tgz#5a7306e367c539b9f6543499de8dd519fac37a8b" 265 | 266 | "@types/node@*", "@types/node@^10.1.0", "@types/node@^10.5.2": 267 | version "10.5.3" 268 | resolved "https://registry.yarnpkg.com/@types/node/-/node-10.5.3.tgz#5bcfaf088ad17894232012877669634c06b20cc5" 269 | 270 | "@types/node@^8.0.53": 271 | version "8.10.21" 272 | resolved "https://registry.yarnpkg.com/@types/node/-/node-8.10.21.tgz#12b3f2359b27aa05a45d886c8ba1eb8d1a77e285" 273 | 274 | "@types/range-parser@*": 275 | version "1.2.2" 276 | resolved "https://registry.yarnpkg.com/@types/range-parser/-/range-parser-1.2.2.tgz#fa8e1ad1d474688a757140c91de6dace6f4abc8d" 277 | 278 | "@types/request@*", "@types/request@^2.47.0": 279 | version "2.47.1" 280 | resolved "https://registry.yarnpkg.com/@types/request/-/request-2.47.1.tgz#25410d3afbdac04c91a94ad9efc9824100735824" 281 | dependencies: 282 | "@types/caseless" "*" 283 | "@types/form-data" "*" 284 | "@types/node" "*" 285 | "@types/tough-cookie" "*" 286 | 287 | "@types/serve-static@*": 288 | version "1.13.2" 289 | resolved "https://registry.yarnpkg.com/@types/serve-static/-/serve-static-1.13.2.tgz#f5ac4d7a6420a99a6a45af4719f4dcd8cd907a48" 290 | dependencies: 291 | "@types/express-serve-static-core" "*" 292 | "@types/mime" "*" 293 | 294 | "@types/tough-cookie@*": 295 | version "2.3.3" 296 | resolved "https://registry.yarnpkg.com/@types/tough-cookie/-/tough-cookie-2.3.3.tgz#7f226d67d654ec9070e755f46daebf014628e9d9" 297 | 298 | abbrev@1: 299 | version "1.1.1" 300 | resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.1.tgz#f8f2c887ad10bf67f634f005b6987fed3179aac8" 301 | 302 | accepts@~1.3.5: 303 | version "1.3.5" 304 | resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.5.tgz#eb777df6011723a3b14e8a72c0805c8e86746bd2" 305 | dependencies: 306 | mime-types "~2.1.18" 307 | negotiator "0.6.1" 308 | 309 | acorn-es7-plugin@^1.0.12: 310 | version "1.1.7" 311 | resolved "https://registry.yarnpkg.com/acorn-es7-plugin/-/acorn-es7-plugin-1.1.7.tgz#f2ee1f3228a90eead1245f9ab1922eb2e71d336b" 312 | 313 | acorn@^5.0.0: 314 | version "5.7.1" 315 | resolved "https://registry.yarnpkg.com/acorn/-/acorn-5.7.1.tgz#f095829297706a7c9776958c0afc8930a9b9d9d8" 316 | 317 | ajv@^5.1.0: 318 | version "5.5.2" 319 | resolved "https://registry.yarnpkg.com/ajv/-/ajv-5.5.2.tgz#73b5eeca3fab653e3d3f9422b341ad42205dc965" 320 | dependencies: 321 | co "^4.6.0" 322 | fast-deep-equal "^1.0.0" 323 | fast-json-stable-stringify "^2.0.0" 324 | json-schema-traverse "^0.3.0" 325 | 326 | ansi-regex@^2.0.0: 327 | version "2.1.1" 328 | resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df" 329 | 330 | ansi-regex@^3.0.0: 331 | version "3.0.0" 332 | resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-3.0.0.tgz#ed0317c322064f79466c02966bddb605ab37d998" 333 | 334 | ansi-styles@^2.2.1: 335 | version "2.2.1" 336 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-2.2.1.tgz#b432dd3358b634cf75e1e4664368240533c1ddbe" 337 | 338 | ansi-styles@^3.2.1: 339 | version "3.2.1" 340 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d" 341 | dependencies: 342 | color-convert "^1.9.0" 343 | 344 | aproba@^1.0.3: 345 | version "1.2.0" 346 | resolved "https://registry.yarnpkg.com/aproba/-/aproba-1.2.0.tgz#6802e6264efd18c790a1b0d517f0f2627bf2c94a" 347 | 348 | are-we-there-yet@~1.1.2: 349 | version "1.1.5" 350 | resolved "https://registry.yarnpkg.com/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz#4b35c2944f062a8bfcda66410760350fe9ddfc21" 351 | dependencies: 352 | delegates "^1.0.0" 353 | readable-stream "^2.0.6" 354 | 355 | argparse@^1.0.7: 356 | version "1.0.10" 357 | resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.10.tgz#bcd6791ea5ae09725e17e5ad988134cd40b3d911" 358 | dependencies: 359 | sprintf-js "~1.0.2" 360 | 361 | arr-diff@^4.0.0: 362 | version "4.0.0" 363 | resolved "https://registry.yarnpkg.com/arr-diff/-/arr-diff-4.0.0.tgz#d6461074febfec71e7e15235761a329a5dc7c520" 364 | 365 | arr-flatten@^1.1.0: 366 | version "1.1.0" 367 | resolved "https://registry.yarnpkg.com/arr-flatten/-/arr-flatten-1.1.0.tgz#36048bbff4e7b47e136644316c99669ea5ae91f1" 368 | 369 | arr-union@^3.1.0: 370 | version "3.1.0" 371 | resolved "https://registry.yarnpkg.com/arr-union/-/arr-union-3.1.0.tgz#e39b09aea9def866a8f206e288af63919bae39c4" 372 | 373 | array-filter@^1.0.0: 374 | version "1.0.0" 375 | resolved "https://registry.yarnpkg.com/array-filter/-/array-filter-1.0.0.tgz#baf79e62e6ef4c2a4c0b831232daffec251f9d83" 376 | 377 | array-flatten@1.1.1: 378 | version "1.1.1" 379 | resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2" 380 | 381 | array-union@^1.0.1: 382 | version "1.0.2" 383 | resolved "https://registry.yarnpkg.com/array-union/-/array-union-1.0.2.tgz#9a34410e4f4e3da23dea375be5be70f24778ec39" 384 | dependencies: 385 | array-uniq "^1.0.1" 386 | 387 | array-uniq@^1.0.1, array-uniq@^1.0.3: 388 | version "1.0.3" 389 | resolved "https://registry.yarnpkg.com/array-uniq/-/array-uniq-1.0.3.tgz#af6ac877a25cc7f74e058894753858dfdb24fdb6" 390 | 391 | array-unique@^0.3.2: 392 | version "0.3.2" 393 | resolved "https://registry.yarnpkg.com/array-unique/-/array-unique-0.3.2.tgz#a894b75d4bc4f6cd679ef3244a9fd8f46ae2d428" 394 | 395 | arrify@^1.0.0, arrify@^1.0.1: 396 | version "1.0.1" 397 | resolved "https://registry.yarnpkg.com/arrify/-/arrify-1.0.1.tgz#898508da2226f380df904728456849c1501a4b0d" 398 | 399 | ascli@~1: 400 | version "1.0.1" 401 | resolved "https://registry.yarnpkg.com/ascli/-/ascli-1.0.1.tgz#bcfa5974a62f18e81cabaeb49732ab4a88f906bc" 402 | dependencies: 403 | colour "~0.7.1" 404 | optjs "~3.2.2" 405 | 406 | asn1@~0.2.3: 407 | version "0.2.3" 408 | resolved "https://registry.yarnpkg.com/asn1/-/asn1-0.2.3.tgz#dac8787713c9966849fc8180777ebe9c1ddf3b86" 409 | 410 | assert-plus@1.0.0, assert-plus@^1.0.0: 411 | version "1.0.0" 412 | resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525" 413 | 414 | assign-symbols@^1.0.0: 415 | version "1.0.0" 416 | resolved "https://registry.yarnpkg.com/assign-symbols/-/assign-symbols-1.0.0.tgz#59667f41fadd4f20ccbc2bb96b8d4f7f78ec0367" 417 | 418 | async@^2.0.1, async@^2.3.0, async@^2.4.0: 419 | version "2.6.1" 420 | resolved "https://registry.yarnpkg.com/async/-/async-2.6.1.tgz#b245a23ca71930044ec53fa46aa00a3e87c6a610" 421 | dependencies: 422 | lodash "^4.17.10" 423 | 424 | asynckit@^0.4.0: 425 | version "0.4.0" 426 | resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" 427 | 428 | atob@^2.1.1: 429 | version "2.1.1" 430 | resolved "https://registry.yarnpkg.com/atob/-/atob-2.1.1.tgz#ae2d5a729477f289d60dd7f96a6314a22dd6c22a" 431 | 432 | aws-sign2@~0.7.0: 433 | version "0.7.0" 434 | resolved "https://registry.yarnpkg.com/aws-sign2/-/aws-sign2-0.7.0.tgz#b46e890934a9591f2d2f6f86d7e6a9f1b3fe76a8" 435 | 436 | aws4@^1.6.0: 437 | version "1.7.0" 438 | resolved "https://registry.yarnpkg.com/aws4/-/aws4-1.7.0.tgz#d4d0e9b9dbfca77bf08eeb0a8a471550fe39e289" 439 | 440 | axios@^0.18.0: 441 | version "0.18.0" 442 | resolved "https://registry.yarnpkg.com/axios/-/axios-0.18.0.tgz#32d53e4851efdc0a11993b6cd000789d70c05102" 443 | dependencies: 444 | follow-redirects "^1.3.0" 445 | is-buffer "^1.1.5" 446 | 447 | babel-code-frame@^6.22.0: 448 | version "6.26.0" 449 | resolved "https://registry.yarnpkg.com/babel-code-frame/-/babel-code-frame-6.26.0.tgz#63fd43f7dc1e3bb7ce35947db8fe369a3f58c74b" 450 | dependencies: 451 | chalk "^1.1.3" 452 | esutils "^2.0.2" 453 | js-tokens "^3.0.2" 454 | 455 | balanced-match@^1.0.0: 456 | version "1.0.0" 457 | resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz#89b4d199ab2bee49de164ea02b89ce462d71b767" 458 | 459 | base@^0.11.1: 460 | version "0.11.2" 461 | resolved "https://registry.yarnpkg.com/base/-/base-0.11.2.tgz#7bde5ced145b6d551a90db87f83c558b4eb48a8f" 462 | dependencies: 463 | cache-base "^1.0.1" 464 | class-utils "^0.3.5" 465 | component-emitter "^1.2.1" 466 | define-property "^1.0.0" 467 | isobject "^3.0.1" 468 | mixin-deep "^1.2.0" 469 | pascalcase "^0.1.1" 470 | 471 | bcrypt-pbkdf@^1.0.0: 472 | version "1.0.2" 473 | resolved "https://registry.yarnpkg.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz#a4301d389b6a43f9b67ff3ca11a3f6637e360e9e" 474 | dependencies: 475 | tweetnacl "^0.14.3" 476 | 477 | body-parser@1.18.2: 478 | version "1.18.2" 479 | resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.18.2.tgz#87678a19d84b47d859b83199bd59bce222b10454" 480 | dependencies: 481 | bytes "3.0.0" 482 | content-type "~1.0.4" 483 | debug "2.6.9" 484 | depd "~1.1.1" 485 | http-errors "~1.6.2" 486 | iconv-lite "0.4.19" 487 | on-finished "~2.3.0" 488 | qs "6.5.1" 489 | raw-body "2.3.2" 490 | type-is "~1.6.15" 491 | 492 | brace-expansion@^1.1.7: 493 | version "1.1.11" 494 | resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" 495 | dependencies: 496 | balanced-match "^1.0.0" 497 | concat-map "0.0.1" 498 | 499 | braces@^2.3.1: 500 | version "2.3.2" 501 | resolved "https://registry.yarnpkg.com/braces/-/braces-2.3.2.tgz#5979fd3f14cd531565e5fa2df1abfff1dfaee729" 502 | dependencies: 503 | arr-flatten "^1.1.0" 504 | array-unique "^0.3.2" 505 | extend-shallow "^2.0.1" 506 | fill-range "^4.0.0" 507 | isobject "^3.0.1" 508 | repeat-element "^1.1.2" 509 | snapdragon "^0.8.1" 510 | snapdragon-node "^2.0.1" 511 | split-string "^3.0.2" 512 | to-regex "^3.0.1" 513 | 514 | buffer-equal-constant-time@1.0.1: 515 | version "1.0.1" 516 | resolved "https://registry.yarnpkg.com/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz#f8e71132f7ffe6e01a5c9697a4c6f3e48d5cc819" 517 | 518 | buffer-from@^1.0.0: 519 | version "1.1.0" 520 | resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.0.tgz#87fcaa3a298358e0ade6e442cfce840740d1ad04" 521 | 522 | builtin-modules@^1.1.1: 523 | version "1.1.1" 524 | resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-1.1.1.tgz#270f076c5a72c02f5b65a47df94c5fe3a278892f" 525 | 526 | bun@^0.0.12: 527 | version "0.0.12" 528 | resolved "https://registry.yarnpkg.com/bun/-/bun-0.0.12.tgz#d54fae69f895557f275423bc14b404030b20a5fc" 529 | dependencies: 530 | readable-stream "~1.0.32" 531 | 532 | bytebuffer@~5: 533 | version "5.0.1" 534 | resolved "https://registry.yarnpkg.com/bytebuffer/-/bytebuffer-5.0.1.tgz#582eea4b1a873b6d020a48d58df85f0bba6cfddd" 535 | dependencies: 536 | long "~3" 537 | 538 | bytes@3.0.0: 539 | version "3.0.0" 540 | resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.0.0.tgz#d32815404d689699f85a4ea4fa8755dd13a96048" 541 | 542 | cache-base@^1.0.1: 543 | version "1.0.1" 544 | resolved "https://registry.yarnpkg.com/cache-base/-/cache-base-1.0.1.tgz#0a7f46416831c8b662ee36fe4e7c59d76f666ab2" 545 | dependencies: 546 | collection-visit "^1.0.0" 547 | component-emitter "^1.2.1" 548 | get-value "^2.0.6" 549 | has-value "^1.0.0" 550 | isobject "^3.0.1" 551 | set-value "^2.0.0" 552 | to-object-path "^0.3.0" 553 | union-value "^1.0.0" 554 | unset-value "^1.0.0" 555 | 556 | call-me-maybe@^1.0.1: 557 | version "1.0.1" 558 | resolved "https://registry.yarnpkg.com/call-me-maybe/-/call-me-maybe-1.0.1.tgz#26d208ea89e37b5cbde60250a15f031c16a4d66b" 559 | 560 | call-signature@0.0.2: 561 | version "0.0.2" 562 | resolved "https://registry.yarnpkg.com/call-signature/-/call-signature-0.0.2.tgz#a84abc825a55ef4cb2b028bd74e205a65b9a4996" 563 | 564 | camelcase@^2.0.1: 565 | version "2.1.1" 566 | resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-2.1.1.tgz#7c1d16d679a1bbe59ca02cacecfb011e201f5a1f" 567 | 568 | capture-stack-trace@^1.0.0: 569 | version "1.0.0" 570 | resolved "https://registry.yarnpkg.com/capture-stack-trace/-/capture-stack-trace-1.0.0.tgz#4a6fa07399c26bba47f0b2496b4d0fb408c5550d" 571 | 572 | caseless@~0.12.0: 573 | version "0.12.0" 574 | resolved "https://registry.yarnpkg.com/caseless/-/caseless-0.12.0.tgz#1b681c21ff84033c826543090689420d187151dc" 575 | 576 | chalk@^1.1.3: 577 | version "1.1.3" 578 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98" 579 | dependencies: 580 | ansi-styles "^2.2.1" 581 | escape-string-regexp "^1.0.2" 582 | has-ansi "^2.0.0" 583 | strip-ansi "^3.0.0" 584 | supports-color "^2.0.0" 585 | 586 | chalk@^2.3.0: 587 | version "2.4.1" 588 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.1.tgz#18c49ab16a037b6eb0152cc83e3471338215b66e" 589 | dependencies: 590 | ansi-styles "^3.2.1" 591 | escape-string-regexp "^1.0.5" 592 | supports-color "^5.3.0" 593 | 594 | chownr@^1.0.1: 595 | version "1.0.1" 596 | resolved "https://registry.yarnpkg.com/chownr/-/chownr-1.0.1.tgz#e2a75042a9551908bebd25b8523d5f9769d79181" 597 | 598 | class-utils@^0.3.5: 599 | version "0.3.6" 600 | resolved "https://registry.yarnpkg.com/class-utils/-/class-utils-0.3.6.tgz#f93369ae8b9a7ce02fd41faad0ca83033190c463" 601 | dependencies: 602 | arr-union "^3.1.0" 603 | define-property "^0.2.5" 604 | isobject "^3.0.0" 605 | static-extend "^0.1.1" 606 | 607 | cliui@^3.0.3: 608 | version "3.2.0" 609 | resolved "https://registry.yarnpkg.com/cliui/-/cliui-3.2.0.tgz#120601537a916d29940f934da3b48d585a39213d" 610 | dependencies: 611 | string-width "^1.0.1" 612 | strip-ansi "^3.0.1" 613 | wrap-ansi "^2.0.0" 614 | 615 | co@^4.6.0: 616 | version "4.6.0" 617 | resolved "https://registry.yarnpkg.com/co/-/co-4.6.0.tgz#6ea6bdf3d853ae54ccb8e47bfa0bf3f9031fb184" 618 | 619 | code-point-at@^1.0.0: 620 | version "1.1.0" 621 | resolved "https://registry.yarnpkg.com/code-point-at/-/code-point-at-1.1.0.tgz#0d070b4d043a5bea33a2f1a40e2edb3d9a4ccf77" 622 | 623 | collection-visit@^1.0.0: 624 | version "1.0.0" 625 | resolved "https://registry.yarnpkg.com/collection-visit/-/collection-visit-1.0.0.tgz#4bc0373c164bc3291b4d368c829cf1a80a59dca0" 626 | dependencies: 627 | map-visit "^1.0.0" 628 | object-visit "^1.0.0" 629 | 630 | color-convert@^1.9.0: 631 | version "1.9.2" 632 | resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.2.tgz#49881b8fba67df12a96bdf3f56c0aab9e7913147" 633 | dependencies: 634 | color-name "1.1.1" 635 | 636 | color-name@1.1.1: 637 | version "1.1.1" 638 | resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.1.tgz#4b1415304cf50028ea81643643bd82ea05803689" 639 | 640 | colour@~0.7.1: 641 | version "0.7.1" 642 | resolved "https://registry.yarnpkg.com/colour/-/colour-0.7.1.tgz#9cb169917ec5d12c0736d3e8685746df1cadf778" 643 | 644 | combined-stream@1.0.6, combined-stream@~1.0.5: 645 | version "1.0.6" 646 | resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.6.tgz#723e7df6e801ac5613113a7e445a9b69cb632818" 647 | dependencies: 648 | delayed-stream "~1.0.0" 649 | 650 | commander@^2.12.1: 651 | version "2.16.0" 652 | resolved "https://registry.yarnpkg.com/commander/-/commander-2.16.0.tgz#f16390593996ceb4f3eeb020b31d78528f7f8a50" 653 | 654 | component-emitter@^1.2.1: 655 | version "1.2.1" 656 | resolved "https://registry.yarnpkg.com/component-emitter/-/component-emitter-1.2.1.tgz#137918d6d78283f7df7a6b7c5a63e140e69425e6" 657 | 658 | compressible@^2.0.12: 659 | version "2.0.14" 660 | resolved "https://registry.yarnpkg.com/compressible/-/compressible-2.0.14.tgz#326c5f507fbb055f54116782b969a81b67a29da7" 661 | dependencies: 662 | mime-db ">= 1.34.0 < 2" 663 | 664 | concat-map@0.0.1: 665 | version "0.0.1" 666 | resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" 667 | 668 | concat-stream@^1.5.0, concat-stream@^1.6.0: 669 | version "1.6.2" 670 | resolved "https://registry.yarnpkg.com/concat-stream/-/concat-stream-1.6.2.tgz#904bdf194cd3122fc675c77fc4ac3d4ff0fd1a34" 671 | dependencies: 672 | buffer-from "^1.0.0" 673 | inherits "^2.0.3" 674 | readable-stream "^2.2.2" 675 | typedarray "^0.0.6" 676 | 677 | configstore@^3.1.2: 678 | version "3.1.2" 679 | resolved "https://registry.yarnpkg.com/configstore/-/configstore-3.1.2.tgz#c6f25defaeef26df12dd33414b001fe81a543f8f" 680 | dependencies: 681 | dot-prop "^4.1.0" 682 | graceful-fs "^4.1.2" 683 | make-dir "^1.0.0" 684 | unique-string "^1.0.0" 685 | write-file-atomic "^2.0.0" 686 | xdg-basedir "^3.0.0" 687 | 688 | console-control-strings@^1.0.0, console-control-strings@~1.1.0: 689 | version "1.1.0" 690 | resolved "https://registry.yarnpkg.com/console-control-strings/-/console-control-strings-1.1.0.tgz#3d7cf4464db6446ea644bf4b39507f9851008e8e" 691 | 692 | content-disposition@0.5.2: 693 | version "0.5.2" 694 | resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.2.tgz#0cf68bb9ddf5f2be7961c3a85178cb85dba78cb4" 695 | 696 | content-type@~1.0.4: 697 | version "1.0.4" 698 | resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.4.tgz#e138cc75e040c727b1966fe5e5f8c9aee256fe3b" 699 | 700 | cookie-signature@1.0.6: 701 | version "1.0.6" 702 | resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c" 703 | 704 | cookie@0.3.1: 705 | version "0.3.1" 706 | resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.3.1.tgz#e7e0a1f9ef43b4c8ba925c5c5a96e806d16873bb" 707 | 708 | copy-descriptor@^0.1.0: 709 | version "0.1.1" 710 | resolved "https://registry.yarnpkg.com/copy-descriptor/-/copy-descriptor-0.1.1.tgz#676f6eb3c39997c2ee1ac3a924fd6124748f578d" 711 | 712 | core-js@^2.0.0: 713 | version "2.5.7" 714 | resolved "https://registry.yarnpkg.com/core-js/-/core-js-2.5.7.tgz#f972608ff0cead68b841a16a932d0b183791814e" 715 | 716 | core-util-is@1.0.2, core-util-is@~1.0.0: 717 | version "1.0.2" 718 | resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" 719 | 720 | cors@^2.8.4: 721 | version "2.8.4" 722 | resolved "https://registry.yarnpkg.com/cors/-/cors-2.8.4.tgz#2bd381f2eb201020105cd50ea59da63090694686" 723 | dependencies: 724 | object-assign "^4" 725 | vary "^1" 726 | 727 | create-error-class@^3.0.2: 728 | version "3.0.2" 729 | resolved "https://registry.yarnpkg.com/create-error-class/-/create-error-class-3.0.2.tgz#06be7abef947a3f14a30fd610671d401bca8b7b6" 730 | dependencies: 731 | capture-stack-trace "^1.0.0" 732 | 733 | crypto-random-string@^1.0.0: 734 | version "1.0.0" 735 | resolved "https://registry.yarnpkg.com/crypto-random-string/-/crypto-random-string-1.0.0.tgz#a230f64f568310e1498009940790ec99545bca7e" 736 | 737 | dashdash@^1.12.0: 738 | version "1.14.1" 739 | resolved "https://registry.yarnpkg.com/dashdash/-/dashdash-1.14.1.tgz#853cfa0f7cbe2fed5de20326b8dd581035f6e2f0" 740 | dependencies: 741 | assert-plus "^1.0.0" 742 | 743 | debug@2.6.9, debug@^2.1.2, debug@^2.2.0, debug@^2.3.3: 744 | version "2.6.9" 745 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f" 746 | dependencies: 747 | ms "2.0.0" 748 | 749 | debug@^3.1.0: 750 | version "3.1.0" 751 | resolved "https://registry.yarnpkg.com/debug/-/debug-3.1.0.tgz#5bb5a0672628b64149566ba16819e61518c67261" 752 | dependencies: 753 | ms "2.0.0" 754 | 755 | decamelize@^1.1.1: 756 | version "1.2.0" 757 | resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290" 758 | 759 | decode-uri-component@^0.2.0: 760 | version "0.2.0" 761 | resolved "https://registry.yarnpkg.com/decode-uri-component/-/decode-uri-component-0.2.0.tgz#eb3913333458775cb84cd1a1fae062106bb87545" 762 | 763 | deep-equal@^1.0.1: 764 | version "1.0.1" 765 | resolved "https://registry.yarnpkg.com/deep-equal/-/deep-equal-1.0.1.tgz#f5d260292b660e084eff4cdbc9f08ad3247448b5" 766 | 767 | deep-extend@^0.6.0: 768 | version "0.6.0" 769 | resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.6.0.tgz#c4fa7c95404a17a9c3e8ca7e1537312b736330ac" 770 | 771 | define-properties@^1.1.2: 772 | version "1.1.2" 773 | resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.2.tgz#83a73f2fea569898fb737193c8f873caf6d45c94" 774 | dependencies: 775 | foreach "^2.0.5" 776 | object-keys "^1.0.8" 777 | 778 | define-property@^0.2.5: 779 | version "0.2.5" 780 | resolved "https://registry.yarnpkg.com/define-property/-/define-property-0.2.5.tgz#c35b1ef918ec3c990f9a5bc57be04aacec5c8116" 781 | dependencies: 782 | is-descriptor "^0.1.0" 783 | 784 | define-property@^1.0.0: 785 | version "1.0.0" 786 | resolved "https://registry.yarnpkg.com/define-property/-/define-property-1.0.0.tgz#769ebaaf3f4a63aad3af9e8d304c9bbe79bfb0e6" 787 | dependencies: 788 | is-descriptor "^1.0.0" 789 | 790 | define-property@^2.0.2: 791 | version "2.0.2" 792 | resolved "https://registry.yarnpkg.com/define-property/-/define-property-2.0.2.tgz#d459689e8d654ba77e02a817f8710d702cb16e9d" 793 | dependencies: 794 | is-descriptor "^1.0.2" 795 | isobject "^3.0.1" 796 | 797 | delayed-stream@~1.0.0: 798 | version "1.0.0" 799 | resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" 800 | 801 | delegates@^1.0.0: 802 | version "1.0.0" 803 | resolved "https://registry.yarnpkg.com/delegates/-/delegates-1.0.0.tgz#84c6e159b81904fdca59a0ef44cd870d31250f9a" 804 | 805 | depd@1.1.1: 806 | version "1.1.1" 807 | resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.1.tgz#5783b4e1c459f06fa5ca27f991f3d06e7a310359" 808 | 809 | depd@~1.1.1, depd@~1.1.2: 810 | version "1.1.2" 811 | resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.2.tgz#9bcd52e14c097763e749b274c4346ed2e560b5a9" 812 | 813 | destroy@~1.0.4: 814 | version "1.0.4" 815 | resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.0.4.tgz#978857442c44749e4206613e37946205826abd80" 816 | 817 | detect-libc@^1.0.2: 818 | version "1.0.3" 819 | resolved "https://registry.yarnpkg.com/detect-libc/-/detect-libc-1.0.3.tgz#fa137c4bd698edf55cd5cd02ac559f91a4c4ba9b" 820 | 821 | diff-match-patch@^1.0.0: 822 | version "1.0.1" 823 | resolved "https://registry.yarnpkg.com/diff-match-patch/-/diff-match-patch-1.0.1.tgz#d5f880213d82fbc124d2b95111fb3c033dbad7fa" 824 | 825 | diff@^3.2.0: 826 | version "3.5.0" 827 | resolved "https://registry.yarnpkg.com/diff/-/diff-3.5.0.tgz#800c0dd1e0a8bfbc95835c202ad220fe317e5a12" 828 | 829 | dir-glob@^2.0.0: 830 | version "2.0.0" 831 | resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-2.0.0.tgz#0b205d2b6aef98238ca286598a8204d29d0a0034" 832 | dependencies: 833 | arrify "^1.0.1" 834 | path-type "^3.0.0" 835 | 836 | dom-storage@2.1.0: 837 | version "2.1.0" 838 | resolved "https://registry.yarnpkg.com/dom-storage/-/dom-storage-2.1.0.tgz#00fb868bc9201357ea243c7bcfd3304c1e34ea39" 839 | 840 | dot-prop@^4.1.0: 841 | version "4.2.0" 842 | resolved "https://registry.yarnpkg.com/dot-prop/-/dot-prop-4.2.0.tgz#1f19e0c2e1aa0e32797c49799f2837ac6af69c57" 843 | dependencies: 844 | is-obj "^1.0.0" 845 | 846 | duplexify@^3.5.0, duplexify@^3.6.0: 847 | version "3.6.0" 848 | resolved "https://registry.yarnpkg.com/duplexify/-/duplexify-3.6.0.tgz#592903f5d80b38d037220541264d69a198fb3410" 849 | dependencies: 850 | end-of-stream "^1.0.0" 851 | inherits "^2.0.1" 852 | readable-stream "^2.0.0" 853 | stream-shift "^1.0.0" 854 | 855 | eastasianwidth@^0.2.0: 856 | version "0.2.0" 857 | resolved "https://registry.yarnpkg.com/eastasianwidth/-/eastasianwidth-0.2.0.tgz#696ce2ec0aa0e6ea93a397ffcf24aa7840c827cb" 858 | 859 | ecc-jsbn@~0.1.1: 860 | version "0.1.1" 861 | resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.1.tgz#0fc73a9ed5f0d53c38193398523ef7e543777505" 862 | dependencies: 863 | jsbn "~0.1.0" 864 | 865 | ecdsa-sig-formatter@1.0.10: 866 | version "1.0.10" 867 | resolved "https://registry.yarnpkg.com/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.10.tgz#1c595000f04a8897dfb85000892a0f4c33af86c3" 868 | dependencies: 869 | safe-buffer "^5.0.1" 870 | 871 | ee-first@1.1.1: 872 | version "1.1.1" 873 | resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d" 874 | 875 | ejs@^2.6.1: 876 | version "2.6.1" 877 | resolved "https://registry.yarnpkg.com/ejs/-/ejs-2.6.1.tgz#498ec0d495655abc6f23cd61868d926464071aa0" 878 | 879 | empower-core@^1.2.0: 880 | version "1.2.0" 881 | resolved "https://registry.yarnpkg.com/empower-core/-/empower-core-1.2.0.tgz#ce3fb2484d5187fa29c23fba8344b0b2fdf5601c" 882 | dependencies: 883 | call-signature "0.0.2" 884 | core-js "^2.0.0" 885 | 886 | empower@^1.3.0: 887 | version "1.3.0" 888 | resolved "https://registry.yarnpkg.com/empower/-/empower-1.3.0.tgz#6b05e77625e77dc44945c4328562c3020b01fa4b" 889 | dependencies: 890 | core-js "^2.0.0" 891 | empower-core "^1.2.0" 892 | 893 | encodeurl@~1.0.2: 894 | version "1.0.2" 895 | resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59" 896 | 897 | end-of-stream@^1.0.0, end-of-stream@^1.1.0: 898 | version "1.4.1" 899 | resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.1.tgz#ed29634d19baba463b6ce6b80a37213eab71ec43" 900 | dependencies: 901 | once "^1.4.0" 902 | 903 | ent@^2.2.0: 904 | version "2.2.0" 905 | resolved "https://registry.yarnpkg.com/ent/-/ent-2.2.0.tgz#e964219325a21d05f44466a2f686ed6ce5f5dd1d" 906 | 907 | escape-html@~1.0.3: 908 | version "1.0.3" 909 | resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988" 910 | 911 | escape-string-regexp@^1.0.2, escape-string-regexp@^1.0.5: 912 | version "1.0.5" 913 | resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" 914 | 915 | esprima@^4.0.0: 916 | version "4.0.1" 917 | resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71" 918 | 919 | espurify@^1.6.0: 920 | version "1.8.1" 921 | resolved "https://registry.yarnpkg.com/espurify/-/espurify-1.8.1.tgz#5746c6c1ab42d302de10bd1d5bf7f0e8c0515056" 922 | dependencies: 923 | core-js "^2.0.0" 924 | 925 | estraverse@^4.1.0, estraverse@^4.2.0: 926 | version "4.2.0" 927 | resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-4.2.0.tgz#0dee3fed31fcd469618ce7342099fc1afa0bdb13" 928 | 929 | esutils@^2.0.2: 930 | version "2.0.2" 931 | resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.2.tgz#0abf4f1caa5bcb1f7a9d8acc6dea4faaa04bac9b" 932 | 933 | etag@~1.8.1: 934 | version "1.8.1" 935 | resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887" 936 | 937 | expand-brackets@^2.1.4: 938 | version "2.1.4" 939 | resolved "https://registry.yarnpkg.com/expand-brackets/-/expand-brackets-2.1.4.tgz#b77735e315ce30f6b6eff0f83b04151a22449622" 940 | dependencies: 941 | debug "^2.3.3" 942 | define-property "^0.2.5" 943 | extend-shallow "^2.0.1" 944 | posix-character-classes "^0.1.0" 945 | regex-not "^1.0.0" 946 | snapdragon "^0.8.1" 947 | to-regex "^3.0.1" 948 | 949 | express@^4.16.2, express@^4.16.3: 950 | version "4.16.3" 951 | resolved "https://registry.yarnpkg.com/express/-/express-4.16.3.tgz#6af8a502350db3246ecc4becf6b5a34d22f7ed53" 952 | dependencies: 953 | accepts "~1.3.5" 954 | array-flatten "1.1.1" 955 | body-parser "1.18.2" 956 | content-disposition "0.5.2" 957 | content-type "~1.0.4" 958 | cookie "0.3.1" 959 | cookie-signature "1.0.6" 960 | debug "2.6.9" 961 | depd "~1.1.2" 962 | encodeurl "~1.0.2" 963 | escape-html "~1.0.3" 964 | etag "~1.8.1" 965 | finalhandler "1.1.1" 966 | fresh "0.5.2" 967 | merge-descriptors "1.0.1" 968 | methods "~1.1.2" 969 | on-finished "~2.3.0" 970 | parseurl "~1.3.2" 971 | path-to-regexp "0.1.7" 972 | proxy-addr "~2.0.3" 973 | qs "6.5.1" 974 | range-parser "~1.2.0" 975 | safe-buffer "5.1.1" 976 | send "0.16.2" 977 | serve-static "1.13.2" 978 | setprototypeof "1.1.0" 979 | statuses "~1.4.0" 980 | type-is "~1.6.16" 981 | utils-merge "1.0.1" 982 | vary "~1.1.2" 983 | 984 | extend-shallow@^2.0.1: 985 | version "2.0.1" 986 | resolved "https://registry.yarnpkg.com/extend-shallow/-/extend-shallow-2.0.1.tgz#51af7d614ad9a9f610ea1bafbb989d6b1c56890f" 987 | dependencies: 988 | is-extendable "^0.1.0" 989 | 990 | extend-shallow@^3.0.0, extend-shallow@^3.0.2: 991 | version "3.0.2" 992 | resolved "https://registry.yarnpkg.com/extend-shallow/-/extend-shallow-3.0.2.tgz#26a71aaf073b39fb2127172746131c2704028db8" 993 | dependencies: 994 | assign-symbols "^1.0.0" 995 | is-extendable "^1.0.1" 996 | 997 | extend@^3.0.0, extend@^3.0.1, extend@~3.0.1: 998 | version "3.0.2" 999 | resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.2.tgz#f8b1136b4071fbd8eb140aff858b1019ec2915fa" 1000 | 1001 | extglob@^2.0.4: 1002 | version "2.0.4" 1003 | resolved "https://registry.yarnpkg.com/extglob/-/extglob-2.0.4.tgz#ad00fe4dc612a9232e8718711dc5cb5ab0285543" 1004 | dependencies: 1005 | array-unique "^0.3.2" 1006 | define-property "^1.0.0" 1007 | expand-brackets "^2.1.4" 1008 | extend-shallow "^2.0.1" 1009 | fragment-cache "^0.2.1" 1010 | regex-not "^1.0.0" 1011 | snapdragon "^0.8.1" 1012 | to-regex "^3.0.1" 1013 | 1014 | extsprintf@1.3.0: 1015 | version "1.3.0" 1016 | resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.3.0.tgz#96918440e3041a7a414f8c52e3c574eb3c3e1e05" 1017 | 1018 | extsprintf@^1.2.0: 1019 | version "1.4.0" 1020 | resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.4.0.tgz#e2689f8f356fad62cca65a3a91c5df5f9551692f" 1021 | 1022 | fast-deep-equal@^1.0.0: 1023 | version "1.1.0" 1024 | resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz#c053477817c86b51daa853c81e059b733d023614" 1025 | 1026 | fast-glob@^2.0.2: 1027 | version "2.2.2" 1028 | resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-2.2.2.tgz#71723338ac9b4e0e2fff1d6748a2a13d5ed352bf" 1029 | dependencies: 1030 | "@mrmlnc/readdir-enhanced" "^2.2.1" 1031 | "@nodelib/fs.stat" "^1.0.1" 1032 | glob-parent "^3.1.0" 1033 | is-glob "^4.0.0" 1034 | merge2 "^1.2.1" 1035 | micromatch "^3.1.10" 1036 | 1037 | fast-json-stable-stringify@^2.0.0: 1038 | version "2.0.0" 1039 | resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz#d5142c0caee6b1189f87d3a76111064f86c8bbf2" 1040 | 1041 | faye-websocket@0.11.1: 1042 | version "0.11.1" 1043 | resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.11.1.tgz#f0efe18c4f56e4f40afc7e06c719fd5ee6188f38" 1044 | dependencies: 1045 | websocket-driver ">=0.5.1" 1046 | 1047 | fill-range@^4.0.0: 1048 | version "4.0.0" 1049 | resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-4.0.0.tgz#d544811d428f98eb06a63dc402d2403c328c38f7" 1050 | dependencies: 1051 | extend-shallow "^2.0.1" 1052 | is-number "^3.0.0" 1053 | repeat-string "^1.6.1" 1054 | to-regex-range "^2.1.0" 1055 | 1056 | finalhandler@1.1.1: 1057 | version "1.1.1" 1058 | resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.1.1.tgz#eebf4ed840079c83f4249038c9d703008301b105" 1059 | dependencies: 1060 | debug "2.6.9" 1061 | encodeurl "~1.0.2" 1062 | escape-html "~1.0.3" 1063 | on-finished "~2.3.0" 1064 | parseurl "~1.3.2" 1065 | statuses "~1.4.0" 1066 | unpipe "~1.0.0" 1067 | 1068 | find-up@^2.1.0: 1069 | version "2.1.0" 1070 | resolved "https://registry.yarnpkg.com/find-up/-/find-up-2.1.0.tgz#45d1b7e506c717ddd482775a2b77920a3c0c57a7" 1071 | dependencies: 1072 | locate-path "^2.0.0" 1073 | 1074 | firebase-admin@~5.13.1: 1075 | version "5.13.1" 1076 | resolved "https://registry.yarnpkg.com/firebase-admin/-/firebase-admin-5.13.1.tgz#79cfa2ce20c90061ae09176e33b7767c1eb02f96" 1077 | dependencies: 1078 | "@firebase/app" "^0.3.1" 1079 | "@firebase/database" "^0.3.1" 1080 | "@google-cloud/firestore" "^0.15.4" 1081 | "@google-cloud/storage" "^1.6.0" 1082 | "@types/google-cloud__storage" "^1.1.7" 1083 | "@types/node" "^8.0.53" 1084 | jsonwebtoken "8.1.0" 1085 | node-forge "0.7.4" 1086 | 1087 | firebase-functions@^2.0.0: 1088 | version "2.0.0" 1089 | resolved "https://registry.yarnpkg.com/firebase-functions/-/firebase-functions-2.0.0.tgz#c9df411bdd746b58703c6911409114c613a6392e" 1090 | dependencies: 1091 | "@types/cors" "^2.8.1" 1092 | "@types/express" "^4.11.1" 1093 | "@types/jsonwebtoken" "^7.2.6" 1094 | "@types/lodash" "^4.14.34" 1095 | cors "^2.8.4" 1096 | express "^4.16.2" 1097 | jsonwebtoken "^8.2.1" 1098 | lodash "^4.6.1" 1099 | 1100 | follow-redirects@^1.3.0: 1101 | version "1.5.1" 1102 | resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.5.1.tgz#67a8f14f5a1f67f962c2c46469c79eaec0a90291" 1103 | dependencies: 1104 | debug "^3.1.0" 1105 | 1106 | for-in@^1.0.2: 1107 | version "1.0.2" 1108 | resolved "https://registry.yarnpkg.com/for-in/-/for-in-1.0.2.tgz#81068d295a8142ec0ac726c6e2200c30fb6d5e80" 1109 | 1110 | foreach@^2.0.5: 1111 | version "2.0.5" 1112 | resolved "https://registry.yarnpkg.com/foreach/-/foreach-2.0.5.tgz#0bee005018aeb260d0a3af3ae658dd0136ec1b99" 1113 | 1114 | forever-agent@~0.6.1: 1115 | version "0.6.1" 1116 | resolved "https://registry.yarnpkg.com/forever-agent/-/forever-agent-0.6.1.tgz#fbc71f0c41adeb37f96c577ad1ed42d8fdacca91" 1117 | 1118 | form-data@~2.3.1: 1119 | version "2.3.2" 1120 | resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.3.2.tgz#4970498be604c20c005d4f5c23aecd21d6b49099" 1121 | dependencies: 1122 | asynckit "^0.4.0" 1123 | combined-stream "1.0.6" 1124 | mime-types "^2.1.12" 1125 | 1126 | forwarded@~0.1.2: 1127 | version "0.1.2" 1128 | resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.1.2.tgz#98c23dab1175657b8c0573e8ceccd91b0ff18c84" 1129 | 1130 | fragment-cache@^0.2.1: 1131 | version "0.2.1" 1132 | resolved "https://registry.yarnpkg.com/fragment-cache/-/fragment-cache-0.2.1.tgz#4290fad27f13e89be7f33799c6bc5a0abfff0d19" 1133 | dependencies: 1134 | map-cache "^0.2.2" 1135 | 1136 | fresh@0.5.2: 1137 | version "0.5.2" 1138 | resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7" 1139 | 1140 | fs-minipass@^1.2.5: 1141 | version "1.2.5" 1142 | resolved "https://registry.yarnpkg.com/fs-minipass/-/fs-minipass-1.2.5.tgz#06c277218454ec288df77ada54a03b8702aacb9d" 1143 | dependencies: 1144 | minipass "^2.2.1" 1145 | 1146 | fs.realpath@^1.0.0: 1147 | version "1.0.0" 1148 | resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" 1149 | 1150 | functional-red-black-tree@^1.0.1: 1151 | version "1.0.1" 1152 | resolved "https://registry.yarnpkg.com/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz#1b0ab3bd553b2a0d6399d29c0e3ea0b252078327" 1153 | 1154 | gauge@~2.7.3: 1155 | version "2.7.4" 1156 | resolved "https://registry.yarnpkg.com/gauge/-/gauge-2.7.4.tgz#2c03405c7538c39d7eb37b317022e325fb018bf7" 1157 | dependencies: 1158 | aproba "^1.0.3" 1159 | console-control-strings "^1.0.0" 1160 | has-unicode "^2.0.0" 1161 | object-assign "^4.1.0" 1162 | signal-exit "^3.0.0" 1163 | string-width "^1.0.1" 1164 | strip-ansi "^3.0.1" 1165 | wide-align "^1.1.0" 1166 | 1167 | gcp-metadata@^0.6.1, gcp-metadata@^0.6.3: 1168 | version "0.6.3" 1169 | resolved "https://registry.yarnpkg.com/gcp-metadata/-/gcp-metadata-0.6.3.tgz#4550c08859c528b370459bd77a7187ea0bdbc4ab" 1170 | dependencies: 1171 | axios "^0.18.0" 1172 | extend "^3.0.1" 1173 | retry-axios "0.3.2" 1174 | 1175 | gcs-resumable-upload@^0.10.2: 1176 | version "0.10.2" 1177 | resolved "https://registry.yarnpkg.com/gcs-resumable-upload/-/gcs-resumable-upload-0.10.2.tgz#7f29b3ee23dcec4170367c0711418249c660545f" 1178 | dependencies: 1179 | configstore "^3.1.2" 1180 | google-auto-auth "^0.10.0" 1181 | pumpify "^1.4.0" 1182 | request "^2.85.0" 1183 | stream-events "^1.0.3" 1184 | 1185 | get-value@^2.0.3, get-value@^2.0.6: 1186 | version "2.0.6" 1187 | resolved "https://registry.yarnpkg.com/get-value/-/get-value-2.0.6.tgz#dc15ca1c672387ca76bd37ac0a395ba2042a2c28" 1188 | 1189 | getpass@^0.1.1: 1190 | version "0.1.7" 1191 | resolved "https://registry.yarnpkg.com/getpass/-/getpass-0.1.7.tgz#5eff8e3e684d569ae4cb2b1282604e8ba62149fa" 1192 | dependencies: 1193 | assert-plus "^1.0.0" 1194 | 1195 | glob-parent@^3.1.0: 1196 | version "3.1.0" 1197 | resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-3.1.0.tgz#9e6af6299d8d3bd2bd40430832bd113df906c5ae" 1198 | dependencies: 1199 | is-glob "^3.1.0" 1200 | path-dirname "^1.0.0" 1201 | 1202 | glob-to-regexp@^0.3.0: 1203 | version "0.3.0" 1204 | resolved "https://registry.yarnpkg.com/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz#8c5a1494d2066c570cc3bfe4496175acc4d502ab" 1205 | 1206 | glob@^7.0.5, glob@^7.1.1, glob@^7.1.2: 1207 | version "7.1.2" 1208 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.2.tgz#c19c9df9a028702d678612384a6552404c636d15" 1209 | dependencies: 1210 | fs.realpath "^1.0.0" 1211 | inflight "^1.0.4" 1212 | inherits "2" 1213 | minimatch "^3.0.4" 1214 | once "^1.3.0" 1215 | path-is-absolute "^1.0.0" 1216 | 1217 | globby@^8.0.0, globby@^8.0.1: 1218 | version "8.0.1" 1219 | resolved "https://registry.yarnpkg.com/globby/-/globby-8.0.1.tgz#b5ad48b8aa80b35b814fc1281ecc851f1d2b5b50" 1220 | dependencies: 1221 | array-union "^1.0.1" 1222 | dir-glob "^2.0.0" 1223 | fast-glob "^2.0.2" 1224 | glob "^7.1.2" 1225 | ignore "^3.3.5" 1226 | pify "^3.0.0" 1227 | slash "^1.0.0" 1228 | 1229 | google-auth-library@^1.3.1, google-auth-library@^1.6.0, google-auth-library@^1.6.1: 1230 | version "1.6.1" 1231 | resolved "https://registry.yarnpkg.com/google-auth-library/-/google-auth-library-1.6.1.tgz#9c73d831ad720c0c3048ab89d0ffdec714d07dd2" 1232 | dependencies: 1233 | axios "^0.18.0" 1234 | gcp-metadata "^0.6.3" 1235 | gtoken "^2.3.0" 1236 | jws "^3.1.5" 1237 | lodash.isstring "^4.0.1" 1238 | lru-cache "^4.1.3" 1239 | retry-axios "^0.3.2" 1240 | 1241 | google-auto-auth@^0.10.0: 1242 | version "0.10.1" 1243 | resolved "https://registry.yarnpkg.com/google-auto-auth/-/google-auto-auth-0.10.1.tgz#68834a6f3da59a6cb27fce56f76e3d99ee49d0a2" 1244 | dependencies: 1245 | async "^2.3.0" 1246 | gcp-metadata "^0.6.1" 1247 | google-auth-library "^1.3.1" 1248 | request "^2.79.0" 1249 | 1250 | google-gax@^0.17.1: 1251 | version "0.17.1" 1252 | resolved "https://registry.yarnpkg.com/google-gax/-/google-gax-0.17.1.tgz#825ead4ab68f1cb3a702ed0a6c64d7ac9b882747" 1253 | dependencies: 1254 | duplexify "^3.6.0" 1255 | extend "^3.0.1" 1256 | globby "^8.0.1" 1257 | google-auth-library "^1.6.1" 1258 | google-proto-files "^0.16.0" 1259 | grpc "^1.12.2" 1260 | is-stream-ended "^0.1.4" 1261 | lodash "^4.17.10" 1262 | protobufjs "^6.8.6" 1263 | retry-request "^4.0.0" 1264 | through2 "^2.0.3" 1265 | 1266 | google-p12-pem@^1.0.0: 1267 | version "1.0.2" 1268 | resolved "https://registry.yarnpkg.com/google-p12-pem/-/google-p12-pem-1.0.2.tgz#c8a3843504012283a0dbffc7430b7c753ecd4b07" 1269 | dependencies: 1270 | node-forge "^0.7.4" 1271 | pify "^3.0.0" 1272 | 1273 | google-proto-files@^0.16.0, google-proto-files@^0.16.1: 1274 | version "0.16.1" 1275 | resolved "https://registry.yarnpkg.com/google-proto-files/-/google-proto-files-0.16.1.tgz#e422e4c0cfd65c481b63f3c0e0cca03ba9cd97ce" 1276 | dependencies: 1277 | globby "^8.0.0" 1278 | power-assert "^1.4.4" 1279 | protobufjs "^6.8.0" 1280 | 1281 | graceful-fs@^4.1.11, graceful-fs@^4.1.2: 1282 | version "4.1.11" 1283 | resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.1.11.tgz#0e8bdfe4d1ddb8854d64e04ea7c00e2a026e5658" 1284 | 1285 | grpc@^1.12.2: 1286 | version "1.13.1" 1287 | resolved "https://registry.yarnpkg.com/grpc/-/grpc-1.13.1.tgz#9b5c49d4e56309b6e3bd631f8948b7b298d88790" 1288 | dependencies: 1289 | lodash "^4.17.5" 1290 | nan "^2.0.0" 1291 | node-pre-gyp "^0.10.0" 1292 | protobufjs "^5.0.3" 1293 | 1294 | gtoken@^2.3.0: 1295 | version "2.3.0" 1296 | resolved "https://registry.yarnpkg.com/gtoken/-/gtoken-2.3.0.tgz#4e0ffc16432d7041a1b3dbc1d97aac17a5dc964a" 1297 | dependencies: 1298 | axios "^0.18.0" 1299 | google-p12-pem "^1.0.0" 1300 | jws "^3.1.4" 1301 | mime "^2.2.0" 1302 | pify "^3.0.0" 1303 | 1304 | har-schema@^2.0.0: 1305 | version "2.0.0" 1306 | resolved "https://registry.yarnpkg.com/har-schema/-/har-schema-2.0.0.tgz#a94c2224ebcac04782a0d9035521f24735b7ec92" 1307 | 1308 | har-validator@~5.0.3: 1309 | version "5.0.3" 1310 | resolved "https://registry.yarnpkg.com/har-validator/-/har-validator-5.0.3.tgz#ba402c266194f15956ef15e0fcf242993f6a7dfd" 1311 | dependencies: 1312 | ajv "^5.1.0" 1313 | har-schema "^2.0.0" 1314 | 1315 | has-ansi@^2.0.0: 1316 | version "2.0.0" 1317 | resolved "https://registry.yarnpkg.com/has-ansi/-/has-ansi-2.0.0.tgz#34f5049ce1ecdf2b0649af3ef24e45ed35416d91" 1318 | dependencies: 1319 | ansi-regex "^2.0.0" 1320 | 1321 | has-flag@^3.0.0: 1322 | version "3.0.0" 1323 | resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd" 1324 | 1325 | has-unicode@^2.0.0: 1326 | version "2.0.1" 1327 | resolved "https://registry.yarnpkg.com/has-unicode/-/has-unicode-2.0.1.tgz#e0e6fe6a28cf51138855e086d1691e771de2a8b9" 1328 | 1329 | has-value@^0.3.1: 1330 | version "0.3.1" 1331 | resolved "https://registry.yarnpkg.com/has-value/-/has-value-0.3.1.tgz#7b1f58bada62ca827ec0a2078025654845995e1f" 1332 | dependencies: 1333 | get-value "^2.0.3" 1334 | has-values "^0.1.4" 1335 | isobject "^2.0.0" 1336 | 1337 | has-value@^1.0.0: 1338 | version "1.0.0" 1339 | resolved "https://registry.yarnpkg.com/has-value/-/has-value-1.0.0.tgz#18b281da585b1c5c51def24c930ed29a0be6b177" 1340 | dependencies: 1341 | get-value "^2.0.6" 1342 | has-values "^1.0.0" 1343 | isobject "^3.0.0" 1344 | 1345 | has-values@^0.1.4: 1346 | version "0.1.4" 1347 | resolved "https://registry.yarnpkg.com/has-values/-/has-values-0.1.4.tgz#6d61de95d91dfca9b9a02089ad384bff8f62b771" 1348 | 1349 | has-values@^1.0.0: 1350 | version "1.0.0" 1351 | resolved "https://registry.yarnpkg.com/has-values/-/has-values-1.0.0.tgz#95b0b63fec2146619a6fe57fe75628d5a39efe4f" 1352 | dependencies: 1353 | is-number "^3.0.0" 1354 | kind-of "^4.0.0" 1355 | 1356 | hash-stream-validation@^0.2.1: 1357 | version "0.2.1" 1358 | resolved "https://registry.yarnpkg.com/hash-stream-validation/-/hash-stream-validation-0.2.1.tgz#ecc9b997b218be5bb31298628bb807869b73dcd1" 1359 | dependencies: 1360 | through2 "^2.0.0" 1361 | 1362 | http-errors@1.6.2: 1363 | version "1.6.2" 1364 | resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.6.2.tgz#0a002cc85707192a7e7946ceedc11155f60ec736" 1365 | dependencies: 1366 | depd "1.1.1" 1367 | inherits "2.0.3" 1368 | setprototypeof "1.0.3" 1369 | statuses ">= 1.3.1 < 2" 1370 | 1371 | http-errors@~1.6.2: 1372 | version "1.6.3" 1373 | resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.6.3.tgz#8b55680bb4be283a0b5bf4ea2e38580be1d9320d" 1374 | dependencies: 1375 | depd "~1.1.2" 1376 | inherits "2.0.3" 1377 | setprototypeof "1.1.0" 1378 | statuses ">= 1.4.0 < 2" 1379 | 1380 | http-parser-js@>=0.4.0: 1381 | version "0.4.13" 1382 | resolved "https://registry.yarnpkg.com/http-parser-js/-/http-parser-js-0.4.13.tgz#3bd6d6fde6e3172c9334c3b33b6c193d80fe1137" 1383 | 1384 | http-signature@~1.2.0: 1385 | version "1.2.0" 1386 | resolved "https://registry.yarnpkg.com/http-signature/-/http-signature-1.2.0.tgz#9aecd925114772f3d95b65a60abb8f7c18fbace1" 1387 | dependencies: 1388 | assert-plus "^1.0.0" 1389 | jsprim "^1.2.2" 1390 | sshpk "^1.7.0" 1391 | 1392 | iconv-lite@0.4.19: 1393 | version "0.4.19" 1394 | resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.19.tgz#f7468f60135f5e5dad3399c0a81be9a1603a082b" 1395 | 1396 | iconv-lite@^0.4.4: 1397 | version "0.4.23" 1398 | resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.23.tgz#297871f63be507adcfbfca715d0cd0eed84e9a63" 1399 | dependencies: 1400 | safer-buffer ">= 2.1.2 < 3" 1401 | 1402 | ignore-walk@^3.0.1: 1403 | version "3.0.1" 1404 | resolved "https://registry.yarnpkg.com/ignore-walk/-/ignore-walk-3.0.1.tgz#a83e62e7d272ac0e3b551aaa82831a19b69f82f8" 1405 | dependencies: 1406 | minimatch "^3.0.4" 1407 | 1408 | ignore@^3.3.5: 1409 | version "3.3.10" 1410 | resolved "https://registry.yarnpkg.com/ignore/-/ignore-3.3.10.tgz#0a97fb876986e8081c631160f8f9f389157f0043" 1411 | 1412 | imurmurhash@^0.1.4: 1413 | version "0.1.4" 1414 | resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" 1415 | 1416 | indexof@0.0.1: 1417 | version "0.0.1" 1418 | resolved "https://registry.yarnpkg.com/indexof/-/indexof-0.0.1.tgz#82dc336d232b9062179d05ab3293a66059fd435d" 1419 | 1420 | inflight@^1.0.4: 1421 | version "1.0.6" 1422 | resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" 1423 | dependencies: 1424 | once "^1.3.0" 1425 | wrappy "1" 1426 | 1427 | inherits@2, inherits@2.0.3, inherits@^2.0.1, inherits@^2.0.3, inherits@~2.0.1, inherits@~2.0.3: 1428 | version "2.0.3" 1429 | resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de" 1430 | 1431 | ini@~1.3.0: 1432 | version "1.3.5" 1433 | resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.5.tgz#eee25f56db1c9ec6085e0c22778083f596abf927" 1434 | 1435 | invert-kv@^1.0.0: 1436 | version "1.0.0" 1437 | resolved "https://registry.yarnpkg.com/invert-kv/-/invert-kv-1.0.0.tgz#104a8e4aaca6d3d8cd157a8ef8bfab2d7a3ffdb6" 1438 | 1439 | ipaddr.js@1.6.0: 1440 | version "1.6.0" 1441 | resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.6.0.tgz#e3fa357b773da619f26e95f049d055c72796f86b" 1442 | 1443 | is-accessor-descriptor@^0.1.6: 1444 | version "0.1.6" 1445 | resolved "https://registry.yarnpkg.com/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz#a9e12cb3ae8d876727eeef3843f8a0897b5c98d6" 1446 | dependencies: 1447 | kind-of "^3.0.2" 1448 | 1449 | is-accessor-descriptor@^1.0.0: 1450 | version "1.0.0" 1451 | resolved "https://registry.yarnpkg.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz#169c2f6d3df1f992618072365c9b0ea1f6878656" 1452 | dependencies: 1453 | kind-of "^6.0.0" 1454 | 1455 | is-buffer@^1.1.5: 1456 | version "1.1.6" 1457 | resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.6.tgz#efaa2ea9daa0d7ab2ea13a97b2b8ad51fefbe8be" 1458 | 1459 | is-data-descriptor@^0.1.4: 1460 | version "0.1.4" 1461 | resolved "https://registry.yarnpkg.com/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz#0b5ee648388e2c860282e793f1856fec3f301b56" 1462 | dependencies: 1463 | kind-of "^3.0.2" 1464 | 1465 | is-data-descriptor@^1.0.0: 1466 | version "1.0.0" 1467 | resolved "https://registry.yarnpkg.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz#d84876321d0e7add03990406abbbbd36ba9268c7" 1468 | dependencies: 1469 | kind-of "^6.0.0" 1470 | 1471 | is-descriptor@^0.1.0: 1472 | version "0.1.6" 1473 | resolved "https://registry.yarnpkg.com/is-descriptor/-/is-descriptor-0.1.6.tgz#366d8240dde487ca51823b1ab9f07a10a78251ca" 1474 | dependencies: 1475 | is-accessor-descriptor "^0.1.6" 1476 | is-data-descriptor "^0.1.4" 1477 | kind-of "^5.0.0" 1478 | 1479 | is-descriptor@^1.0.0, is-descriptor@^1.0.2: 1480 | version "1.0.2" 1481 | resolved "https://registry.yarnpkg.com/is-descriptor/-/is-descriptor-1.0.2.tgz#3b159746a66604b04f8c81524ba365c5f14d86ec" 1482 | dependencies: 1483 | is-accessor-descriptor "^1.0.0" 1484 | is-data-descriptor "^1.0.0" 1485 | kind-of "^6.0.2" 1486 | 1487 | is-extendable@^0.1.0, is-extendable@^0.1.1: 1488 | version "0.1.1" 1489 | resolved "https://registry.yarnpkg.com/is-extendable/-/is-extendable-0.1.1.tgz#62b110e289a471418e3ec36a617d472e301dfc89" 1490 | 1491 | is-extendable@^1.0.1: 1492 | version "1.0.1" 1493 | resolved "https://registry.yarnpkg.com/is-extendable/-/is-extendable-1.0.1.tgz#a7470f9e426733d81bd81e1155264e3a3507cab4" 1494 | dependencies: 1495 | is-plain-object "^2.0.4" 1496 | 1497 | is-extglob@^2.1.0, is-extglob@^2.1.1: 1498 | version "2.1.1" 1499 | resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" 1500 | 1501 | is-fullwidth-code-point@^1.0.0: 1502 | version "1.0.0" 1503 | resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz#ef9e31386f031a7f0d643af82fde50c457ef00cb" 1504 | dependencies: 1505 | number-is-nan "^1.0.0" 1506 | 1507 | is-fullwidth-code-point@^2.0.0: 1508 | version "2.0.0" 1509 | resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz#a3b30a5c4f199183167aaab93beefae3ddfb654f" 1510 | 1511 | is-glob@^3.1.0: 1512 | version "3.1.0" 1513 | resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-3.1.0.tgz#7ba5ae24217804ac70707b96922567486cc3e84a" 1514 | dependencies: 1515 | is-extglob "^2.1.0" 1516 | 1517 | is-glob@^4.0.0: 1518 | version "4.0.0" 1519 | resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.0.tgz#9521c76845cc2610a85203ddf080a958c2ffabc0" 1520 | dependencies: 1521 | is-extglob "^2.1.1" 1522 | 1523 | is-number@^3.0.0: 1524 | version "3.0.0" 1525 | resolved "https://registry.yarnpkg.com/is-number/-/is-number-3.0.0.tgz#24fd6201a4782cf50561c810276afc7d12d71195" 1526 | dependencies: 1527 | kind-of "^3.0.2" 1528 | 1529 | is-obj@^1.0.0: 1530 | version "1.0.1" 1531 | resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-1.0.1.tgz#3e4729ac1f5fde025cd7d83a896dab9f4f67db0f" 1532 | 1533 | is-plain-object@^2.0.1, is-plain-object@^2.0.3, is-plain-object@^2.0.4: 1534 | version "2.0.4" 1535 | resolved "https://registry.yarnpkg.com/is-plain-object/-/is-plain-object-2.0.4.tgz#2c163b3fafb1b606d9d17928f05c2a1c38e07677" 1536 | dependencies: 1537 | isobject "^3.0.1" 1538 | 1539 | is-stream-ended@^0.1.0, is-stream-ended@^0.1.4: 1540 | version "0.1.4" 1541 | resolved "https://registry.yarnpkg.com/is-stream-ended/-/is-stream-ended-0.1.4.tgz#f50224e95e06bce0e356d440a4827cd35b267eda" 1542 | 1543 | is-typedarray@~1.0.0: 1544 | version "1.0.0" 1545 | resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a" 1546 | 1547 | is-windows@^1.0.2: 1548 | version "1.0.2" 1549 | resolved "https://registry.yarnpkg.com/is-windows/-/is-windows-1.0.2.tgz#d1850eb9791ecd18e6182ce12a30f396634bb19d" 1550 | 1551 | is@^3.0.1, is@^3.2.0, is@^3.2.1: 1552 | version "3.2.1" 1553 | resolved "https://registry.yarnpkg.com/is/-/is-3.2.1.tgz#d0ac2ad55eb7b0bec926a5266f6c662aaa83dca5" 1554 | 1555 | isarray@0.0.1: 1556 | version "0.0.1" 1557 | resolved "https://registry.yarnpkg.com/isarray/-/isarray-0.0.1.tgz#8a18acfca9a8f4177e09abfc6038939b05d1eedf" 1558 | 1559 | isarray@1.0.0, isarray@~1.0.0: 1560 | version "1.0.0" 1561 | resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" 1562 | 1563 | isobject@^2.0.0: 1564 | version "2.1.0" 1565 | resolved "https://registry.yarnpkg.com/isobject/-/isobject-2.1.0.tgz#f065561096a3f1da2ef46272f815c840d87e0c89" 1566 | dependencies: 1567 | isarray "1.0.0" 1568 | 1569 | isobject@^3.0.0, isobject@^3.0.1: 1570 | version "3.0.1" 1571 | resolved "https://registry.yarnpkg.com/isobject/-/isobject-3.0.1.tgz#4e431e92b11a9731636aa1f9c8d1ccbcfdab78df" 1572 | 1573 | isstream@~0.1.2: 1574 | version "0.1.2" 1575 | resolved "https://registry.yarnpkg.com/isstream/-/isstream-0.1.2.tgz#47e63f7af55afa6f92e1500e690eb8b8529c099a" 1576 | 1577 | js-tokens@^3.0.2: 1578 | version "3.0.2" 1579 | resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-3.0.2.tgz#9866df395102130e38f7f996bceb65443209c25b" 1580 | 1581 | js-yaml@^3.7.0: 1582 | version "3.12.0" 1583 | resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.12.0.tgz#eaed656ec8344f10f527c6bfa1b6e2244de167d1" 1584 | dependencies: 1585 | argparse "^1.0.7" 1586 | esprima "^4.0.0" 1587 | 1588 | jsbn@~0.1.0: 1589 | version "0.1.1" 1590 | resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-0.1.1.tgz#a5e654c2e5a2deb5f201d96cefbca80c0ef2f513" 1591 | 1592 | json-schema-traverse@^0.3.0: 1593 | version "0.3.1" 1594 | resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz#349a6d44c53a51de89b40805c5d5e59b417d3340" 1595 | 1596 | json-schema@0.2.3: 1597 | version "0.2.3" 1598 | resolved "https://registry.yarnpkg.com/json-schema/-/json-schema-0.2.3.tgz#b480c892e59a2f05954ce727bd3f2a4e882f9e13" 1599 | 1600 | json-stringify-safe@~5.0.1: 1601 | version "5.0.1" 1602 | resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" 1603 | 1604 | jsonwebtoken@8.1.0: 1605 | version "8.1.0" 1606 | resolved "https://registry.yarnpkg.com/jsonwebtoken/-/jsonwebtoken-8.1.0.tgz#c6397cd2e5fd583d65c007a83dc7bb78e6982b83" 1607 | dependencies: 1608 | jws "^3.1.4" 1609 | lodash.includes "^4.3.0" 1610 | lodash.isboolean "^3.0.3" 1611 | lodash.isinteger "^4.0.4" 1612 | lodash.isnumber "^3.0.3" 1613 | lodash.isplainobject "^4.0.6" 1614 | lodash.isstring "^4.0.1" 1615 | lodash.once "^4.0.0" 1616 | ms "^2.0.0" 1617 | xtend "^4.0.1" 1618 | 1619 | jsonwebtoken@^8.2.1: 1620 | version "8.3.0" 1621 | resolved "https://registry.yarnpkg.com/jsonwebtoken/-/jsonwebtoken-8.3.0.tgz#056c90eee9a65ed6e6c72ddb0a1d325109aaf643" 1622 | dependencies: 1623 | jws "^3.1.5" 1624 | lodash.includes "^4.3.0" 1625 | lodash.isboolean "^3.0.3" 1626 | lodash.isinteger "^4.0.4" 1627 | lodash.isnumber "^3.0.3" 1628 | lodash.isplainobject "^4.0.6" 1629 | lodash.isstring "^4.0.1" 1630 | lodash.once "^4.0.0" 1631 | ms "^2.1.1" 1632 | 1633 | jsprim@^1.2.2: 1634 | version "1.4.1" 1635 | resolved "https://registry.yarnpkg.com/jsprim/-/jsprim-1.4.1.tgz#313e66bc1e5cc06e438bc1b7499c2e5c56acb6a2" 1636 | dependencies: 1637 | assert-plus "1.0.0" 1638 | extsprintf "1.3.0" 1639 | json-schema "0.2.3" 1640 | verror "1.10.0" 1641 | 1642 | jwa@^1.1.5: 1643 | version "1.1.6" 1644 | resolved "https://registry.yarnpkg.com/jwa/-/jwa-1.1.6.tgz#87240e76c9808dbde18783cf2264ef4929ee50e6" 1645 | dependencies: 1646 | buffer-equal-constant-time "1.0.1" 1647 | ecdsa-sig-formatter "1.0.10" 1648 | safe-buffer "^5.0.1" 1649 | 1650 | jws@^3.1.4, jws@^3.1.5: 1651 | version "3.1.5" 1652 | resolved "https://registry.yarnpkg.com/jws/-/jws-3.1.5.tgz#80d12d05b293d1e841e7cb8b4e69e561adcf834f" 1653 | dependencies: 1654 | jwa "^1.1.5" 1655 | safe-buffer "^5.0.1" 1656 | 1657 | kind-of@^3.0.2, kind-of@^3.0.3, kind-of@^3.2.0: 1658 | version "3.2.2" 1659 | resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-3.2.2.tgz#31ea21a734bab9bbb0f32466d893aea51e4a3c64" 1660 | dependencies: 1661 | is-buffer "^1.1.5" 1662 | 1663 | kind-of@^4.0.0: 1664 | version "4.0.0" 1665 | resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-4.0.0.tgz#20813df3d712928b207378691a45066fae72dd57" 1666 | dependencies: 1667 | is-buffer "^1.1.5" 1668 | 1669 | kind-of@^5.0.0: 1670 | version "5.1.0" 1671 | resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-5.1.0.tgz#729c91e2d857b7a419a1f9aa65685c4c33f5845d" 1672 | 1673 | kind-of@^6.0.0, kind-of@^6.0.2: 1674 | version "6.0.2" 1675 | resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-6.0.2.tgz#01146b36a6218e64e58f3a8d66de5d7fc6f6d051" 1676 | 1677 | lcid@^1.0.0: 1678 | version "1.0.0" 1679 | resolved "https://registry.yarnpkg.com/lcid/-/lcid-1.0.0.tgz#308accafa0bc483a3867b4b6f2b9506251d1b835" 1680 | dependencies: 1681 | invert-kv "^1.0.0" 1682 | 1683 | locate-path@^2.0.0: 1684 | version "2.0.0" 1685 | resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-2.0.0.tgz#2b568b265eec944c6d9c0de9c3dbbbca0354cd8e" 1686 | dependencies: 1687 | p-locate "^2.0.0" 1688 | path-exists "^3.0.0" 1689 | 1690 | lodash.includes@^4.3.0: 1691 | version "4.3.0" 1692 | resolved "https://registry.yarnpkg.com/lodash.includes/-/lodash.includes-4.3.0.tgz#60bb98a87cb923c68ca1e51325483314849f553f" 1693 | 1694 | lodash.isboolean@^3.0.3: 1695 | version "3.0.3" 1696 | resolved "https://registry.yarnpkg.com/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz#6c2e171db2a257cd96802fd43b01b20d5f5870f6" 1697 | 1698 | lodash.isinteger@^4.0.4: 1699 | version "4.0.4" 1700 | resolved "https://registry.yarnpkg.com/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz#619c0af3d03f8b04c31f5882840b77b11cd68343" 1701 | 1702 | lodash.isnumber@^3.0.3: 1703 | version "3.0.3" 1704 | resolved "https://registry.yarnpkg.com/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz#3ce76810c5928d03352301ac287317f11c0b1ffc" 1705 | 1706 | lodash.isplainobject@^4.0.6: 1707 | version "4.0.6" 1708 | resolved "https://registry.yarnpkg.com/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz#7c526a52d89b45c45cc690b88163be0497f550cb" 1709 | 1710 | lodash.isstring@^4.0.1: 1711 | version "4.0.1" 1712 | resolved "https://registry.yarnpkg.com/lodash.isstring/-/lodash.isstring-4.0.1.tgz#d527dfb5456eca7cc9bb95d5daeaf88ba54a5451" 1713 | 1714 | lodash.merge@^4.6.1: 1715 | version "4.6.1" 1716 | resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.1.tgz#adc25d9cb99b9391c59624f379fbba60d7111d54" 1717 | 1718 | lodash.once@^4.0.0: 1719 | version "4.1.1" 1720 | resolved "https://registry.yarnpkg.com/lodash.once/-/lodash.once-4.1.1.tgz#0dd3971213c7c56df880977d504c88fb471a97ac" 1721 | 1722 | lodash@^4.17.10, lodash@^4.17.5, lodash@^4.6.1: 1723 | version "4.17.10" 1724 | resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.10.tgz#1b7793cf7259ea38fb3661d4d38b3260af8ae4e7" 1725 | 1726 | log-driver@1.2.7: 1727 | version "1.2.7" 1728 | resolved "https://registry.yarnpkg.com/log-driver/-/log-driver-1.2.7.tgz#63b95021f0702fedfa2c9bb0a24e7797d71871d8" 1729 | 1730 | long@^4.0.0: 1731 | version "4.0.0" 1732 | resolved "https://registry.yarnpkg.com/long/-/long-4.0.0.tgz#9a7b71cfb7d361a194ea555241c92f7468d5bf28" 1733 | 1734 | long@~3: 1735 | version "3.2.0" 1736 | resolved "https://registry.yarnpkg.com/long/-/long-3.2.0.tgz#d821b7138ca1cb581c172990ef14db200b5c474b" 1737 | 1738 | lru-cache@^4.1.3: 1739 | version "4.1.3" 1740 | resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-4.1.3.tgz#a1175cf3496dfc8436c156c334b4955992bce69c" 1741 | dependencies: 1742 | pseudomap "^1.0.2" 1743 | yallist "^2.1.2" 1744 | 1745 | make-dir@^1.0.0: 1746 | version "1.3.0" 1747 | resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-1.3.0.tgz#79c1033b80515bd6d24ec9933e860ca75ee27f0c" 1748 | dependencies: 1749 | pify "^3.0.0" 1750 | 1751 | map-cache@^0.2.2: 1752 | version "0.2.2" 1753 | resolved "https://registry.yarnpkg.com/map-cache/-/map-cache-0.2.2.tgz#c32abd0bd6525d9b051645bb4f26ac5dc98a0dbf" 1754 | 1755 | map-visit@^1.0.0: 1756 | version "1.0.0" 1757 | resolved "https://registry.yarnpkg.com/map-visit/-/map-visit-1.0.0.tgz#ecdca8f13144e660f1b5bd41f12f3479d98dfb8f" 1758 | dependencies: 1759 | object-visit "^1.0.0" 1760 | 1761 | media-typer@0.3.0: 1762 | version "0.3.0" 1763 | resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748" 1764 | 1765 | merge-descriptors@1.0.1: 1766 | version "1.0.1" 1767 | resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz#b00aaa556dd8b44568150ec9d1b953f3f90cbb61" 1768 | 1769 | merge2@^1.2.1: 1770 | version "1.2.2" 1771 | resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.2.2.tgz#03212e3da8d86c4d8523cebd6318193414f94e34" 1772 | 1773 | methmeth@^1.1.0: 1774 | version "1.1.0" 1775 | resolved "https://registry.yarnpkg.com/methmeth/-/methmeth-1.1.0.tgz#e80a26618e52f5c4222861bb748510bd10e29089" 1776 | 1777 | methods@~1.1.2: 1778 | version "1.1.2" 1779 | resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee" 1780 | 1781 | micromatch@^3.1.10: 1782 | version "3.1.10" 1783 | resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-3.1.10.tgz#70859bc95c9840952f359a068a3fc49f9ecfac23" 1784 | dependencies: 1785 | arr-diff "^4.0.0" 1786 | array-unique "^0.3.2" 1787 | braces "^2.3.1" 1788 | define-property "^2.0.2" 1789 | extend-shallow "^3.0.2" 1790 | extglob "^2.0.4" 1791 | fragment-cache "^0.2.1" 1792 | kind-of "^6.0.2" 1793 | nanomatch "^1.2.9" 1794 | object.pick "^1.3.0" 1795 | regex-not "^1.0.0" 1796 | snapdragon "^0.8.1" 1797 | to-regex "^3.0.2" 1798 | 1799 | "mime-db@>= 1.34.0 < 2", mime-db@~1.35.0: 1800 | version "1.35.0" 1801 | resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.35.0.tgz#0569d657466491283709663ad379a99b90d9ab47" 1802 | 1803 | mime-types@^2.0.8, mime-types@^2.1.12, mime-types@~2.1.17, mime-types@~2.1.18: 1804 | version "2.1.19" 1805 | resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.19.tgz#71e464537a7ef81c15f2db9d97e913fc0ff606f0" 1806 | dependencies: 1807 | mime-db "~1.35.0" 1808 | 1809 | mime@1.4.1: 1810 | version "1.4.1" 1811 | resolved "https://registry.yarnpkg.com/mime/-/mime-1.4.1.tgz#121f9ebc49e3766f311a76e1fa1c8003c4b03aa6" 1812 | 1813 | mime@^2.2.0: 1814 | version "2.3.1" 1815 | resolved "https://registry.yarnpkg.com/mime/-/mime-2.3.1.tgz#b1621c54d63b97c47d3cfe7f7215f7d64517c369" 1816 | 1817 | minimatch@^3.0.4: 1818 | version "3.0.4" 1819 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083" 1820 | dependencies: 1821 | brace-expansion "^1.1.7" 1822 | 1823 | minimist@0.0.8: 1824 | version "0.0.8" 1825 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-0.0.8.tgz#857fcabfc3397d2625b8228262e86aa7a011b05d" 1826 | 1827 | minimist@^1.2.0: 1828 | version "1.2.0" 1829 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.0.tgz#a35008b20f41383eec1fb914f4cd5df79a264284" 1830 | 1831 | minipass@^2.2.1, minipass@^2.3.3: 1832 | version "2.3.3" 1833 | resolved "https://registry.yarnpkg.com/minipass/-/minipass-2.3.3.tgz#a7dcc8b7b833f5d368759cce544dccb55f50f233" 1834 | dependencies: 1835 | safe-buffer "^5.1.2" 1836 | yallist "^3.0.0" 1837 | 1838 | minizlib@^1.1.0: 1839 | version "1.1.0" 1840 | resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-1.1.0.tgz#11e13658ce46bc3a70a267aac58359d1e0c29ceb" 1841 | dependencies: 1842 | minipass "^2.2.1" 1843 | 1844 | mixin-deep@^1.2.0: 1845 | version "1.3.1" 1846 | resolved "https://registry.yarnpkg.com/mixin-deep/-/mixin-deep-1.3.1.tgz#a49e7268dce1a0d9698e45326c5626df3543d0fe" 1847 | dependencies: 1848 | for-in "^1.0.2" 1849 | is-extendable "^1.0.1" 1850 | 1851 | mkdirp@^0.5.0, mkdirp@^0.5.1: 1852 | version "0.5.1" 1853 | resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.1.tgz#30057438eac6cf7f8c4767f38648d6697d75c903" 1854 | dependencies: 1855 | minimist "0.0.8" 1856 | 1857 | modelo@^4.2.0: 1858 | version "4.2.3" 1859 | resolved "https://registry.yarnpkg.com/modelo/-/modelo-4.2.3.tgz#b278588a4db87fc1e5107ae3a277c0876f38d894" 1860 | 1861 | ms@2.0.0: 1862 | version "2.0.0" 1863 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8" 1864 | 1865 | ms@^2.0.0, ms@^2.1.1: 1866 | version "2.1.1" 1867 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.1.tgz#30a5864eb3ebb0a66f2ebe6d727af06a09d86e0a" 1868 | 1869 | nan@^2.0.0: 1870 | version "2.10.0" 1871 | resolved "https://registry.yarnpkg.com/nan/-/nan-2.10.0.tgz#96d0cd610ebd58d4b4de9cc0c6828cda99c7548f" 1872 | 1873 | nanomatch@^1.2.9: 1874 | version "1.2.13" 1875 | resolved "https://registry.yarnpkg.com/nanomatch/-/nanomatch-1.2.13.tgz#b87a8aa4fc0de8fe6be88895b38983ff265bd119" 1876 | dependencies: 1877 | arr-diff "^4.0.0" 1878 | array-unique "^0.3.2" 1879 | define-property "^2.0.2" 1880 | extend-shallow "^3.0.2" 1881 | fragment-cache "^0.2.1" 1882 | is-windows "^1.0.2" 1883 | kind-of "^6.0.2" 1884 | object.pick "^1.3.0" 1885 | regex-not "^1.0.0" 1886 | snapdragon "^0.8.1" 1887 | to-regex "^3.0.1" 1888 | 1889 | needle@^2.2.1: 1890 | version "2.2.1" 1891 | resolved "https://registry.yarnpkg.com/needle/-/needle-2.2.1.tgz#b5e325bd3aae8c2678902fa296f729455d1d3a7d" 1892 | dependencies: 1893 | debug "^2.1.2" 1894 | iconv-lite "^0.4.4" 1895 | sax "^1.2.4" 1896 | 1897 | negotiator@0.6.1: 1898 | version "0.6.1" 1899 | resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.1.tgz#2b327184e8992101177b28563fb5e7102acd0ca9" 1900 | 1901 | node-forge@0.7.4: 1902 | version "0.7.4" 1903 | resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-0.7.4.tgz#8e6e9f563a1e32213aa7508cded22aa791dbf986" 1904 | 1905 | node-forge@^0.7.4: 1906 | version "0.7.5" 1907 | resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-0.7.5.tgz#6c152c345ce11c52f465c2abd957e8639cd674df" 1908 | 1909 | node-pre-gyp@^0.10.0: 1910 | version "0.10.3" 1911 | resolved "https://registry.yarnpkg.com/node-pre-gyp/-/node-pre-gyp-0.10.3.tgz#3070040716afdc778747b61b6887bf78880b80fc" 1912 | dependencies: 1913 | detect-libc "^1.0.2" 1914 | mkdirp "^0.5.1" 1915 | needle "^2.2.1" 1916 | nopt "^4.0.1" 1917 | npm-packlist "^1.1.6" 1918 | npmlog "^4.0.2" 1919 | rc "^1.2.7" 1920 | rimraf "^2.6.1" 1921 | semver "^5.3.0" 1922 | tar "^4" 1923 | 1924 | nopt@^4.0.1: 1925 | version "4.0.1" 1926 | resolved "https://registry.yarnpkg.com/nopt/-/nopt-4.0.1.tgz#d0d4685afd5415193c8c7505602d0d17cd64474d" 1927 | dependencies: 1928 | abbrev "1" 1929 | osenv "^0.1.4" 1930 | 1931 | npm-bundled@^1.0.1: 1932 | version "1.0.3" 1933 | resolved "https://registry.yarnpkg.com/npm-bundled/-/npm-bundled-1.0.3.tgz#7e71703d973af3370a9591bafe3a63aca0be2308" 1934 | 1935 | npm-packlist@^1.1.6: 1936 | version "1.1.11" 1937 | resolved "https://registry.yarnpkg.com/npm-packlist/-/npm-packlist-1.1.11.tgz#84e8c683cbe7867d34b1d357d893ce29e28a02de" 1938 | dependencies: 1939 | ignore-walk "^3.0.1" 1940 | npm-bundled "^1.0.1" 1941 | 1942 | npmlog@^4.0.2: 1943 | version "4.1.2" 1944 | resolved "https://registry.yarnpkg.com/npmlog/-/npmlog-4.1.2.tgz#08a7f2a8bf734604779a9efa4ad5cc717abb954b" 1945 | dependencies: 1946 | are-we-there-yet "~1.1.2" 1947 | console-control-strings "~1.1.0" 1948 | gauge "~2.7.3" 1949 | set-blocking "~2.0.0" 1950 | 1951 | number-is-nan@^1.0.0: 1952 | version "1.0.1" 1953 | resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d" 1954 | 1955 | oauth-sign@~0.8.2: 1956 | version "0.8.2" 1957 | resolved "https://registry.yarnpkg.com/oauth-sign/-/oauth-sign-0.8.2.tgz#46a6ab7f0aead8deae9ec0565780b7d4efeb9d43" 1958 | 1959 | oauth2-nodejs@^0.2.0: 1960 | version "0.2.0" 1961 | resolved "https://registry.yarnpkg.com/oauth2-nodejs/-/oauth2-nodejs-0.2.0.tgz#3993ec5ee2a1741caa3563e17aed514b4838848c" 1962 | integrity sha512-3oE+VJPCuLXv5F6JmvNphQgyh4xfrCjt0qeGkvTDToRIjbxmoLz2zuC9tuNXiAHTxfUlLh4OPeYPKLVWzonkvA== 1963 | 1964 | object-assign@^4, object-assign@^4.1.0: 1965 | version "4.1.1" 1966 | resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" 1967 | 1968 | object-copy@^0.1.0: 1969 | version "0.1.0" 1970 | resolved "https://registry.yarnpkg.com/object-copy/-/object-copy-0.1.0.tgz#7e7d858b781bd7c991a41ba975ed3812754e998c" 1971 | dependencies: 1972 | copy-descriptor "^0.1.0" 1973 | define-property "^0.2.5" 1974 | kind-of "^3.0.3" 1975 | 1976 | object-keys@^1.0.0, object-keys@^1.0.8: 1977 | version "1.0.12" 1978 | resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.0.12.tgz#09c53855377575310cca62f55bb334abff7b3ed2" 1979 | 1980 | object-visit@^1.0.0: 1981 | version "1.0.1" 1982 | resolved "https://registry.yarnpkg.com/object-visit/-/object-visit-1.0.1.tgz#f79c4493af0c5377b59fe39d395e41042dd045bb" 1983 | dependencies: 1984 | isobject "^3.0.0" 1985 | 1986 | object.pick@^1.3.0: 1987 | version "1.3.0" 1988 | resolved "https://registry.yarnpkg.com/object.pick/-/object.pick-1.3.0.tgz#87a10ac4c1694bd2e1cbf53591a66141fb5dd747" 1989 | dependencies: 1990 | isobject "^3.0.1" 1991 | 1992 | on-finished@~2.3.0: 1993 | version "2.3.0" 1994 | resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.3.0.tgz#20f1336481b083cd75337992a16971aa2d906947" 1995 | dependencies: 1996 | ee-first "1.1.1" 1997 | 1998 | once@^1.3.0, once@^1.3.1, once@^1.4.0: 1999 | version "1.4.0" 2000 | resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" 2001 | dependencies: 2002 | wrappy "1" 2003 | 2004 | optjs@~3.2.2: 2005 | version "3.2.2" 2006 | resolved "https://registry.yarnpkg.com/optjs/-/optjs-3.2.2.tgz#69a6ce89c442a44403141ad2f9b370bd5bb6f4ee" 2007 | 2008 | os-homedir@^1.0.0: 2009 | version "1.0.2" 2010 | resolved "https://registry.yarnpkg.com/os-homedir/-/os-homedir-1.0.2.tgz#ffbc4988336e0e833de0c168c7ef152121aa7fb3" 2011 | 2012 | os-locale@^1.4.0: 2013 | version "1.4.0" 2014 | resolved "https://registry.yarnpkg.com/os-locale/-/os-locale-1.4.0.tgz#20f9f17ae29ed345e8bde583b13d2009803c14d9" 2015 | dependencies: 2016 | lcid "^1.0.0" 2017 | 2018 | os-tmpdir@^1.0.0: 2019 | version "1.0.2" 2020 | resolved "https://registry.yarnpkg.com/os-tmpdir/-/os-tmpdir-1.0.2.tgz#bbe67406c79aa85c5cfec766fe5734555dfa1274" 2021 | 2022 | osenv@^0.1.4: 2023 | version "0.1.5" 2024 | resolved "https://registry.yarnpkg.com/osenv/-/osenv-0.1.5.tgz#85cdfafaeb28e8677f416e287592b5f3f49ea410" 2025 | dependencies: 2026 | os-homedir "^1.0.0" 2027 | os-tmpdir "^1.0.0" 2028 | 2029 | p-limit@^1.1.0: 2030 | version "1.3.0" 2031 | resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-1.3.0.tgz#b86bd5f0c25690911c7590fcbfc2010d54b3ccb8" 2032 | dependencies: 2033 | p-try "^1.0.0" 2034 | 2035 | p-locate@^2.0.0: 2036 | version "2.0.0" 2037 | resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-2.0.0.tgz#20a0103b222a70c8fd39cc2e580680f3dde5ec43" 2038 | dependencies: 2039 | p-limit "^1.1.0" 2040 | 2041 | p-try@^1.0.0: 2042 | version "1.0.0" 2043 | resolved "https://registry.yarnpkg.com/p-try/-/p-try-1.0.0.tgz#cbc79cdbaf8fd4228e13f621f2b1a237c1b207b3" 2044 | 2045 | parseurl@~1.3.2: 2046 | version "1.3.2" 2047 | resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.2.tgz#fc289d4ed8993119460c156253262cdc8de65bf3" 2048 | 2049 | pascalcase@^0.1.1: 2050 | version "0.1.1" 2051 | resolved "https://registry.yarnpkg.com/pascalcase/-/pascalcase-0.1.1.tgz#b363e55e8006ca6fe21784d2db22bd15d7917f14" 2052 | 2053 | path-dirname@^1.0.0: 2054 | version "1.0.2" 2055 | resolved "https://registry.yarnpkg.com/path-dirname/-/path-dirname-1.0.2.tgz#cc33d24d525e099a5388c0336c6e32b9160609e0" 2056 | 2057 | path-exists@^3.0.0: 2058 | version "3.0.0" 2059 | resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-3.0.0.tgz#ce0ebeaa5f78cb18925ea7d810d7b59b010fd515" 2060 | 2061 | path-is-absolute@^1.0.0: 2062 | version "1.0.1" 2063 | resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" 2064 | 2065 | path-parse@^1.0.5: 2066 | version "1.0.5" 2067 | resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.5.tgz#3c1adf871ea9cd6c9431b6ea2bd74a0ff055c4c1" 2068 | 2069 | path-to-regexp@0.1.7: 2070 | version "0.1.7" 2071 | resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.7.tgz#df604178005f522f15eb4490e7247a1bfaa67f8c" 2072 | 2073 | path-type@^3.0.0: 2074 | version "3.0.0" 2075 | resolved "https://registry.yarnpkg.com/path-type/-/path-type-3.0.0.tgz#cef31dc8e0a1a3bb0d105c0cd97cf3bf47f4e36f" 2076 | dependencies: 2077 | pify "^3.0.0" 2078 | 2079 | performance-now@^2.1.0: 2080 | version "2.1.0" 2081 | resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-2.1.0.tgz#6309f4e0e5fa913ec1c69307ae364b4b377c9e7b" 2082 | 2083 | pify@^3.0.0: 2084 | version "3.0.0" 2085 | resolved "https://registry.yarnpkg.com/pify/-/pify-3.0.0.tgz#e5a4acd2c101fdf3d9a4d07f0dbc4db49dd28176" 2086 | 2087 | pkg-up@^2.0.0: 2088 | version "2.0.0" 2089 | resolved "https://registry.yarnpkg.com/pkg-up/-/pkg-up-2.0.0.tgz#c819ac728059a461cab1c3889a2be3c49a004d7f" 2090 | dependencies: 2091 | find-up "^2.1.0" 2092 | 2093 | posix-character-classes@^0.1.0: 2094 | version "0.1.1" 2095 | resolved "https://registry.yarnpkg.com/posix-character-classes/-/posix-character-classes-0.1.1.tgz#01eac0fe3b5af71a2a6c02feabb8c1fef7e00eab" 2096 | 2097 | power-assert-context-formatter@^1.0.7: 2098 | version "1.2.0" 2099 | resolved "https://registry.yarnpkg.com/power-assert-context-formatter/-/power-assert-context-formatter-1.2.0.tgz#8fbe72692288ec5a7203cdf215c8b838a6061d2a" 2100 | dependencies: 2101 | core-js "^2.0.0" 2102 | power-assert-context-traversal "^1.2.0" 2103 | 2104 | power-assert-context-reducer-ast@^1.0.7: 2105 | version "1.2.0" 2106 | resolved "https://registry.yarnpkg.com/power-assert-context-reducer-ast/-/power-assert-context-reducer-ast-1.2.0.tgz#c7ca1c9e39a6fb717f7ac5fe9e76e192bf525df3" 2107 | dependencies: 2108 | acorn "^5.0.0" 2109 | acorn-es7-plugin "^1.0.12" 2110 | core-js "^2.0.0" 2111 | espurify "^1.6.0" 2112 | estraverse "^4.2.0" 2113 | 2114 | power-assert-context-traversal@^1.2.0: 2115 | version "1.2.0" 2116 | resolved "https://registry.yarnpkg.com/power-assert-context-traversal/-/power-assert-context-traversal-1.2.0.tgz#f6e71454baf640de5c1c9c270349f5c9ab0b2e94" 2117 | dependencies: 2118 | core-js "^2.0.0" 2119 | estraverse "^4.1.0" 2120 | 2121 | power-assert-formatter@^1.4.1: 2122 | version "1.4.1" 2123 | resolved "https://registry.yarnpkg.com/power-assert-formatter/-/power-assert-formatter-1.4.1.tgz#5dc125ed50a3dfb1dda26c19347f3bf58ec2884a" 2124 | dependencies: 2125 | core-js "^2.0.0" 2126 | power-assert-context-formatter "^1.0.7" 2127 | power-assert-context-reducer-ast "^1.0.7" 2128 | power-assert-renderer-assertion "^1.0.7" 2129 | power-assert-renderer-comparison "^1.0.7" 2130 | power-assert-renderer-diagram "^1.0.7" 2131 | power-assert-renderer-file "^1.0.7" 2132 | 2133 | power-assert-renderer-assertion@^1.0.7: 2134 | version "1.2.0" 2135 | resolved "https://registry.yarnpkg.com/power-assert-renderer-assertion/-/power-assert-renderer-assertion-1.2.0.tgz#3db6ffcda106b37bc1e06432ad0d748a682b147a" 2136 | dependencies: 2137 | power-assert-renderer-base "^1.1.1" 2138 | power-assert-util-string-width "^1.2.0" 2139 | 2140 | power-assert-renderer-base@^1.1.1: 2141 | version "1.1.1" 2142 | resolved "https://registry.yarnpkg.com/power-assert-renderer-base/-/power-assert-renderer-base-1.1.1.tgz#96a650c6fd05ee1bc1f66b54ad61442c8b3f63eb" 2143 | 2144 | power-assert-renderer-comparison@^1.0.7: 2145 | version "1.2.0" 2146 | resolved "https://registry.yarnpkg.com/power-assert-renderer-comparison/-/power-assert-renderer-comparison-1.2.0.tgz#e4f88113225a69be8aa586ead05aef99462c0495" 2147 | dependencies: 2148 | core-js "^2.0.0" 2149 | diff-match-patch "^1.0.0" 2150 | power-assert-renderer-base "^1.1.1" 2151 | stringifier "^1.3.0" 2152 | type-name "^2.0.1" 2153 | 2154 | power-assert-renderer-diagram@^1.0.7: 2155 | version "1.2.0" 2156 | resolved "https://registry.yarnpkg.com/power-assert-renderer-diagram/-/power-assert-renderer-diagram-1.2.0.tgz#37f66e8542e5677c5b58e6d72b01c0d9a30e2219" 2157 | dependencies: 2158 | core-js "^2.0.0" 2159 | power-assert-renderer-base "^1.1.1" 2160 | power-assert-util-string-width "^1.2.0" 2161 | stringifier "^1.3.0" 2162 | 2163 | power-assert-renderer-file@^1.0.7: 2164 | version "1.2.0" 2165 | resolved "https://registry.yarnpkg.com/power-assert-renderer-file/-/power-assert-renderer-file-1.2.0.tgz#3f4bebd9e1455d75cf2ac541e7bb515a87d4ce4b" 2166 | dependencies: 2167 | power-assert-renderer-base "^1.1.1" 2168 | 2169 | power-assert-util-string-width@^1.2.0: 2170 | version "1.2.0" 2171 | resolved "https://registry.yarnpkg.com/power-assert-util-string-width/-/power-assert-util-string-width-1.2.0.tgz#6e06d5e3581bb876c5d377c53109fffa95bd91a0" 2172 | dependencies: 2173 | eastasianwidth "^0.2.0" 2174 | 2175 | power-assert@^1.4.4: 2176 | version "1.6.0" 2177 | resolved "https://registry.yarnpkg.com/power-assert/-/power-assert-1.6.0.tgz#3a9d2b943cf0d6fc6a623766869c4460838c05fb" 2178 | dependencies: 2179 | define-properties "^1.1.2" 2180 | empower "^1.3.0" 2181 | power-assert-formatter "^1.4.1" 2182 | universal-deep-strict-equal "^1.2.1" 2183 | xtend "^4.0.0" 2184 | 2185 | process-nextick-args@~2.0.0: 2186 | version "2.0.0" 2187 | resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.0.tgz#a37d732f4271b4ab1ad070d35508e8290788ffaa" 2188 | 2189 | protobufjs@^5.0.3: 2190 | version "5.0.3" 2191 | resolved "https://registry.yarnpkg.com/protobufjs/-/protobufjs-5.0.3.tgz#e4dfe9fb67c90b2630d15868249bcc4961467a17" 2192 | dependencies: 2193 | ascli "~1" 2194 | bytebuffer "~5" 2195 | glob "^7.0.5" 2196 | yargs "^3.10.0" 2197 | 2198 | protobufjs@^6.8.0, protobufjs@^6.8.6: 2199 | version "6.8.8" 2200 | resolved "https://registry.yarnpkg.com/protobufjs/-/protobufjs-6.8.8.tgz#c8b4f1282fd7a90e6f5b109ed11c84af82908e7c" 2201 | dependencies: 2202 | "@protobufjs/aspromise" "^1.1.2" 2203 | "@protobufjs/base64" "^1.1.2" 2204 | "@protobufjs/codegen" "^2.0.4" 2205 | "@protobufjs/eventemitter" "^1.1.0" 2206 | "@protobufjs/fetch" "^1.1.0" 2207 | "@protobufjs/float" "^1.0.2" 2208 | "@protobufjs/inquire" "^1.1.0" 2209 | "@protobufjs/path" "^1.1.2" 2210 | "@protobufjs/pool" "^1.1.0" 2211 | "@protobufjs/utf8" "^1.1.0" 2212 | "@types/long" "^4.0.0" 2213 | "@types/node" "^10.1.0" 2214 | long "^4.0.0" 2215 | 2216 | proxy-addr@~2.0.3: 2217 | version "2.0.3" 2218 | resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.3.tgz#355f262505a621646b3130a728eb647e22055341" 2219 | dependencies: 2220 | forwarded "~0.1.2" 2221 | ipaddr.js "1.6.0" 2222 | 2223 | pseudomap@^1.0.2: 2224 | version "1.0.2" 2225 | resolved "https://registry.yarnpkg.com/pseudomap/-/pseudomap-1.0.2.tgz#f052a28da70e618917ef0a8ac34c1ae5a68286b3" 2226 | 2227 | pump@^2.0.0: 2228 | version "2.0.1" 2229 | resolved "https://registry.yarnpkg.com/pump/-/pump-2.0.1.tgz#12399add6e4cf7526d973cbc8b5ce2e2908b3909" 2230 | dependencies: 2231 | end-of-stream "^1.1.0" 2232 | once "^1.3.1" 2233 | 2234 | pumpify@^1.4.0, pumpify@^1.5.1: 2235 | version "1.5.1" 2236 | resolved "https://registry.yarnpkg.com/pumpify/-/pumpify-1.5.1.tgz#36513be246ab27570b1a374a5ce278bfd74370ce" 2237 | dependencies: 2238 | duplexify "^3.6.0" 2239 | inherits "^2.0.3" 2240 | pump "^2.0.0" 2241 | 2242 | punycode@^1.4.1: 2243 | version "1.4.1" 2244 | resolved "https://registry.yarnpkg.com/punycode/-/punycode-1.4.1.tgz#c0d5a63b2718800ad8e1eb0fa5269c84dd41845e" 2245 | 2246 | qs@6.5.1: 2247 | version "6.5.1" 2248 | resolved "https://registry.yarnpkg.com/qs/-/qs-6.5.1.tgz#349cdf6eef89ec45c12d7d5eb3fc0c870343a6d8" 2249 | 2250 | qs@~6.5.1: 2251 | version "6.5.2" 2252 | resolved "https://registry.yarnpkg.com/qs/-/qs-6.5.2.tgz#cb3ae806e8740444584ef154ce8ee98d403f3e36" 2253 | 2254 | range-parser@~1.2.0: 2255 | version "1.2.0" 2256 | resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.0.tgz#f49be6b487894ddc40dcc94a322f611092e00d5e" 2257 | 2258 | raw-body@2.3.2: 2259 | version "2.3.2" 2260 | resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.3.2.tgz#bcd60c77d3eb93cde0050295c3f379389bc88f89" 2261 | dependencies: 2262 | bytes "3.0.0" 2263 | http-errors "1.6.2" 2264 | iconv-lite "0.4.19" 2265 | unpipe "1.0.0" 2266 | 2267 | rc@^1.2.7: 2268 | version "1.2.8" 2269 | resolved "https://registry.yarnpkg.com/rc/-/rc-1.2.8.tgz#cd924bf5200a075b83c188cd6b9e211b7fc0d3ed" 2270 | dependencies: 2271 | deep-extend "^0.6.0" 2272 | ini "~1.3.0" 2273 | minimist "^1.2.0" 2274 | strip-json-comments "~2.0.1" 2275 | 2276 | readable-stream@^2.0.0, readable-stream@^2.0.6, readable-stream@^2.1.5, readable-stream@^2.2.2: 2277 | version "2.3.6" 2278 | resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.6.tgz#b11c27d88b8ff1fbe070643cf94b0c79ae1b0aaf" 2279 | dependencies: 2280 | core-util-is "~1.0.0" 2281 | inherits "~2.0.3" 2282 | isarray "~1.0.0" 2283 | process-nextick-args "~2.0.0" 2284 | safe-buffer "~5.1.1" 2285 | string_decoder "~1.1.1" 2286 | util-deprecate "~1.0.1" 2287 | 2288 | readable-stream@~1.0.32: 2289 | version "1.0.34" 2290 | resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-1.0.34.tgz#125820e34bc842d2f2aaafafe4c2916ee32c157c" 2291 | dependencies: 2292 | core-util-is "~1.0.0" 2293 | inherits "~2.0.1" 2294 | isarray "0.0.1" 2295 | string_decoder "~0.10.x" 2296 | 2297 | regex-not@^1.0.0, regex-not@^1.0.2: 2298 | version "1.0.2" 2299 | resolved "https://registry.yarnpkg.com/regex-not/-/regex-not-1.0.2.tgz#1f4ece27e00b0b65e0247a6810e6a85d83a5752c" 2300 | dependencies: 2301 | extend-shallow "^3.0.2" 2302 | safe-regex "^1.1.0" 2303 | 2304 | repeat-element@^1.1.2: 2305 | version "1.1.2" 2306 | resolved "https://registry.yarnpkg.com/repeat-element/-/repeat-element-1.1.2.tgz#ef089a178d1483baae4d93eb98b4f9e4e11d990a" 2307 | 2308 | repeat-string@^1.6.1: 2309 | version "1.6.1" 2310 | resolved "https://registry.yarnpkg.com/repeat-string/-/repeat-string-1.6.1.tgz#8dcae470e1c88abc2d600fff4a776286da75e637" 2311 | 2312 | request@^2.79.0, request@^2.81.0, request@^2.85.0, request@^2.87.0: 2313 | version "2.87.0" 2314 | resolved "https://registry.yarnpkg.com/request/-/request-2.87.0.tgz#32f00235cd08d482b4d0d68db93a829c0ed5756e" 2315 | dependencies: 2316 | aws-sign2 "~0.7.0" 2317 | aws4 "^1.6.0" 2318 | caseless "~0.12.0" 2319 | combined-stream "~1.0.5" 2320 | extend "~3.0.1" 2321 | forever-agent "~0.6.1" 2322 | form-data "~2.3.1" 2323 | har-validator "~5.0.3" 2324 | http-signature "~1.2.0" 2325 | is-typedarray "~1.0.0" 2326 | isstream "~0.1.2" 2327 | json-stringify-safe "~5.0.1" 2328 | mime-types "~2.1.17" 2329 | oauth-sign "~0.8.2" 2330 | performance-now "^2.1.0" 2331 | qs "~6.5.1" 2332 | safe-buffer "^5.1.1" 2333 | tough-cookie "~2.3.3" 2334 | tunnel-agent "^0.6.0" 2335 | uuid "^3.1.0" 2336 | 2337 | resolve-url@^0.2.1: 2338 | version "0.2.1" 2339 | resolved "https://registry.yarnpkg.com/resolve-url/-/resolve-url-0.2.1.tgz#2c637fe77c893afd2a663fe21aa9080068e2052a" 2340 | 2341 | resolve@^1.3.2: 2342 | version "1.8.1" 2343 | resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.8.1.tgz#82f1ec19a423ac1fbd080b0bab06ba36e84a7a26" 2344 | dependencies: 2345 | path-parse "^1.0.5" 2346 | 2347 | ret@~0.1.10: 2348 | version "0.1.15" 2349 | resolved "https://registry.yarnpkg.com/ret/-/ret-0.1.15.tgz#b8a4825d5bdb1fc3f6f53c2bc33f81388681c7bc" 2350 | 2351 | retry-axios@0.3.2, retry-axios@^0.3.2: 2352 | version "0.3.2" 2353 | resolved "https://registry.yarnpkg.com/retry-axios/-/retry-axios-0.3.2.tgz#5757c80f585b4cc4c4986aa2ffd47a60c6d35e13" 2354 | 2355 | retry-request@^3.0.0: 2356 | version "3.3.2" 2357 | resolved "https://registry.yarnpkg.com/retry-request/-/retry-request-3.3.2.tgz#fd8e0079e7b0dfc7056e500b6f089437db0da4df" 2358 | dependencies: 2359 | request "^2.81.0" 2360 | through2 "^2.0.0" 2361 | 2362 | retry-request@^4.0.0: 2363 | version "4.0.0" 2364 | resolved "https://registry.yarnpkg.com/retry-request/-/retry-request-4.0.0.tgz#5c366166279b3e10e9d7aa13274467a05cb69290" 2365 | dependencies: 2366 | through2 "^2.0.0" 2367 | 2368 | rimraf@^2.6.1: 2369 | version "2.6.2" 2370 | resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-2.6.2.tgz#2ed8150d24a16ea8651e6d6ef0f47c4158ce7a36" 2371 | dependencies: 2372 | glob "^7.0.5" 2373 | 2374 | safe-buffer@5.1.1: 2375 | version "5.1.1" 2376 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.1.tgz#893312af69b2123def71f57889001671eeb2c853" 2377 | 2378 | safe-buffer@^5.0.1, safe-buffer@^5.1.1, safe-buffer@^5.1.2, safe-buffer@~5.1.0, safe-buffer@~5.1.1: 2379 | version "5.1.2" 2380 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d" 2381 | 2382 | safe-regex@^1.1.0: 2383 | version "1.1.0" 2384 | resolved "https://registry.yarnpkg.com/safe-regex/-/safe-regex-1.1.0.tgz#40a3669f3b077d1e943d44629e157dd48023bf2e" 2385 | dependencies: 2386 | ret "~0.1.10" 2387 | 2388 | "safer-buffer@>= 2.1.2 < 3", safer-buffer@^2.0.2: 2389 | version "2.1.2" 2390 | resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" 2391 | 2392 | sax@^1.2.4: 2393 | version "1.2.4" 2394 | resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9" 2395 | 2396 | secure-random-string@^1.1.0: 2397 | version "1.1.0" 2398 | resolved "https://registry.yarnpkg.com/secure-random-string/-/secure-random-string-1.1.0.tgz#c47d2f20b6d93db1255edb4dadf5e03188ab978e" 2399 | 2400 | semver@^5.3.0: 2401 | version "5.5.0" 2402 | resolved "https://registry.yarnpkg.com/semver/-/semver-5.5.0.tgz#dc4bbc7a6ca9d916dee5d43516f0092b58f7b8ab" 2403 | 2404 | send@0.16.2: 2405 | version "0.16.2" 2406 | resolved "https://registry.yarnpkg.com/send/-/send-0.16.2.tgz#6ecca1e0f8c156d141597559848df64730a6bbc1" 2407 | dependencies: 2408 | debug "2.6.9" 2409 | depd "~1.1.2" 2410 | destroy "~1.0.4" 2411 | encodeurl "~1.0.2" 2412 | escape-html "~1.0.3" 2413 | etag "~1.8.1" 2414 | fresh "0.5.2" 2415 | http-errors "~1.6.2" 2416 | mime "1.4.1" 2417 | ms "2.0.0" 2418 | on-finished "~2.3.0" 2419 | range-parser "~1.2.0" 2420 | statuses "~1.4.0" 2421 | 2422 | serve-static@1.13.2: 2423 | version "1.13.2" 2424 | resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.13.2.tgz#095e8472fd5b46237db50ce486a43f4b86c6cec1" 2425 | dependencies: 2426 | encodeurl "~1.0.2" 2427 | escape-html "~1.0.3" 2428 | parseurl "~1.3.2" 2429 | send "0.16.2" 2430 | 2431 | set-blocking@~2.0.0: 2432 | version "2.0.0" 2433 | resolved "https://registry.yarnpkg.com/set-blocking/-/set-blocking-2.0.0.tgz#045f9782d011ae9a6803ddd382b24392b3d890f7" 2434 | 2435 | set-value@^0.4.3: 2436 | version "0.4.3" 2437 | resolved "https://registry.yarnpkg.com/set-value/-/set-value-0.4.3.tgz#7db08f9d3d22dc7f78e53af3c3bf4666ecdfccf1" 2438 | dependencies: 2439 | extend-shallow "^2.0.1" 2440 | is-extendable "^0.1.1" 2441 | is-plain-object "^2.0.1" 2442 | to-object-path "^0.3.0" 2443 | 2444 | set-value@^2.0.0: 2445 | version "2.0.0" 2446 | resolved "https://registry.yarnpkg.com/set-value/-/set-value-2.0.0.tgz#71ae4a88f0feefbbf52d1ea604f3fb315ebb6274" 2447 | dependencies: 2448 | extend-shallow "^2.0.1" 2449 | is-extendable "^0.1.1" 2450 | is-plain-object "^2.0.3" 2451 | split-string "^3.0.1" 2452 | 2453 | setprototypeof@1.0.3: 2454 | version "1.0.3" 2455 | resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.0.3.tgz#66567e37043eeb4f04d91bd658c0cbefb55b8e04" 2456 | 2457 | setprototypeof@1.1.0: 2458 | version "1.1.0" 2459 | resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.1.0.tgz#d0bd85536887b6fe7c0d818cb962d9d91c54e656" 2460 | 2461 | signal-exit@^3.0.0, signal-exit@^3.0.2: 2462 | version "3.0.2" 2463 | resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.2.tgz#b5fdc08f1287ea1178628e415e25132b73646c6d" 2464 | 2465 | slash@^1.0.0: 2466 | version "1.0.0" 2467 | resolved "https://registry.yarnpkg.com/slash/-/slash-1.0.0.tgz#c41f2f6c39fc16d1cd17ad4b5d896114ae470d55" 2468 | 2469 | snakeize@^0.1.0: 2470 | version "0.1.0" 2471 | resolved "https://registry.yarnpkg.com/snakeize/-/snakeize-0.1.0.tgz#10c088d8b58eb076b3229bb5a04e232ce126422d" 2472 | 2473 | snapdragon-node@^2.0.1: 2474 | version "2.1.1" 2475 | resolved "https://registry.yarnpkg.com/snapdragon-node/-/snapdragon-node-2.1.1.tgz#6c175f86ff14bdb0724563e8f3c1b021a286853b" 2476 | dependencies: 2477 | define-property "^1.0.0" 2478 | isobject "^3.0.0" 2479 | snapdragon-util "^3.0.1" 2480 | 2481 | snapdragon-util@^3.0.1: 2482 | version "3.0.1" 2483 | resolved "https://registry.yarnpkg.com/snapdragon-util/-/snapdragon-util-3.0.1.tgz#f956479486f2acd79700693f6f7b805e45ab56e2" 2484 | dependencies: 2485 | kind-of "^3.2.0" 2486 | 2487 | snapdragon@^0.8.1: 2488 | version "0.8.2" 2489 | resolved "https://registry.yarnpkg.com/snapdragon/-/snapdragon-0.8.2.tgz#64922e7c565b0e14204ba1aa7d6964278d25182d" 2490 | dependencies: 2491 | base "^0.11.1" 2492 | debug "^2.2.0" 2493 | define-property "^0.2.5" 2494 | extend-shallow "^2.0.1" 2495 | map-cache "^0.2.2" 2496 | source-map "^0.5.6" 2497 | source-map-resolve "^0.5.0" 2498 | use "^3.1.0" 2499 | 2500 | source-map-resolve@^0.5.0: 2501 | version "0.5.2" 2502 | resolved "https://registry.yarnpkg.com/source-map-resolve/-/source-map-resolve-0.5.2.tgz#72e2cc34095543e43b2c62b2c4c10d4a9054f259" 2503 | dependencies: 2504 | atob "^2.1.1" 2505 | decode-uri-component "^0.2.0" 2506 | resolve-url "^0.2.1" 2507 | source-map-url "^0.4.0" 2508 | urix "^0.1.0" 2509 | 2510 | source-map-url@^0.4.0: 2511 | version "0.4.0" 2512 | resolved "https://registry.yarnpkg.com/source-map-url/-/source-map-url-0.4.0.tgz#3e935d7ddd73631b97659956d55128e87b5084a3" 2513 | 2514 | source-map@^0.5.6: 2515 | version "0.5.7" 2516 | resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.7.tgz#8a039d2d1021d22d1ea14c80d8ea468ba2ef3fcc" 2517 | 2518 | split-array-stream@^1.0.0: 2519 | version "1.0.3" 2520 | resolved "https://registry.yarnpkg.com/split-array-stream/-/split-array-stream-1.0.3.tgz#d2b75a8e5e0d824d52fdec8b8225839dc2e35dfa" 2521 | dependencies: 2522 | async "^2.4.0" 2523 | is-stream-ended "^0.1.0" 2524 | 2525 | split-array-stream@^2.0.0: 2526 | version "2.0.0" 2527 | resolved "https://registry.yarnpkg.com/split-array-stream/-/split-array-stream-2.0.0.tgz#85a4f8bfe14421d7bca7f33a6d176d0c076a53b1" 2528 | dependencies: 2529 | is-stream-ended "^0.1.4" 2530 | 2531 | split-string@^3.0.1, split-string@^3.0.2: 2532 | version "3.1.0" 2533 | resolved "https://registry.yarnpkg.com/split-string/-/split-string-3.1.0.tgz#7cb09dda3a86585705c64b39a6466038682e8fe2" 2534 | dependencies: 2535 | extend-shallow "^3.0.0" 2536 | 2537 | sprintf-js@~1.0.2: 2538 | version "1.0.3" 2539 | resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c" 2540 | 2541 | sshpk@^1.7.0: 2542 | version "1.14.2" 2543 | resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.14.2.tgz#c6fc61648a3d9c4e764fd3fcdf4ea105e492ba98" 2544 | dependencies: 2545 | asn1 "~0.2.3" 2546 | assert-plus "^1.0.0" 2547 | dashdash "^1.12.0" 2548 | getpass "^0.1.1" 2549 | safer-buffer "^2.0.2" 2550 | optionalDependencies: 2551 | bcrypt-pbkdf "^1.0.0" 2552 | ecc-jsbn "~0.1.1" 2553 | jsbn "~0.1.0" 2554 | tweetnacl "~0.14.0" 2555 | 2556 | static-extend@^0.1.1: 2557 | version "0.1.2" 2558 | resolved "https://registry.yarnpkg.com/static-extend/-/static-extend-0.1.2.tgz#60809c39cbff55337226fd5e0b520f341f1fb5c6" 2559 | dependencies: 2560 | define-property "^0.2.5" 2561 | object-copy "^0.1.0" 2562 | 2563 | "statuses@>= 1.3.1 < 2", "statuses@>= 1.4.0 < 2": 2564 | version "1.5.0" 2565 | resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.5.0.tgz#161c7dac177659fd9811f43771fa99381478628c" 2566 | 2567 | statuses@~1.4.0: 2568 | version "1.4.0" 2569 | resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.4.0.tgz#bb73d446da2796106efcc1b601a253d6c46bd087" 2570 | 2571 | stream-events@^1.0.1, stream-events@^1.0.3, stream-events@^1.0.4: 2572 | version "1.0.4" 2573 | resolved "https://registry.yarnpkg.com/stream-events/-/stream-events-1.0.4.tgz#73bfd4007b8f677b46ec699f14e9e2304c2f0a9e" 2574 | dependencies: 2575 | stubs "^3.0.0" 2576 | 2577 | stream-shift@^1.0.0: 2578 | version "1.0.0" 2579 | resolved "https://registry.yarnpkg.com/stream-shift/-/stream-shift-1.0.0.tgz#d5c752825e5367e786f78e18e445ea223a155952" 2580 | 2581 | string-format-obj@^1.1.0: 2582 | version "1.1.1" 2583 | resolved "https://registry.yarnpkg.com/string-format-obj/-/string-format-obj-1.1.1.tgz#c7612ca4e2ad923812a81db192dc291850aa1f65" 2584 | 2585 | string-width@^1.0.1: 2586 | version "1.0.2" 2587 | resolved "https://registry.yarnpkg.com/string-width/-/string-width-1.0.2.tgz#118bdf5b8cdc51a2a7e70d211e07e2b0b9b107d3" 2588 | dependencies: 2589 | code-point-at "^1.0.0" 2590 | is-fullwidth-code-point "^1.0.0" 2591 | strip-ansi "^3.0.0" 2592 | 2593 | "string-width@^1.0.2 || 2": 2594 | version "2.1.1" 2595 | resolved "https://registry.yarnpkg.com/string-width/-/string-width-2.1.1.tgz#ab93f27a8dc13d28cac815c462143a6d9012ae9e" 2596 | dependencies: 2597 | is-fullwidth-code-point "^2.0.0" 2598 | strip-ansi "^4.0.0" 2599 | 2600 | string_decoder@~0.10.x: 2601 | version "0.10.31" 2602 | resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-0.10.31.tgz#62e203bc41766c6c28c9fc84301dab1c5310fa94" 2603 | 2604 | string_decoder@~1.1.1: 2605 | version "1.1.1" 2606 | resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8" 2607 | dependencies: 2608 | safe-buffer "~5.1.0" 2609 | 2610 | stringifier@^1.3.0: 2611 | version "1.3.0" 2612 | resolved "https://registry.yarnpkg.com/stringifier/-/stringifier-1.3.0.tgz#def18342f6933db0f2dbfc9aa02175b448c17959" 2613 | dependencies: 2614 | core-js "^2.0.0" 2615 | traverse "^0.6.6" 2616 | type-name "^2.0.1" 2617 | 2618 | strip-ansi@^3.0.0, strip-ansi@^3.0.1: 2619 | version "3.0.1" 2620 | resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-3.0.1.tgz#6a385fb8853d952d5ff05d0e8aaf94278dc63dcf" 2621 | dependencies: 2622 | ansi-regex "^2.0.0" 2623 | 2624 | strip-ansi@^4.0.0: 2625 | version "4.0.0" 2626 | resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-4.0.0.tgz#a8479022eb1ac368a871389b635262c505ee368f" 2627 | dependencies: 2628 | ansi-regex "^3.0.0" 2629 | 2630 | strip-json-comments@~2.0.1: 2631 | version "2.0.1" 2632 | resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz#3c531942e908c2697c0ec344858c286c7ca0a60a" 2633 | 2634 | stubs@^3.0.0: 2635 | version "3.0.0" 2636 | resolved "https://registry.yarnpkg.com/stubs/-/stubs-3.0.0.tgz#e8d2ba1fa9c90570303c030b6900f7d5f89abe5b" 2637 | 2638 | supports-color@^2.0.0: 2639 | version "2.0.0" 2640 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-2.0.0.tgz#535d045ce6b6363fa40117084629995e9df324c7" 2641 | 2642 | supports-color@^5.3.0: 2643 | version "5.4.0" 2644 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.4.0.tgz#1c6b337402c2137605efe19f10fec390f6faab54" 2645 | dependencies: 2646 | has-flag "^3.0.0" 2647 | 2648 | tar@^4: 2649 | version "4.4.4" 2650 | resolved "https://registry.yarnpkg.com/tar/-/tar-4.4.4.tgz#ec8409fae9f665a4355cc3b4087d0820232bb8cd" 2651 | dependencies: 2652 | chownr "^1.0.1" 2653 | fs-minipass "^1.2.5" 2654 | minipass "^2.3.3" 2655 | minizlib "^1.1.0" 2656 | mkdirp "^0.5.0" 2657 | safe-buffer "^5.1.2" 2658 | yallist "^3.0.2" 2659 | 2660 | through2@^2.0.0, through2@^2.0.3: 2661 | version "2.0.3" 2662 | resolved "https://registry.yarnpkg.com/through2/-/through2-2.0.3.tgz#0004569b37c7c74ba39c43f3ced78d1ad94140be" 2663 | dependencies: 2664 | readable-stream "^2.1.5" 2665 | xtend "~4.0.1" 2666 | 2667 | to-object-path@^0.3.0: 2668 | version "0.3.0" 2669 | resolved "https://registry.yarnpkg.com/to-object-path/-/to-object-path-0.3.0.tgz#297588b7b0e7e0ac08e04e672f85c1f4999e17af" 2670 | dependencies: 2671 | kind-of "^3.0.2" 2672 | 2673 | to-regex-range@^2.1.0: 2674 | version "2.1.1" 2675 | resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-2.1.1.tgz#7c80c17b9dfebe599e27367e0d4dd5590141db38" 2676 | dependencies: 2677 | is-number "^3.0.0" 2678 | repeat-string "^1.6.1" 2679 | 2680 | to-regex@^3.0.1, to-regex@^3.0.2: 2681 | version "3.0.2" 2682 | resolved "https://registry.yarnpkg.com/to-regex/-/to-regex-3.0.2.tgz#13cfdd9b336552f30b51f33a8ae1b42a7a7599ce" 2683 | dependencies: 2684 | define-property "^2.0.2" 2685 | extend-shallow "^3.0.2" 2686 | regex-not "^1.0.2" 2687 | safe-regex "^1.1.0" 2688 | 2689 | tough-cookie@~2.3.3: 2690 | version "2.3.4" 2691 | resolved "https://registry.yarnpkg.com/tough-cookie/-/tough-cookie-2.3.4.tgz#ec60cee38ac675063ffc97a5c18970578ee83655" 2692 | dependencies: 2693 | punycode "^1.4.1" 2694 | 2695 | traverse@^0.6.6: 2696 | version "0.6.6" 2697 | resolved "https://registry.yarnpkg.com/traverse/-/traverse-0.6.6.tgz#cbdf560fd7b9af632502fed40f918c157ea97137" 2698 | 2699 | tslib@1.9.0: 2700 | version "1.9.0" 2701 | resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.9.0.tgz#e37a86fda8cbbaf23a057f473c9f4dc64e5fc2e8" 2702 | 2703 | tslib@^1.8.0, tslib@^1.8.1: 2704 | version "1.9.3" 2705 | resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.9.3.tgz#d7e4dd79245d85428c4d7e4822a79917954ca286" 2706 | 2707 | tslint@^5.8.0: 2708 | version "5.11.0" 2709 | resolved "https://registry.yarnpkg.com/tslint/-/tslint-5.11.0.tgz#98f30c02eae3cde7006201e4c33cb08b48581eed" 2710 | dependencies: 2711 | babel-code-frame "^6.22.0" 2712 | builtin-modules "^1.1.1" 2713 | chalk "^2.3.0" 2714 | commander "^2.12.1" 2715 | diff "^3.2.0" 2716 | glob "^7.1.1" 2717 | js-yaml "^3.7.0" 2718 | minimatch "^3.0.4" 2719 | resolve "^1.3.2" 2720 | semver "^5.3.0" 2721 | tslib "^1.8.0" 2722 | tsutils "^2.27.2" 2723 | 2724 | tsutils@^2.27.2: 2725 | version "2.29.0" 2726 | resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-2.29.0.tgz#32b488501467acbedd4b85498673a0812aca0b99" 2727 | dependencies: 2728 | tslib "^1.8.1" 2729 | 2730 | tunnel-agent@^0.6.0: 2731 | version "0.6.0" 2732 | resolved "https://registry.yarnpkg.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz#27a5dea06b36b04a0a9966774b290868f0fc40fd" 2733 | dependencies: 2734 | safe-buffer "^5.0.1" 2735 | 2736 | tweetnacl@^0.14.3, tweetnacl@~0.14.0: 2737 | version "0.14.5" 2738 | resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-0.14.5.tgz#5ae68177f192d4456269d108afa93ff8743f4f64" 2739 | 2740 | type-is@~1.6.15, type-is@~1.6.16: 2741 | version "1.6.16" 2742 | resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.16.tgz#f89ce341541c672b25ee7ae3c73dee3b2be50194" 2743 | dependencies: 2744 | media-typer "0.3.0" 2745 | mime-types "~2.1.18" 2746 | 2747 | type-name@^2.0.1: 2748 | version "2.0.2" 2749 | resolved "https://registry.yarnpkg.com/type-name/-/type-name-2.0.2.tgz#efe7d4123d8ac52afff7f40c7e4dec5266008fb4" 2750 | 2751 | typedarray@^0.0.6: 2752 | version "0.0.6" 2753 | resolved "https://registry.yarnpkg.com/typedarray/-/typedarray-0.0.6.tgz#867ac74e3864187b1d3d47d996a78ec5c8830777" 2754 | 2755 | typescript@^2.5.3: 2756 | version "2.9.2" 2757 | resolved "https://registry.yarnpkg.com/typescript/-/typescript-2.9.2.tgz#1cbf61d05d6b96269244eb6a3bce4bd914e0f00c" 2758 | 2759 | union-value@^1.0.0: 2760 | version "1.0.0" 2761 | resolved "https://registry.yarnpkg.com/union-value/-/union-value-1.0.0.tgz#5c71c34cb5bad5dcebe3ea0cd08207ba5aa1aea4" 2762 | dependencies: 2763 | arr-union "^3.1.0" 2764 | get-value "^2.0.6" 2765 | is-extendable "^0.1.1" 2766 | set-value "^0.4.3" 2767 | 2768 | unique-string@^1.0.0: 2769 | version "1.0.0" 2770 | resolved "https://registry.yarnpkg.com/unique-string/-/unique-string-1.0.0.tgz#9e1057cca851abb93398f8b33ae187b99caec11a" 2771 | dependencies: 2772 | crypto-random-string "^1.0.0" 2773 | 2774 | universal-deep-strict-equal@^1.2.1: 2775 | version "1.2.2" 2776 | resolved "https://registry.yarnpkg.com/universal-deep-strict-equal/-/universal-deep-strict-equal-1.2.2.tgz#0da4ac2f73cff7924c81fa4de018ca562ca2b0a7" 2777 | dependencies: 2778 | array-filter "^1.0.0" 2779 | indexof "0.0.1" 2780 | object-keys "^1.0.0" 2781 | 2782 | unpipe@1.0.0, unpipe@~1.0.0: 2783 | version "1.0.0" 2784 | resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec" 2785 | 2786 | unset-value@^1.0.0: 2787 | version "1.0.0" 2788 | resolved "https://registry.yarnpkg.com/unset-value/-/unset-value-1.0.0.tgz#8376873f7d2335179ffb1e6fc3a8ed0dfc8ab559" 2789 | dependencies: 2790 | has-value "^0.3.1" 2791 | isobject "^3.0.0" 2792 | 2793 | urix@^0.1.0: 2794 | version "0.1.0" 2795 | resolved "https://registry.yarnpkg.com/urix/-/urix-0.1.0.tgz#da937f7a62e21fec1fd18d49b35c2935067a6c72" 2796 | 2797 | use@^3.1.0: 2798 | version "3.1.1" 2799 | resolved "https://registry.yarnpkg.com/use/-/use-3.1.1.tgz#d50c8cac79a19fbc20f2911f56eb973f4e10070f" 2800 | 2801 | util-deprecate@~1.0.1: 2802 | version "1.0.2" 2803 | resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" 2804 | 2805 | utils-merge@1.0.1: 2806 | version "1.0.1" 2807 | resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713" 2808 | 2809 | uuid@^3.1.0: 2810 | version "3.3.2" 2811 | resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.3.2.tgz#1b4af4955eb3077c501c23872fc6513811587131" 2812 | 2813 | vary@^1, vary@~1.1.2: 2814 | version "1.1.2" 2815 | resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc" 2816 | 2817 | verror@1.10.0: 2818 | version "1.10.0" 2819 | resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.0.tgz#3a105ca17053af55d6e270c1f8288682e18da400" 2820 | dependencies: 2821 | assert-plus "^1.0.0" 2822 | core-util-is "1.0.2" 2823 | extsprintf "^1.2.0" 2824 | 2825 | websocket-driver@>=0.5.1: 2826 | version "0.7.0" 2827 | resolved "https://registry.yarnpkg.com/websocket-driver/-/websocket-driver-0.7.0.tgz#0caf9d2d755d93aee049d4bdd0d3fe2cca2a24eb" 2828 | dependencies: 2829 | http-parser-js ">=0.4.0" 2830 | websocket-extensions ">=0.1.1" 2831 | 2832 | websocket-extensions@>=0.1.1: 2833 | version "0.1.3" 2834 | resolved "https://registry.yarnpkg.com/websocket-extensions/-/websocket-extensions-0.1.3.tgz#5d2ff22977003ec687a4b87073dfbbac146ccf29" 2835 | 2836 | wide-align@^1.1.0: 2837 | version "1.1.3" 2838 | resolved "https://registry.yarnpkg.com/wide-align/-/wide-align-1.1.3.tgz#ae074e6bdc0c14a431e804e624549c633b000457" 2839 | dependencies: 2840 | string-width "^1.0.2 || 2" 2841 | 2842 | window-size@^0.1.4: 2843 | version "0.1.4" 2844 | resolved "https://registry.yarnpkg.com/window-size/-/window-size-0.1.4.tgz#f8e1aa1ee5a53ec5bf151ffa09742a6ad7697876" 2845 | 2846 | wrap-ansi@^2.0.0: 2847 | version "2.1.0" 2848 | resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-2.1.0.tgz#d8fc3d284dd05794fe84973caecdd1cf824fdd85" 2849 | dependencies: 2850 | string-width "^1.0.1" 2851 | strip-ansi "^3.0.1" 2852 | 2853 | wrappy@1: 2854 | version "1.0.2" 2855 | resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" 2856 | 2857 | write-file-atomic@^2.0.0: 2858 | version "2.3.0" 2859 | resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-2.3.0.tgz#1ff61575c2e2a4e8e510d6fa4e243cce183999ab" 2860 | dependencies: 2861 | graceful-fs "^4.1.11" 2862 | imurmurhash "^0.1.4" 2863 | signal-exit "^3.0.2" 2864 | 2865 | xdg-basedir@^3.0.0: 2866 | version "3.0.0" 2867 | resolved "https://registry.yarnpkg.com/xdg-basedir/-/xdg-basedir-3.0.0.tgz#496b2cc109eca8dbacfe2dc72b603c17c5870ad4" 2868 | 2869 | xmlhttprequest@1.8.0: 2870 | version "1.8.0" 2871 | resolved "https://registry.yarnpkg.com/xmlhttprequest/-/xmlhttprequest-1.8.0.tgz#67fe075c5c24fef39f9d65f5f7b7fe75171968fc" 2872 | 2873 | xtend@^4.0.0, xtend@^4.0.1, xtend@~4.0.1: 2874 | version "4.0.1" 2875 | resolved "https://registry.yarnpkg.com/xtend/-/xtend-4.0.1.tgz#a5c6d532be656e23db820efb943a1f04998d63af" 2876 | 2877 | y18n@^3.2.0: 2878 | version "3.2.1" 2879 | resolved "https://registry.yarnpkg.com/y18n/-/y18n-3.2.1.tgz#6d15fba884c08679c0d77e88e7759e811e07fa41" 2880 | 2881 | yallist@^2.1.2: 2882 | version "2.1.2" 2883 | resolved "https://registry.yarnpkg.com/yallist/-/yallist-2.1.2.tgz#1c11f9218f076089a47dd512f93c6699a6a81d52" 2884 | 2885 | yallist@^3.0.0, yallist@^3.0.2: 2886 | version "3.0.2" 2887 | resolved "https://registry.yarnpkg.com/yallist/-/yallist-3.0.2.tgz#8452b4bb7e83c7c188d8041c1a837c773d6d8bb9" 2888 | 2889 | yargs@^3.10.0: 2890 | version "3.32.0" 2891 | resolved "https://registry.yarnpkg.com/yargs/-/yargs-3.32.0.tgz#03088e9ebf9e756b69751611d2a5ef591482c995" 2892 | dependencies: 2893 | camelcase "^2.0.1" 2894 | cliui "^3.0.3" 2895 | decamelize "^1.1.1" 2896 | os-locale "^1.4.0" 2897 | string-width "^1.0.1" 2898 | window-size "^0.1.4" 2899 | y18n "^3.2.0" 2900 | --------------------------------------------------------------------------------