├── .gitignore ├── CHANGELOG.md ├── index.js ├── Gruntfile.js ├── LICENSE ├── package.json ├── README.md ├── test └── postgrestore.js ├── lib └── postgrestore.js └── yarn.lock /.gitignore: -------------------------------------------------------------------------------- 1 | .idea 2 | node_modules 3 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # 0.0.1 (2014-12-04) 2 | 3 | Initial release -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | module.exports = require('./lib/postgrestore.js'); -------------------------------------------------------------------------------- /Gruntfile.js: -------------------------------------------------------------------------------- 1 | module.exports = function(grunt) { 2 | 3 | grunt.loadNpmTasks('grunt-mocha-test'); 4 | 5 | grunt.initConfig({ 6 | mochaTest: { 7 | test: { 8 | options: { 9 | reporter: 'spec', 10 | timeout: 10000 11 | }, 12 | src: ['test/**/*.js'] 13 | } 14 | } 15 | }); 16 | 17 | grunt.registerTask('test', [ 'mochaTest']); 18 | }; -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2014 Battochon 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | 23 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "passwordless-postgrestore", 3 | "version": "1.0.0", 4 | "description": "PostgreSQL TokenStore for Passwordless", 5 | "main": "index.js", 6 | "scripts": { 7 | "test": "grunt test" 8 | }, 9 | "dependencies": { 10 | "bcrypt": "^1.0.2", 11 | "passwordless-tokenstore": "0.0.10", 12 | "pg": "^6.4.0" 13 | }, 14 | "devDependencies": { 15 | "chai": "^4.0.2", 16 | "chance": "^1.0.10", 17 | "grunt": "^1.0.1", 18 | "grunt-mocha-test": "^0.13.2", 19 | "mocha": "^3.4.2", 20 | "node-uuid": "^1.4.1", 21 | "passwordless-tokenstore-test": "^0.1.4" 22 | }, 23 | "repository": { 24 | "type": "git", 25 | "url": "https://github.com/Battochon/passwordless-postgrestore.git" 26 | }, 27 | "keywords": [ 28 | "postgresql", 29 | "passwordless", 30 | "token", 31 | "otpw", 32 | "one-time-password", 33 | "store", 34 | "tokenstore" 35 | ], 36 | "author": "Bruno MARQUES (http://marques.io)", 37 | "license": "MIT", 38 | "bugs": { 39 | "url": "https://github.com/battochon/passwordless-postgrestore/issues" 40 | }, 41 | "homepage": "https://github.com/battochon/passwordless-postgrestore" 42 | } 43 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Passwordless-PostgreStore 2 | 3 | This module provides token storage for [Passwordless](https://github.com/florianheinemann/passwordless), a node.js module for express that allows website authentication without password using verification through email or other means. Visit the project's [website](https://passwordless.net) for more details. 4 | 5 | Tokens are stored in a PostgreSQL database and are hashed and salted using [bcrypt](https://github.com/ncb000gt/node.bcrypt.js/). 6 | 7 | ## Usage 8 | 9 | First, install the module: 10 | 11 | `$ npm install passwordless-postgrestore --save` 12 | 13 | Afterwards, follow the guide for [Passwordless](https://github.com/florianheinemann/passwordless). A typical implementation may look like this: 14 | 15 | ```javascript 16 | var passwordless = require('passwordless'); 17 | var PostgreStore = require('passwordless-postgrestore'); 18 | 19 | passwordless.init(new PostgreStore('postgres://user:password@localhost/database')); 20 | 21 | passwordless.addDelivery( 22 | function(tokenToSend, uidToSend, recipient, callback) { 23 | // Send out a token 24 | }); 25 | 26 | app.use(passwordless.sessionSupport()); 27 | app.use(passwordless.acceptToken()); 28 | ``` 29 | 30 | ## Initialization 31 | 32 | ```javascript 33 | new PostgreStore(connectionString, [options]); 34 | ``` 35 | * **connectionString:** *(String)* Mandatory. PostgreSQL connection string 36 | * **[options]:** *(Object)* Optional. Some configuration option. See below exemple 37 | 38 | Example: 39 | ```javascript 40 | passwordless.init(new PostgreStore('postgres://user:password@localhost/database', { 41 | pgstore: { 42 | table: 'not_default_table_name', // *(String)* Optional. Use another table to store token, default is 'passwordless' 43 | pgPoolSize: '100' // *(Number)* Optional. Postgre client pool size 44 | } 45 | })); 46 | ``` 47 | 48 | ## PostgreSQL table creation 49 | You could use this SQL statement to create the token table, or you can customize it according to your needs : 50 | 51 | ``` 52 | CREATE TABLE passwordless 53 | ( 54 | id serial NOT NULL, 55 | uid character varying(160), 56 | token character varying(60) NOT NULL, 57 | origin text, 58 | ttl bigint, 59 | CONSTRAINT passwordless_pkey PRIMARY KEY (id), 60 | CONSTRAINT passwordless_token_key UNIQUE (token), 61 | CONSTRAINT passwordless_uid_key UNIQUE (uid) 62 | ) 63 | ``` 64 | 65 | ## Hash and salt 66 | As the tokens are equivalent to passwords (even though only for a limited time) they have to be protected in the same way. passwordless-postgrestore uses [bcrypt](https://github.com/ncb000gt/node.bcrypt.js/) with automatically created random salts. To generate the salt 10 rounds are used. 67 | 68 | ## Tests 69 | 70 | `$ npm test` 71 | 72 | ## License 73 | 74 | [MIT License](http://opensource.org/licenses/MIT) 75 | 76 | ## Author 77 | Bruno MARQUES (http://marques.io) (I just adapted code from Florian Heinemann [@thesumofall](http://twitter.com/thesumofall/)) 78 | -------------------------------------------------------------------------------- /test/postgrestore.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | var bcrypt = require('bcrypt'); 4 | var expect = require('chai').expect; 5 | var uuid = require('node-uuid'); 6 | var chance = new require('chance')(); 7 | 8 | var PostgreStore = require('../'); 9 | var TokenStore = require('passwordless-tokenstore'); 10 | 11 | var pg = require('pg'); 12 | 13 | var standardTests = require('passwordless-tokenstore-test'); 14 | 15 | function TokenStoreFactory(options) { 16 | return new PostgreStore(conString, options); 17 | } 18 | 19 | var conString = 'postgres://localhost'; 20 | var pgClient = new pg.Pool(conString); 21 | 22 | pgClient.connect(function (err) { 23 | if (err) { 24 | done(err); 25 | throw new Error('Could not connect to Postgres database, with error : ' + err); 26 | } 27 | }); 28 | 29 | var beforeEachTest = function(done) { 30 | done(); 31 | }; 32 | 33 | var afterEachTest = function(done) { 34 | done(); 35 | }; 36 | 37 | // Call all standard tests 38 | standardTests(TokenStoreFactory, beforeEachTest, afterEachTest, 1000); 39 | 40 | describe('Specific tests', function() { 41 | 42 | beforeEach(function(done) { 43 | beforeEachTest(done); 44 | }); 45 | 46 | afterEach(function(done) { 47 | afterEachTest(done); 48 | }); 49 | 50 | it('should allow the instantiation with an empty constructor', function () { 51 | expect(function() { new PostgreStore() }).to.not.throw; 52 | }); 53 | 54 | it('should allow the instantiation with host and port but no options', function () { 55 | expect(function() { new PostgreStore(conString) }).to.not.throw; 56 | }); 57 | 58 | it('should allow proper instantiation', function () { 59 | expect(function () { TokenStoreFactory() }).to.not.throw; 60 | }); 61 | 62 | it('should disconnect without errors', function () { 63 | var store = TokenStoreFactory(); 64 | expect(function () { store.disconnect() }).to.not.throw; 65 | }); 66 | 67 | it('should store tokens only in their hashed form', function(done) { 68 | var store = TokenStoreFactory(); 69 | var user = chance.email(); 70 | var token = uuid.v4(); 71 | store.storeOrUpdate(token, user, 72 | 1000*60, 'http://' + chance.domain() + '/page.html', 73 | function() { 74 | pgClient.query('SELECT * FROM passwordless WHERE uid=$1',[user], function(err, obj) { 75 | expect(err).to.not.exist; 76 | expect(obj).to.exist; 77 | expect(obj.rows[0].token).to.exist; 78 | expect(obj.rows[0].token).to.not.equal(token); 79 | done(); 80 | }) 81 | }); 82 | }); 83 | 84 | it('should respect the bcrypt difficulty option', function (done) { 85 | var store = TokenStoreFactory({ pgstore: { difficulty: 5 }}); 86 | var user = chance.email(); 87 | var token = uuid.v4(); 88 | store.storeOrUpdate(token, user, 89 | 1000 * 60, 'http://' + chance.domain() + '/page.html', 90 | function () { 91 | pgClient.query('SELECT token FROM passwordless WHERE uid=$1', [user], function (err, obj) { 92 | expect(err).to.not.exist; 93 | expect(obj).to.exist; 94 | expect(obj.rows[0].token).to.exist; 95 | expect(bcrypt.getRounds(obj.rows[0].token)).to.equal(5); 96 | done(); 97 | }) 98 | }); 99 | }); 100 | 101 | it('should store tokens not only hashed but also salted', function(done) { 102 | var store = TokenStoreFactory(); 103 | var user = chance.email(); 104 | var token = uuid.v4(); 105 | var hashedToken1; 106 | store.storeOrUpdate(token, user, 107 | 1000*60, 'http://' + chance.domain() + '/page.html', 108 | function() { 109 | pgClient.query('SELECT * FROM passwordless WHERE uid=$1',[user], function(err, obj) { 110 | expect(err).to.not.exist; 111 | expect(obj).to.exist; 112 | expect(obj.rows[0].token).to.exist; 113 | hashedToken1 = obj.rows[0].token; 114 | store.storeOrUpdate(token, user, 115 | 1000*60, 'http://' + chance.domain() + '/page.html', 116 | function() { 117 | pgClient.query('SELECT * FROM passwordless WHERE uid=$1',[user], function(err, obj) { 118 | expect(err).to.not.exist; 119 | expect(obj).to.exist; 120 | expect(obj.rows[0].token).to.exist; 121 | expect(obj.rows[0].token).to.not.equal(hashedToken1); 122 | done(); 123 | }); 124 | }); 125 | }) 126 | }); 127 | }); 128 | }); 129 | -------------------------------------------------------------------------------- /lib/postgrestore.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | var util = require('util'); 4 | var bcrypt = require('bcrypt'); 5 | var TokenStore = require('passwordless-tokenstore'); 6 | var pg = require('pg'); 7 | 8 | /** 9 | * Constructor of PostgreStore 10 | * @param {String} conString URI as defined by the PostgreSQL specification. Please 11 | * check the documentation for details: 12 | * https://github.com/brianc/node-postgres 13 | * @param {Object} [options] Combines both the options for the PostgreClient as well 14 | * as the options for PostgreStore. For the PostgreClient options please refer back to 15 | * the documentation. PostgreStore understands the following options: 16 | * (1) { pgstore: { difficulty: integer }} to change the bcrypt difficulty. Defaults to: 10 17 | * (1) { pgstore: { table: string }} to change the name of the table used. Defaults to: 'passwordless' 18 | * (2) { pgstore: { pgPoolSize: integer }} to change the pool size of PostgreSQL client. Defaults to: 10 19 | * @constructor 20 | */ 21 | function PostgreStore(conString, options) { 22 | if(!conString){ 23 | throw new Error('Connection String is missing.'); 24 | } 25 | 26 | this._options = options || {}; 27 | this._options.pgstore = this._options.pgstore || {}; 28 | this._difficulty = this._options.pgstore.difficulty || 10; 29 | this._table = this._options.pgstore.table || 'passwordless'; 30 | 31 | this._client = new pg.Pool({ 32 | connectionString: conString, 33 | max: this._options.pgstore.pgPoolSize || 10 34 | }); 35 | 36 | if(!isNumber(this._difficulty) || this._cost < 1) { 37 | throw new Error('bcrypt difficulty must be an integer >= 1'); 38 | } 39 | 40 | delete this._options.pgstore; 41 | 42 | var self = this; 43 | this._client.connect(function(err, client) { 44 | if(err) { 45 | throw new Error('Could not connect to Postgres database, with error : ' + err); 46 | } 47 | }); 48 | } 49 | 50 | util.inherits(PostgreStore, TokenStore); 51 | 52 | /** 53 | * Checks if the provided token / user id combination exists and is 54 | * valid in terms of time-to-live. If yes, the method provides the 55 | * the stored referrer URL if any. 56 | * @param {String} token to be authenticated 57 | * @param {String} uid Unique identifier of an user 58 | * @param {Function} callback in the format (error, valid, referrer). 59 | * In case of error, error will provide details, valid will be false and 60 | * referrer will be null. If the token / uid combination was not found 61 | * found, valid will be false and all else null. Otherwise, valid will 62 | * be true, referrer will (if provided when the token was stored) the 63 | * original URL requested and error will be null. 64 | */ 65 | PostgreStore.prototype.authenticate = function(token, uid, callback) { 66 | if(!token || !uid || !callback) { 67 | throw new Error('TokenStore:authenticate called with invalid parameters'); 68 | } 69 | 70 | var self = this; 71 | 72 | self._client.query('SELECT * FROM ' + self._table + ' WHERE uid=$1',[uid], function(err, result) { 73 | if(err) { 74 | return callback(err, false, null); 75 | } 76 | else if(!result || !result.rows || !result.rows.length || (result && result.rows && result.rows.length > 1)) { 77 | return callback(null, false, null); 78 | } else if(Date.now() > result.rows[0].ttl) { 79 | callback(null, false, null); 80 | } else { 81 | bcrypt.compare(token, result.rows[0].token, function(err, res) { 82 | if(err) { 83 | callback(err, false, null); 84 | } else if(res) { 85 | callback(null, true, result.rows[0].origin || ""); 86 | } else { 87 | callback(null, false, null); 88 | } 89 | }); 90 | } 91 | }); 92 | }; 93 | 94 | /** 95 | * Stores a new token / user ID combination or updates the token of an 96 | * existing user ID if that ID already exists. Hence, a user can only 97 | * have one valid token at a time 98 | * @param {String} token Token that allows authentication of _uid_ 99 | * @param {String} uid Unique identifier of an user 100 | * @param {Number} msToLive Validity of the token in ms 101 | * @param {String} originUrl Originally requested URL or null 102 | * @param {Function} callback Called with callback(error) in case of an 103 | * error or as callback() if the token was successully stored / updated 104 | */ 105 | PostgreStore.prototype.storeOrUpdate = function(token, uid, msToLive, originUrl, callback) { 106 | if(!token || !uid || !msToLive || !callback || !isNumber(msToLive)) { 107 | throw new Error('TokenStore:storeOrUpdate called with invalid parameters'); 108 | } 109 | 110 | var self = this; 111 | bcrypt.hash(token, self._difficulty, function(err, hashedToken) { 112 | if(err) { 113 | return callback(err); 114 | } 115 | 116 | self._client.query('INSERT INTO ' + self._table + '(uid,token, origin, ttl) VALUES($1, $2, $3, $4)',[uid, hashedToken, originUrl, (Date.now() + msToLive)], function(err) { 117 | if(err){ 118 | self._client.query('UPDATE ' + self._table + ' SET token=$1, origin=$2, ttl=$3 WHERE uid=$4',[hashedToken, originUrl, (Date.now() + msToLive), uid], function(err) { 119 | if(err){ 120 | callback(err); 121 | } 122 | else{ 123 | callback(); 124 | } 125 | //self._client.end(); 126 | }); 127 | } 128 | else{ 129 | callback(); 130 | } 131 | }); 132 | }); 133 | }; 134 | 135 | /** 136 | * Invalidates and removes a user and the linked token 137 | * @param {String} uid user ID 138 | * @param {Function} callback called with callback(error) in case of an 139 | * error or as callback() if the uid was successully invalidated 140 | */ 141 | PostgreStore.prototype.invalidateUser = function(uid, callback) { 142 | if(!uid || !callback) { 143 | throw new Error('TokenStore:invalidateUser called with invalid parameters'); 144 | } 145 | 146 | this._client.query('DELETE FROM ' + this._table + ' WHERE uid=$1',[uid], function(err) { 147 | if(err){ 148 | callback(err); 149 | } 150 | else{ 151 | callback(); 152 | } 153 | }); 154 | }; 155 | 156 | /** 157 | * Removes and invalidates all token 158 | * @param {Function} callback Called with callback(error) in case of an 159 | * error or as callback() otherwise 160 | */ 161 | PostgreStore.prototype.clear = function(callback) { 162 | if(!callback) { 163 | throw new Error('TokenStore:clear called with invalid parameters'); 164 | } 165 | this._client.query('DELETE FROM ' + this._table, function(err) { 166 | if(err){ 167 | callback(err); 168 | } 169 | else{ 170 | callback(); 171 | } 172 | }); 173 | }; 174 | 175 | /** 176 | * Number of tokens stored (no matter the validity) 177 | * @param {Function} callback Called with callback(null, count) in case 178 | * of success or with callback(error) in case of an error 179 | */ 180 | PostgreStore.prototype.length = function(callback) { 181 | if(!callback) { 182 | throw new Error('TokenStore:length called with invalid parameters'); 183 | } 184 | 185 | this._client.query('SELECT COUNT(uid) FROM ' + this._table, function(err, result) { 186 | if(err){ 187 | callback(err); 188 | } 189 | else{ 190 | callback(null, parseInt(result.rows[0].count)); 191 | } 192 | }); 193 | }; 194 | 195 | PostgreStore.prototype.disconnect = function() { 196 | self._client.done(); 197 | }; 198 | 199 | function isNumber(n) { 200 | return !isNaN(parseFloat(n)) && isFinite(n); 201 | } 202 | 203 | module.exports = PostgreStore; 204 | -------------------------------------------------------------------------------- /yarn.lock: -------------------------------------------------------------------------------- 1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. 2 | # yarn lockfile v1 3 | 4 | 5 | abbrev@1: 6 | version "1.1.0" 7 | resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.0.tgz#d0554c2256636e2f56e7c2e5ad183f859428d81f" 8 | 9 | ajv@^4.9.1: 10 | version "4.11.8" 11 | resolved "https://registry.yarnpkg.com/ajv/-/ajv-4.11.8.tgz#82ffb02b29e662ae53bdc20af15947706739c536" 12 | dependencies: 13 | co "^4.6.0" 14 | json-stable-stringify "^1.0.1" 15 | 16 | ansi-regex@^2.0.0: 17 | version "2.1.1" 18 | resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df" 19 | 20 | ansi-styles@^2.2.1: 21 | version "2.2.1" 22 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-2.2.1.tgz#b432dd3358b634cf75e1e4664368240533c1ddbe" 23 | 24 | ap@~0.2.0: 25 | version "0.2.0" 26 | resolved "https://registry.yarnpkg.com/ap/-/ap-0.2.0.tgz#ae0942600b29912f0d2b14ec60c45e8f330b6110" 27 | 28 | aproba@^1.0.3: 29 | version "1.1.2" 30 | resolved "https://registry.yarnpkg.com/aproba/-/aproba-1.1.2.tgz#45c6629094de4e96f693ef7eab74ae079c240fc1" 31 | 32 | are-we-there-yet@~1.1.2: 33 | version "1.1.4" 34 | resolved "https://registry.yarnpkg.com/are-we-there-yet/-/are-we-there-yet-1.1.4.tgz#bb5dca382bb94f05e15194373d16fd3ba1ca110d" 35 | dependencies: 36 | delegates "^1.0.0" 37 | readable-stream "^2.0.6" 38 | 39 | argparse@^1.0.2: 40 | version "1.0.9" 41 | resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.9.tgz#73d83bc263f86e97f8cc4f6bae1b0e90a7d22c86" 42 | dependencies: 43 | sprintf-js "~1.0.2" 44 | 45 | array-find-index@^1.0.1: 46 | version "1.0.2" 47 | resolved "https://registry.yarnpkg.com/array-find-index/-/array-find-index-1.0.2.tgz#df010aa1287e164bbda6f9723b0a96a1ec4187a1" 48 | 49 | asn1@~0.2.3: 50 | version "0.2.3" 51 | resolved "https://registry.yarnpkg.com/asn1/-/asn1-0.2.3.tgz#dac8787713c9966849fc8180777ebe9c1ddf3b86" 52 | 53 | assert-plus@1.0.0, assert-plus@^1.0.0: 54 | version "1.0.0" 55 | resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525" 56 | 57 | assert-plus@^0.2.0: 58 | version "0.2.0" 59 | resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-0.2.0.tgz#d74e1b87e7affc0db8aadb7021f3fe48101ab234" 60 | 61 | assertion-error@^1.0.1: 62 | version "1.0.2" 63 | resolved "https://registry.yarnpkg.com/assertion-error/-/assertion-error-1.0.2.tgz#13ca515d86206da0bac66e834dd397d87581094c" 64 | 65 | async@~1.5.2: 66 | version "1.5.2" 67 | resolved "https://registry.yarnpkg.com/async/-/async-1.5.2.tgz#ec6a61ae56480c0c3cb241c95618e20892f9672a" 68 | 69 | asynckit@^0.4.0: 70 | version "0.4.0" 71 | resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" 72 | 73 | aws-sign2@~0.6.0: 74 | version "0.6.0" 75 | resolved "https://registry.yarnpkg.com/aws-sign2/-/aws-sign2-0.6.0.tgz#14342dd38dbcc94d0e5b87d763cd63612c0e794f" 76 | 77 | aws4@^1.2.1: 78 | version "1.6.0" 79 | resolved "https://registry.yarnpkg.com/aws4/-/aws4-1.6.0.tgz#83ef5ca860b2b32e4a0deedee8c771b9db57471e" 80 | 81 | balanced-match@^1.0.0: 82 | version "1.0.0" 83 | resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz#89b4d199ab2bee49de164ea02b89ce462d71b767" 84 | 85 | bcrypt-pbkdf@^1.0.0: 86 | version "1.0.1" 87 | resolved "https://registry.yarnpkg.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.1.tgz#63bc5dcb61331b92bc05fd528953c33462a06f8d" 88 | dependencies: 89 | tweetnacl "^0.14.3" 90 | 91 | bcrypt@^1.0.2: 92 | version "1.0.2" 93 | resolved "https://registry.yarnpkg.com/bcrypt/-/bcrypt-1.0.2.tgz#d05fc5d223173e0e28ec381c0f00cc25ffaf2736" 94 | dependencies: 95 | bindings "1.2.1" 96 | nan "2.5.0" 97 | node-pre-gyp "0.6.32" 98 | 99 | bindings@1.2.1: 100 | version "1.2.1" 101 | resolved "https://registry.yarnpkg.com/bindings/-/bindings-1.2.1.tgz#14ad6113812d2d37d72e67b4cacb4bb726505f11" 102 | 103 | block-stream@*: 104 | version "0.0.9" 105 | resolved "https://registry.yarnpkg.com/block-stream/-/block-stream-0.0.9.tgz#13ebfe778a03205cfe03751481ebb4b3300c126a" 106 | dependencies: 107 | inherits "~2.0.0" 108 | 109 | boom@2.x.x: 110 | version "2.10.1" 111 | resolved "https://registry.yarnpkg.com/boom/-/boom-2.10.1.tgz#39c8918ceff5799f83f9492a848f625add0c766f" 112 | dependencies: 113 | hoek "2.x.x" 114 | 115 | brace-expansion@^1.1.7: 116 | version "1.1.8" 117 | resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.8.tgz#c07b211c7c952ec1f8efd51a77ef0d1d3990a292" 118 | dependencies: 119 | balanced-match "^1.0.0" 120 | concat-map "0.0.1" 121 | 122 | browser-stdout@1.3.0: 123 | version "1.3.0" 124 | resolved "https://registry.yarnpkg.com/browser-stdout/-/browser-stdout-1.3.0.tgz#f351d32969d32fa5d7a5567154263d928ae3bd1f" 125 | 126 | buffer-shims@^1.0.0: 127 | version "1.0.0" 128 | resolved "https://registry.yarnpkg.com/buffer-shims/-/buffer-shims-1.0.0.tgz#9978ce317388c649ad8793028c3477ef044a8b51" 129 | 130 | buffer-writer@1.0.1: 131 | version "1.0.1" 132 | resolved "https://registry.yarnpkg.com/buffer-writer/-/buffer-writer-1.0.1.tgz#22a936901e3029afcd7547eb4487ceb697a3bf08" 133 | 134 | builtin-modules@^1.0.0: 135 | version "1.1.1" 136 | resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-1.1.1.tgz#270f076c5a72c02f5b65a47df94c5fe3a278892f" 137 | 138 | camelcase-keys@^2.0.0: 139 | version "2.1.0" 140 | resolved "https://registry.yarnpkg.com/camelcase-keys/-/camelcase-keys-2.1.0.tgz#308beeaffdf28119051efa1d932213c91b8f92e7" 141 | dependencies: 142 | camelcase "^2.0.0" 143 | map-obj "^1.0.0" 144 | 145 | camelcase@^2.0.0: 146 | version "2.1.1" 147 | resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-2.1.1.tgz#7c1d16d679a1bbe59ca02cacecfb011e201f5a1f" 148 | 149 | caseless@~0.12.0: 150 | version "0.12.0" 151 | resolved "https://registry.yarnpkg.com/caseless/-/caseless-0.12.0.tgz#1b681c21ff84033c826543090689420d187151dc" 152 | 153 | chai@^3.5.0: 154 | version "3.5.0" 155 | resolved "https://registry.yarnpkg.com/chai/-/chai-3.5.0.tgz#4d02637b067fe958bdbfdd3a40ec56fef7373247" 156 | dependencies: 157 | assertion-error "^1.0.1" 158 | deep-eql "^0.1.3" 159 | type-detect "^1.0.0" 160 | 161 | chai@^4.0.2: 162 | version "4.0.2" 163 | resolved "https://registry.yarnpkg.com/chai/-/chai-4.0.2.tgz#2f7327c4de6f385dd7787999e2ab02697a32b83b" 164 | dependencies: 165 | assertion-error "^1.0.1" 166 | check-error "^1.0.1" 167 | deep-eql "^2.0.1" 168 | get-func-name "^2.0.0" 169 | pathval "^1.0.0" 170 | type-detect "^4.0.0" 171 | 172 | chalk@~1.1.1: 173 | version "1.1.3" 174 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98" 175 | dependencies: 176 | ansi-styles "^2.2.1" 177 | escape-string-regexp "^1.0.2" 178 | has-ansi "^2.0.0" 179 | strip-ansi "^3.0.0" 180 | supports-color "^2.0.0" 181 | 182 | chance@^1.0.1, chance@^1.0.10: 183 | version "1.0.10" 184 | resolved "https://registry.yarnpkg.com/chance/-/chance-1.0.10.tgz#03500b04ad94e778dd2891b09ec73a6ad87b1996" 185 | 186 | check-error@^1.0.1: 187 | version "1.0.2" 188 | resolved "https://registry.yarnpkg.com/check-error/-/check-error-1.0.2.tgz#574d312edd88bb5dd8912e9286dd6c0aed4aac82" 189 | 190 | co@^4.6.0: 191 | version "4.6.0" 192 | resolved "https://registry.yarnpkg.com/co/-/co-4.6.0.tgz#6ea6bdf3d853ae54ccb8e47bfa0bf3f9031fb184" 193 | 194 | code-point-at@^1.0.0: 195 | version "1.1.0" 196 | resolved "https://registry.yarnpkg.com/code-point-at/-/code-point-at-1.1.0.tgz#0d070b4d043a5bea33a2f1a40e2edb3d9a4ccf77" 197 | 198 | coffee-script@~1.10.0: 199 | version "1.10.0" 200 | resolved "https://registry.yarnpkg.com/coffee-script/-/coffee-script-1.10.0.tgz#12938bcf9be1948fa006f92e0c4c9e81705108c0" 201 | 202 | colors@~1.1.2: 203 | version "1.1.2" 204 | resolved "https://registry.yarnpkg.com/colors/-/colors-1.1.2.tgz#168a4701756b6a7f51a12ce0c97bfa28c084ed63" 205 | 206 | combined-stream@^1.0.5, combined-stream@~1.0.5: 207 | version "1.0.5" 208 | resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.5.tgz#938370a57b4a51dea2c77c15d5c5fdf895164009" 209 | dependencies: 210 | delayed-stream "~1.0.0" 211 | 212 | commander@0.6.1: 213 | version "0.6.1" 214 | resolved "https://registry.yarnpkg.com/commander/-/commander-0.6.1.tgz#fa68a14f6a945d54dbbe50d8cdb3320e9e3b1a06" 215 | 216 | commander@2.3.0: 217 | version "2.3.0" 218 | resolved "https://registry.yarnpkg.com/commander/-/commander-2.3.0.tgz#fd430e889832ec353b9acd1de217c11cb3eef873" 219 | 220 | commander@2.9.0: 221 | version "2.9.0" 222 | resolved "https://registry.yarnpkg.com/commander/-/commander-2.9.0.tgz#9c99094176e12240cb22d6c5146098400fe0f7d4" 223 | dependencies: 224 | graceful-readlink ">= 1.0.0" 225 | 226 | concat-map@0.0.1: 227 | version "0.0.1" 228 | resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" 229 | 230 | console-control-strings@^1.0.0, console-control-strings@~1.1.0: 231 | version "1.1.0" 232 | resolved "https://registry.yarnpkg.com/console-control-strings/-/console-control-strings-1.1.0.tgz#3d7cf4464db6446ea644bf4b39507f9851008e8e" 233 | 234 | core-util-is@~1.0.0: 235 | version "1.0.2" 236 | resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" 237 | 238 | cryptiles@2.x.x: 239 | version "2.0.5" 240 | resolved "https://registry.yarnpkg.com/cryptiles/-/cryptiles-2.0.5.tgz#3bdfecdc608147c1c67202fa291e7dca59eaa3b8" 241 | dependencies: 242 | boom "2.x.x" 243 | 244 | currently-unhandled@^0.4.1: 245 | version "0.4.1" 246 | resolved "https://registry.yarnpkg.com/currently-unhandled/-/currently-unhandled-0.4.1.tgz#988df33feab191ef799a61369dd76c17adf957ea" 247 | dependencies: 248 | array-find-index "^1.0.1" 249 | 250 | dashdash@^1.12.0: 251 | version "1.14.1" 252 | resolved "https://registry.yarnpkg.com/dashdash/-/dashdash-1.14.1.tgz#853cfa0f7cbe2fed5de20326b8dd581035f6e2f0" 253 | dependencies: 254 | assert-plus "^1.0.0" 255 | 256 | dateformat@~1.0.12: 257 | version "1.0.12" 258 | resolved "https://registry.yarnpkg.com/dateformat/-/dateformat-1.0.12.tgz#9f124b67594c937ff706932e4a642cca8dbbfee9" 259 | dependencies: 260 | get-stdin "^4.0.1" 261 | meow "^3.3.0" 262 | 263 | debug@2.2.0, debug@~2.2.0: 264 | version "2.2.0" 265 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.2.0.tgz#f87057e995b1a1f6ae6a4960664137bc56f039da" 266 | dependencies: 267 | ms "0.7.1" 268 | 269 | debug@2.6.0: 270 | version "2.6.0" 271 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.0.tgz#bc596bcabe7617f11d9fa15361eded5608b8499b" 272 | dependencies: 273 | ms "0.7.2" 274 | 275 | decamelize@^1.1.2: 276 | version "1.2.0" 277 | resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290" 278 | 279 | deep-eql@^0.1.3: 280 | version "0.1.3" 281 | resolved "https://registry.yarnpkg.com/deep-eql/-/deep-eql-0.1.3.tgz#ef558acab8de25206cd713906d74e56930eb69f2" 282 | dependencies: 283 | type-detect "0.1.1" 284 | 285 | deep-eql@^2.0.1: 286 | version "2.0.2" 287 | resolved "https://registry.yarnpkg.com/deep-eql/-/deep-eql-2.0.2.tgz#b1bac06e56f0a76777686d50c9feb75c2ed7679a" 288 | dependencies: 289 | type-detect "^3.0.0" 290 | 291 | deep-extend@~0.4.0: 292 | version "0.4.2" 293 | resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.4.2.tgz#48b699c27e334bf89f10892be432f6e4c7d34a7f" 294 | 295 | delayed-stream@~1.0.0: 296 | version "1.0.0" 297 | resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" 298 | 299 | delegates@^1.0.0: 300 | version "1.0.0" 301 | resolved "https://registry.yarnpkg.com/delegates/-/delegates-1.0.0.tgz#84c6e159b81904fdca59a0ef44cd870d31250f9a" 302 | 303 | diff@1.4.0: 304 | version "1.4.0" 305 | resolved "https://registry.yarnpkg.com/diff/-/diff-1.4.0.tgz#7f28d2eb9ee7b15a97efd89ce63dcfdaa3ccbabf" 306 | 307 | diff@3.2.0: 308 | version "3.2.0" 309 | resolved "https://registry.yarnpkg.com/diff/-/diff-3.2.0.tgz#c9ce393a4b7cbd0b058a725c93df299027868ff9" 310 | 311 | ecc-jsbn@~0.1.1: 312 | version "0.1.1" 313 | resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.1.tgz#0fc73a9ed5f0d53c38193398523ef7e543777505" 314 | dependencies: 315 | jsbn "~0.1.0" 316 | 317 | error-ex@^1.2.0: 318 | version "1.3.1" 319 | resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.1.tgz#f855a86ce61adc4e8621c3cda21e7a7612c3a8dc" 320 | dependencies: 321 | is-arrayish "^0.2.1" 322 | 323 | escape-string-regexp@1.0.2: 324 | version "1.0.2" 325 | resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.2.tgz#4dbc2fe674e71949caf3fb2695ce7f2dc1d9a8d1" 326 | 327 | escape-string-regexp@1.0.5, escape-string-regexp@^1.0.2: 328 | version "1.0.5" 329 | resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" 330 | 331 | esprima@^2.6.0: 332 | version "2.7.3" 333 | resolved "https://registry.yarnpkg.com/esprima/-/esprima-2.7.3.tgz#96e3b70d5779f6ad49cd032673d1c312767ba581" 334 | 335 | eventemitter2@~0.4.13: 336 | version "0.4.14" 337 | resolved "https://registry.yarnpkg.com/eventemitter2/-/eventemitter2-0.4.14.tgz#8f61b75cde012b2e9eb284d4545583b5643b61ab" 338 | 339 | exit@~0.1.1: 340 | version "0.1.2" 341 | resolved "https://registry.yarnpkg.com/exit/-/exit-0.1.2.tgz#0632638f8d877cc82107d30a0fff1a17cba1cd0c" 342 | 343 | extend@~3.0.0: 344 | version "3.0.1" 345 | resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.1.tgz#a755ea7bc1adfcc5a31ce7e762dbaadc5e636444" 346 | 347 | extsprintf@1.0.2: 348 | version "1.0.2" 349 | resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.0.2.tgz#e1080e0658e300b06294990cc70e1502235fd550" 350 | 351 | find-up@^1.0.0: 352 | version "1.1.2" 353 | resolved "https://registry.yarnpkg.com/find-up/-/find-up-1.1.2.tgz#6b2e9822b1a2ce0a60ab64d610eccad53cb24d0f" 354 | dependencies: 355 | path-exists "^2.0.0" 356 | pinkie-promise "^2.0.0" 357 | 358 | findup-sync@~0.3.0: 359 | version "0.3.0" 360 | resolved "https://registry.yarnpkg.com/findup-sync/-/findup-sync-0.3.0.tgz#37930aa5d816b777c03445e1966cc6790a4c0b16" 361 | dependencies: 362 | glob "~5.0.0" 363 | 364 | forever-agent@~0.6.1: 365 | version "0.6.1" 366 | resolved "https://registry.yarnpkg.com/forever-agent/-/forever-agent-0.6.1.tgz#fbc71f0c41adeb37f96c577ad1ed42d8fdacca91" 367 | 368 | form-data@~2.1.1: 369 | version "2.1.4" 370 | resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.1.4.tgz#33c183acf193276ecaa98143a69e94bfee1750d1" 371 | dependencies: 372 | asynckit "^0.4.0" 373 | combined-stream "^1.0.5" 374 | mime-types "^2.1.12" 375 | 376 | fs.realpath@^1.0.0: 377 | version "1.0.0" 378 | resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" 379 | 380 | fstream-ignore@~1.0.5: 381 | version "1.0.5" 382 | resolved "https://registry.yarnpkg.com/fstream-ignore/-/fstream-ignore-1.0.5.tgz#9c31dae34767018fe1d249b24dada67d092da105" 383 | dependencies: 384 | fstream "^1.0.0" 385 | inherits "2" 386 | minimatch "^3.0.0" 387 | 388 | fstream@^1.0.0, fstream@^1.0.2, fstream@~1.0.10: 389 | version "1.0.11" 390 | resolved "https://registry.yarnpkg.com/fstream/-/fstream-1.0.11.tgz#5c1fb1f117477114f0632a0eb4b71b3cb0fd3171" 391 | dependencies: 392 | graceful-fs "^4.1.2" 393 | inherits "~2.0.0" 394 | mkdirp ">=0.5 0" 395 | rimraf "2" 396 | 397 | gauge@~2.7.3: 398 | version "2.7.4" 399 | resolved "https://registry.yarnpkg.com/gauge/-/gauge-2.7.4.tgz#2c03405c7538c39d7eb37b317022e325fb018bf7" 400 | dependencies: 401 | aproba "^1.0.3" 402 | console-control-strings "^1.0.0" 403 | has-unicode "^2.0.0" 404 | object-assign "^4.1.0" 405 | signal-exit "^3.0.0" 406 | string-width "^1.0.1" 407 | strip-ansi "^3.0.1" 408 | wide-align "^1.1.0" 409 | 410 | generic-pool@2.4.3: 411 | version "2.4.3" 412 | resolved "https://registry.yarnpkg.com/generic-pool/-/generic-pool-2.4.3.tgz#780c36f69dfad05a5a045dd37be7adca11a4f6ff" 413 | 414 | get-func-name@^2.0.0: 415 | version "2.0.0" 416 | resolved "https://registry.yarnpkg.com/get-func-name/-/get-func-name-2.0.0.tgz#ead774abee72e20409433a066366023dd6887a41" 417 | 418 | get-stdin@^4.0.1: 419 | version "4.0.1" 420 | resolved "https://registry.yarnpkg.com/get-stdin/-/get-stdin-4.0.1.tgz#b968c6b0a04384324902e8bf1a5df32579a450fe" 421 | 422 | getobject@~0.1.0: 423 | version "0.1.0" 424 | resolved "https://registry.yarnpkg.com/getobject/-/getobject-0.1.0.tgz#047a449789fa160d018f5486ed91320b6ec7885c" 425 | 426 | getpass@^0.1.1: 427 | version "0.1.7" 428 | resolved "https://registry.yarnpkg.com/getpass/-/getpass-0.1.7.tgz#5eff8e3e684d569ae4cb2b1282604e8ba62149fa" 429 | dependencies: 430 | assert-plus "^1.0.0" 431 | 432 | glob@3.2.11: 433 | version "3.2.11" 434 | resolved "https://registry.yarnpkg.com/glob/-/glob-3.2.11.tgz#4a973f635b9190f715d10987d5c00fd2815ebe3d" 435 | dependencies: 436 | inherits "2" 437 | minimatch "0.3" 438 | 439 | glob@7.1.1, glob@^7.0.5: 440 | version "7.1.1" 441 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.1.tgz#805211df04faaf1c63a3600306cdf5ade50b2ec8" 442 | dependencies: 443 | fs.realpath "^1.0.0" 444 | inflight "^1.0.4" 445 | inherits "2" 446 | minimatch "^3.0.2" 447 | once "^1.3.0" 448 | path-is-absolute "^1.0.0" 449 | 450 | glob@~5.0.0: 451 | version "5.0.15" 452 | resolved "https://registry.yarnpkg.com/glob/-/glob-5.0.15.tgz#1bc936b9e02f4a603fcc222ecf7633d30b8b93b1" 453 | dependencies: 454 | inflight "^1.0.4" 455 | inherits "2" 456 | minimatch "2 || 3" 457 | once "^1.3.0" 458 | path-is-absolute "^1.0.0" 459 | 460 | glob@~7.0.0: 461 | version "7.0.6" 462 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.0.6.tgz#211bafaf49e525b8cd93260d14ab136152b3f57a" 463 | dependencies: 464 | fs.realpath "^1.0.0" 465 | inflight "^1.0.4" 466 | inherits "2" 467 | minimatch "^3.0.2" 468 | once "^1.3.0" 469 | path-is-absolute "^1.0.0" 470 | 471 | graceful-fs@^4.1.2: 472 | version "4.1.11" 473 | resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.1.11.tgz#0e8bdfe4d1ddb8854d64e04ea7c00e2a026e5658" 474 | 475 | "graceful-readlink@>= 1.0.0": 476 | version "1.0.1" 477 | resolved "https://registry.yarnpkg.com/graceful-readlink/-/graceful-readlink-1.0.1.tgz#4cafad76bc62f02fa039b2f94e9a3dd3a391a725" 478 | 479 | growl@1.9.2: 480 | version "1.9.2" 481 | resolved "https://registry.yarnpkg.com/growl/-/growl-1.9.2.tgz#0ea7743715db8d8de2c5ede1775e1b45ac85c02f" 482 | 483 | grunt-cli@~1.2.0: 484 | version "1.2.0" 485 | resolved "https://registry.yarnpkg.com/grunt-cli/-/grunt-cli-1.2.0.tgz#562b119ebb069ddb464ace2845501be97b35b6a8" 486 | dependencies: 487 | findup-sync "~0.3.0" 488 | grunt-known-options "~1.1.0" 489 | nopt "~3.0.6" 490 | resolve "~1.1.0" 491 | 492 | grunt-known-options@~1.1.0: 493 | version "1.1.0" 494 | resolved "https://registry.yarnpkg.com/grunt-known-options/-/grunt-known-options-1.1.0.tgz#a4274eeb32fa765da5a7a3b1712617ce3b144149" 495 | 496 | grunt-legacy-log-utils@~1.0.0: 497 | version "1.0.0" 498 | resolved "https://registry.yarnpkg.com/grunt-legacy-log-utils/-/grunt-legacy-log-utils-1.0.0.tgz#a7b8e2d0fb35b5a50f4af986fc112749ebc96f3d" 499 | dependencies: 500 | chalk "~1.1.1" 501 | lodash "~4.3.0" 502 | 503 | grunt-legacy-log@~1.0.0: 504 | version "1.0.0" 505 | resolved "https://registry.yarnpkg.com/grunt-legacy-log/-/grunt-legacy-log-1.0.0.tgz#fb86f1809847bc07dc47843f9ecd6cacb62df2d5" 506 | dependencies: 507 | colors "~1.1.2" 508 | grunt-legacy-log-utils "~1.0.0" 509 | hooker "~0.2.3" 510 | lodash "~3.10.1" 511 | underscore.string "~3.2.3" 512 | 513 | grunt-legacy-util@~1.0.0: 514 | version "1.0.0" 515 | resolved "https://registry.yarnpkg.com/grunt-legacy-util/-/grunt-legacy-util-1.0.0.tgz#386aa78dc6ed50986c2b18957265b1b48abb9b86" 516 | dependencies: 517 | async "~1.5.2" 518 | exit "~0.1.1" 519 | getobject "~0.1.0" 520 | hooker "~0.2.3" 521 | lodash "~4.3.0" 522 | underscore.string "~3.2.3" 523 | which "~1.2.1" 524 | 525 | grunt-mocha-test@^0.13.2: 526 | version "0.13.2" 527 | resolved "https://registry.yarnpkg.com/grunt-mocha-test/-/grunt-mocha-test-0.13.2.tgz#0f3abcc6ab543647b1effc5ab44ebd3702f0ab8c" 528 | dependencies: 529 | hooker "^0.2.3" 530 | mkdirp "^0.5.0" 531 | 532 | grunt@^1.0.1: 533 | version "1.0.1" 534 | resolved "https://registry.yarnpkg.com/grunt/-/grunt-1.0.1.tgz#e8778764e944b18f32bb0f10b9078475c9dfb56b" 535 | dependencies: 536 | coffee-script "~1.10.0" 537 | dateformat "~1.0.12" 538 | eventemitter2 "~0.4.13" 539 | exit "~0.1.1" 540 | findup-sync "~0.3.0" 541 | glob "~7.0.0" 542 | grunt-cli "~1.2.0" 543 | grunt-known-options "~1.1.0" 544 | grunt-legacy-log "~1.0.0" 545 | grunt-legacy-util "~1.0.0" 546 | iconv-lite "~0.4.13" 547 | js-yaml "~3.5.2" 548 | minimatch "~3.0.0" 549 | nopt "~3.0.6" 550 | path-is-absolute "~1.0.0" 551 | rimraf "~2.2.8" 552 | 553 | har-schema@^1.0.5: 554 | version "1.0.5" 555 | resolved "https://registry.yarnpkg.com/har-schema/-/har-schema-1.0.5.tgz#d263135f43307c02c602afc8fe95970c0151369e" 556 | 557 | har-validator@~4.2.1: 558 | version "4.2.1" 559 | resolved "https://registry.yarnpkg.com/har-validator/-/har-validator-4.2.1.tgz#33481d0f1bbff600dd203d75812a6a5fba002e2a" 560 | dependencies: 561 | ajv "^4.9.1" 562 | har-schema "^1.0.5" 563 | 564 | has-ansi@^2.0.0: 565 | version "2.0.0" 566 | resolved "https://registry.yarnpkg.com/has-ansi/-/has-ansi-2.0.0.tgz#34f5049ce1ecdf2b0649af3ef24e45ed35416d91" 567 | dependencies: 568 | ansi-regex "^2.0.0" 569 | 570 | has-flag@^1.0.0: 571 | version "1.0.0" 572 | resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-1.0.0.tgz#9d9e793165ce017a00f00418c43f942a7b1d11fa" 573 | 574 | has-unicode@^2.0.0: 575 | version "2.0.1" 576 | resolved "https://registry.yarnpkg.com/has-unicode/-/has-unicode-2.0.1.tgz#e0e6fe6a28cf51138855e086d1691e771de2a8b9" 577 | 578 | hawk@~3.1.3: 579 | version "3.1.3" 580 | resolved "https://registry.yarnpkg.com/hawk/-/hawk-3.1.3.tgz#078444bd7c1640b0fe540d2c9b73d59678e8e1c4" 581 | dependencies: 582 | boom "2.x.x" 583 | cryptiles "2.x.x" 584 | hoek "2.x.x" 585 | sntp "1.x.x" 586 | 587 | hoek@2.x.x: 588 | version "2.16.3" 589 | resolved "https://registry.yarnpkg.com/hoek/-/hoek-2.16.3.tgz#20bb7403d3cea398e91dc4710a8ff1b8274a25ed" 590 | 591 | hooker@^0.2.3, hooker@~0.2.3: 592 | version "0.2.3" 593 | resolved "https://registry.yarnpkg.com/hooker/-/hooker-0.2.3.tgz#b834f723cc4a242aa65963459df6d984c5d3d959" 594 | 595 | hosted-git-info@^2.1.4: 596 | version "2.5.0" 597 | resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.5.0.tgz#6d60e34b3abbc8313062c3b798ef8d901a07af3c" 598 | 599 | http-signature@~1.1.0: 600 | version "1.1.1" 601 | resolved "https://registry.yarnpkg.com/http-signature/-/http-signature-1.1.1.tgz#df72e267066cd0ac67fb76adf8e134a8fbcf91bf" 602 | dependencies: 603 | assert-plus "^0.2.0" 604 | jsprim "^1.2.2" 605 | sshpk "^1.7.0" 606 | 607 | iconv-lite@~0.4.13: 608 | version "0.4.18" 609 | resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.18.tgz#23d8656b16aae6742ac29732ea8f0336a4789cf2" 610 | 611 | indent-string@^2.1.0: 612 | version "2.1.0" 613 | resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-2.1.0.tgz#8e2d48348742121b4a8218b7a137e9a52049dc80" 614 | dependencies: 615 | repeating "^2.0.0" 616 | 617 | inflight@^1.0.4: 618 | version "1.0.6" 619 | resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" 620 | dependencies: 621 | once "^1.3.0" 622 | wrappy "1" 623 | 624 | inherits@2, inherits@~2.0.0, inherits@~2.0.1: 625 | version "2.0.3" 626 | resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de" 627 | 628 | ini@~1.3.0: 629 | version "1.3.4" 630 | resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.4.tgz#0537cb79daf59b59a1a517dff706c86ec039162e" 631 | 632 | is-arrayish@^0.2.1: 633 | version "0.2.1" 634 | resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d" 635 | 636 | is-builtin-module@^1.0.0: 637 | version "1.0.0" 638 | resolved "https://registry.yarnpkg.com/is-builtin-module/-/is-builtin-module-1.0.0.tgz#540572d34f7ac3119f8f76c30cbc1b1e037affbe" 639 | dependencies: 640 | builtin-modules "^1.0.0" 641 | 642 | is-finite@^1.0.0: 643 | version "1.0.2" 644 | resolved "https://registry.yarnpkg.com/is-finite/-/is-finite-1.0.2.tgz#cc6677695602be550ef11e8b4aa6305342b6d0aa" 645 | dependencies: 646 | number-is-nan "^1.0.0" 647 | 648 | is-fullwidth-code-point@^1.0.0: 649 | version "1.0.0" 650 | resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz#ef9e31386f031a7f0d643af82fde50c457ef00cb" 651 | dependencies: 652 | number-is-nan "^1.0.0" 653 | 654 | is-typedarray@~1.0.0: 655 | version "1.0.0" 656 | resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a" 657 | 658 | is-utf8@^0.2.0: 659 | version "0.2.1" 660 | resolved "https://registry.yarnpkg.com/is-utf8/-/is-utf8-0.2.1.tgz#4b0da1442104d1b336340e80797e865cf39f7d72" 661 | 662 | isarray@~1.0.0: 663 | version "1.0.0" 664 | resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" 665 | 666 | isexe@^2.0.0: 667 | version "2.0.0" 668 | resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" 669 | 670 | isstream@~0.1.2: 671 | version "0.1.2" 672 | resolved "https://registry.yarnpkg.com/isstream/-/isstream-0.1.2.tgz#47e63f7af55afa6f92e1500e690eb8b8529c099a" 673 | 674 | jade@0.26.3: 675 | version "0.26.3" 676 | resolved "https://registry.yarnpkg.com/jade/-/jade-0.26.3.tgz#8f10d7977d8d79f2f6ff862a81b0513ccb25686c" 677 | dependencies: 678 | commander "0.6.1" 679 | mkdirp "0.3.0" 680 | 681 | js-yaml@~3.5.2: 682 | version "3.5.5" 683 | resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.5.5.tgz#0377c38017cabc7322b0d1fbcd25a491641f2fbe" 684 | dependencies: 685 | argparse "^1.0.2" 686 | esprima "^2.6.0" 687 | 688 | jsbn@~0.1.0: 689 | version "0.1.1" 690 | resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-0.1.1.tgz#a5e654c2e5a2deb5f201d96cefbca80c0ef2f513" 691 | 692 | json-schema@0.2.3: 693 | version "0.2.3" 694 | resolved "https://registry.yarnpkg.com/json-schema/-/json-schema-0.2.3.tgz#b480c892e59a2f05954ce727bd3f2a4e882f9e13" 695 | 696 | json-stable-stringify@^1.0.1: 697 | version "1.0.1" 698 | resolved "https://registry.yarnpkg.com/json-stable-stringify/-/json-stable-stringify-1.0.1.tgz#9a759d39c5f2ff503fd5300646ed445f88c4f9af" 699 | dependencies: 700 | jsonify "~0.0.0" 701 | 702 | json-stringify-safe@~5.0.1: 703 | version "5.0.1" 704 | resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" 705 | 706 | json3@3.3.2: 707 | version "3.3.2" 708 | resolved "https://registry.yarnpkg.com/json3/-/json3-3.3.2.tgz#3c0434743df93e2f5c42aee7b19bcb483575f4e1" 709 | 710 | jsonify@~0.0.0: 711 | version "0.0.0" 712 | resolved "https://registry.yarnpkg.com/jsonify/-/jsonify-0.0.0.tgz#2c74b6ee41d93ca51b7b5aaee8f503631d252a73" 713 | 714 | jsprim@^1.2.2: 715 | version "1.4.0" 716 | resolved "https://registry.yarnpkg.com/jsprim/-/jsprim-1.4.0.tgz#a3b87e40298d8c380552d8cc7628a0bb95a22918" 717 | dependencies: 718 | assert-plus "1.0.0" 719 | extsprintf "1.0.2" 720 | json-schema "0.2.3" 721 | verror "1.3.6" 722 | 723 | load-json-file@^1.0.0: 724 | version "1.1.0" 725 | resolved "https://registry.yarnpkg.com/load-json-file/-/load-json-file-1.1.0.tgz#956905708d58b4bab4c2261b04f59f31c99374c0" 726 | dependencies: 727 | graceful-fs "^4.1.2" 728 | parse-json "^2.2.0" 729 | pify "^2.0.0" 730 | pinkie-promise "^2.0.0" 731 | strip-bom "^2.0.0" 732 | 733 | lodash._baseassign@^3.0.0: 734 | version "3.2.0" 735 | resolved "https://registry.yarnpkg.com/lodash._baseassign/-/lodash._baseassign-3.2.0.tgz#8c38a099500f215ad09e59f1722fd0c52bfe0a4e" 736 | dependencies: 737 | lodash._basecopy "^3.0.0" 738 | lodash.keys "^3.0.0" 739 | 740 | lodash._basecopy@^3.0.0: 741 | version "3.0.1" 742 | resolved "https://registry.yarnpkg.com/lodash._basecopy/-/lodash._basecopy-3.0.1.tgz#8da0e6a876cf344c0ad8a54882111dd3c5c7ca36" 743 | 744 | lodash._basecreate@^3.0.0: 745 | version "3.0.3" 746 | resolved "https://registry.yarnpkg.com/lodash._basecreate/-/lodash._basecreate-3.0.3.tgz#1bc661614daa7fc311b7d03bf16806a0213cf821" 747 | 748 | lodash._getnative@^3.0.0: 749 | version "3.9.1" 750 | resolved "https://registry.yarnpkg.com/lodash._getnative/-/lodash._getnative-3.9.1.tgz#570bc7dede46d61cdcde687d65d3eecbaa3aaff5" 751 | 752 | lodash._isiterateecall@^3.0.0: 753 | version "3.0.9" 754 | resolved "https://registry.yarnpkg.com/lodash._isiterateecall/-/lodash._isiterateecall-3.0.9.tgz#5203ad7ba425fae842460e696db9cf3e6aac057c" 755 | 756 | lodash.create@3.1.1: 757 | version "3.1.1" 758 | resolved "https://registry.yarnpkg.com/lodash.create/-/lodash.create-3.1.1.tgz#d7f2849f0dbda7e04682bb8cd72ab022461debe7" 759 | dependencies: 760 | lodash._baseassign "^3.0.0" 761 | lodash._basecreate "^3.0.0" 762 | lodash._isiterateecall "^3.0.0" 763 | 764 | lodash.isarguments@^3.0.0: 765 | version "3.1.0" 766 | resolved "https://registry.yarnpkg.com/lodash.isarguments/-/lodash.isarguments-3.1.0.tgz#2f573d85c6a24289ff00663b491c1d338ff3458a" 767 | 768 | lodash.isarray@^3.0.0: 769 | version "3.0.4" 770 | resolved "https://registry.yarnpkg.com/lodash.isarray/-/lodash.isarray-3.0.4.tgz#79e4eb88c36a8122af86f844aa9bcd851b5fbb55" 771 | 772 | lodash.keys@^3.0.0: 773 | version "3.1.2" 774 | resolved "https://registry.yarnpkg.com/lodash.keys/-/lodash.keys-3.1.2.tgz#4dbc0472b156be50a0b286855d1bd0b0c656098a" 775 | dependencies: 776 | lodash._getnative "^3.0.0" 777 | lodash.isarguments "^3.0.0" 778 | lodash.isarray "^3.0.0" 779 | 780 | lodash@~3.10.1: 781 | version "3.10.1" 782 | resolved "https://registry.yarnpkg.com/lodash/-/lodash-3.10.1.tgz#5bf45e8e49ba4189e17d482789dfd15bd140b7b6" 783 | 784 | lodash@~4.3.0: 785 | version "4.3.0" 786 | resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.3.0.tgz#efd9c4a6ec53f3b05412429915c3e4824e4d25a4" 787 | 788 | loud-rejection@^1.0.0: 789 | version "1.6.0" 790 | resolved "https://registry.yarnpkg.com/loud-rejection/-/loud-rejection-1.6.0.tgz#5b46f80147edee578870f086d04821cf998e551f" 791 | dependencies: 792 | currently-unhandled "^0.4.1" 793 | signal-exit "^3.0.0" 794 | 795 | lru-cache@2: 796 | version "2.7.3" 797 | resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-2.7.3.tgz#6d4524e8b955f95d4f5b58851ce21dd72fb4e952" 798 | 799 | map-obj@^1.0.0, map-obj@^1.0.1: 800 | version "1.0.1" 801 | resolved "https://registry.yarnpkg.com/map-obj/-/map-obj-1.0.1.tgz#d933ceb9205d82bdcf4886f6742bdc2b4dea146d" 802 | 803 | meow@^3.3.0: 804 | version "3.7.0" 805 | resolved "https://registry.yarnpkg.com/meow/-/meow-3.7.0.tgz#72cb668b425228290abbfa856892587308a801fb" 806 | dependencies: 807 | camelcase-keys "^2.0.0" 808 | decamelize "^1.1.2" 809 | loud-rejection "^1.0.0" 810 | map-obj "^1.0.1" 811 | minimist "^1.1.3" 812 | normalize-package-data "^2.3.4" 813 | object-assign "^4.0.1" 814 | read-pkg-up "^1.0.1" 815 | redent "^1.0.0" 816 | trim-newlines "^1.0.0" 817 | 818 | mime-db@~1.27.0: 819 | version "1.27.0" 820 | resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.27.0.tgz#820f572296bbd20ec25ed55e5b5de869e5436eb1" 821 | 822 | mime-types@^2.1.12, mime-types@~2.1.7: 823 | version "2.1.15" 824 | resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.15.tgz#a4ebf5064094569237b8cf70046776d09fc92aed" 825 | dependencies: 826 | mime-db "~1.27.0" 827 | 828 | minimatch@0.3: 829 | version "0.3.0" 830 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-0.3.0.tgz#275d8edaac4f1bb3326472089e7949c8394699dd" 831 | dependencies: 832 | lru-cache "2" 833 | sigmund "~1.0.0" 834 | 835 | "minimatch@2 || 3", minimatch@^3.0.0, minimatch@^3.0.2, minimatch@~3.0.0: 836 | version "3.0.4" 837 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083" 838 | dependencies: 839 | brace-expansion "^1.1.7" 840 | 841 | minimist@0.0.8: 842 | version "0.0.8" 843 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-0.0.8.tgz#857fcabfc3397d2625b8228262e86aa7a011b05d" 844 | 845 | minimist@^1.1.3, minimist@^1.2.0: 846 | version "1.2.0" 847 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.0.tgz#a35008b20f41383eec1fb914f4cd5df79a264284" 848 | 849 | mkdirp@0.3.0: 850 | version "0.3.0" 851 | resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.3.0.tgz#1bbf5ab1ba827af23575143490426455f481fe1e" 852 | 853 | mkdirp@0.5.1, "mkdirp@>=0.5 0", mkdirp@^0.5.0, mkdirp@~0.5.1: 854 | version "0.5.1" 855 | resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.1.tgz#30057438eac6cf7f8c4767f38648d6697d75c903" 856 | dependencies: 857 | minimist "0.0.8" 858 | 859 | mocha@^2.4.5: 860 | version "2.5.3" 861 | resolved "https://registry.yarnpkg.com/mocha/-/mocha-2.5.3.tgz#161be5bdeb496771eb9b35745050b622b5aefc58" 862 | dependencies: 863 | commander "2.3.0" 864 | debug "2.2.0" 865 | diff "1.4.0" 866 | escape-string-regexp "1.0.2" 867 | glob "3.2.11" 868 | growl "1.9.2" 869 | jade "0.26.3" 870 | mkdirp "0.5.1" 871 | supports-color "1.2.0" 872 | to-iso-string "0.0.2" 873 | 874 | mocha@^3.4.2: 875 | version "3.4.2" 876 | resolved "https://registry.yarnpkg.com/mocha/-/mocha-3.4.2.tgz#d0ef4d332126dbf18d0d640c9b382dd48be97594" 877 | dependencies: 878 | browser-stdout "1.3.0" 879 | commander "2.9.0" 880 | debug "2.6.0" 881 | diff "3.2.0" 882 | escape-string-regexp "1.0.5" 883 | glob "7.1.1" 884 | growl "1.9.2" 885 | json3 "3.3.2" 886 | lodash.create "3.1.1" 887 | mkdirp "0.5.1" 888 | supports-color "3.1.2" 889 | 890 | ms@0.7.1: 891 | version "0.7.1" 892 | resolved "https://registry.yarnpkg.com/ms/-/ms-0.7.1.tgz#9cd13c03adbff25b65effde7ce864ee952017098" 893 | 894 | ms@0.7.2: 895 | version "0.7.2" 896 | resolved "https://registry.yarnpkg.com/ms/-/ms-0.7.2.tgz#ae25cf2512b3885a1d95d7f037868d8431124765" 897 | 898 | nan@2.5.0: 899 | version "2.5.0" 900 | resolved "https://registry.yarnpkg.com/nan/-/nan-2.5.0.tgz#aa8f1e34531d807e9e27755b234b4a6ec0c152a8" 901 | 902 | node-pre-gyp@0.6.32: 903 | version "0.6.32" 904 | resolved "https://registry.yarnpkg.com/node-pre-gyp/-/node-pre-gyp-0.6.32.tgz#fc452b376e7319b3d255f5f34853ef6fd8fe1fd5" 905 | dependencies: 906 | mkdirp "~0.5.1" 907 | nopt "~3.0.6" 908 | npmlog "^4.0.1" 909 | rc "~1.1.6" 910 | request "^2.79.0" 911 | rimraf "~2.5.4" 912 | semver "~5.3.0" 913 | tar "~2.2.1" 914 | tar-pack "~3.3.0" 915 | 916 | node-uuid@^1.4.1, node-uuid@^1.4.7: 917 | version "1.4.8" 918 | resolved "https://registry.yarnpkg.com/node-uuid/-/node-uuid-1.4.8.tgz#b040eb0923968afabf8d32fb1f17f1167fdab907" 919 | 920 | nopt@~3.0.6: 921 | version "3.0.6" 922 | resolved "https://registry.yarnpkg.com/nopt/-/nopt-3.0.6.tgz#c6465dbf08abcd4db359317f79ac68a646b28ff9" 923 | dependencies: 924 | abbrev "1" 925 | 926 | normalize-package-data@^2.3.2, normalize-package-data@^2.3.4: 927 | version "2.4.0" 928 | resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.4.0.tgz#12f95a307d58352075a04907b84ac8be98ac012f" 929 | dependencies: 930 | hosted-git-info "^2.1.4" 931 | is-builtin-module "^1.0.0" 932 | semver "2 || 3 || 4 || 5" 933 | validate-npm-package-license "^3.0.1" 934 | 935 | npmlog@^4.0.1: 936 | version "4.1.2" 937 | resolved "https://registry.yarnpkg.com/npmlog/-/npmlog-4.1.2.tgz#08a7f2a8bf734604779a9efa4ad5cc717abb954b" 938 | dependencies: 939 | are-we-there-yet "~1.1.2" 940 | console-control-strings "~1.1.0" 941 | gauge "~2.7.3" 942 | set-blocking "~2.0.0" 943 | 944 | number-is-nan@^1.0.0: 945 | version "1.0.1" 946 | resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d" 947 | 948 | oauth-sign@~0.8.1: 949 | version "0.8.2" 950 | resolved "https://registry.yarnpkg.com/oauth-sign/-/oauth-sign-0.8.2.tgz#46a6ab7f0aead8deae9ec0565780b7d4efeb9d43" 951 | 952 | object-assign@4.1.0, object-assign@^4.0.1, object-assign@^4.1.0: 953 | version "4.1.0" 954 | resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.0.tgz#7a3b3d0e98063d43f4c03f2e8ae6cd51a86883a0" 955 | 956 | once@^1.3.0, once@~1.3.3: 957 | version "1.3.3" 958 | resolved "https://registry.yarnpkg.com/once/-/once-1.3.3.tgz#b2e261557ce4c314ec8304f3fa82663e4297ca20" 959 | dependencies: 960 | wrappy "1" 961 | 962 | packet-reader@0.3.1: 963 | version "0.3.1" 964 | resolved "https://registry.yarnpkg.com/packet-reader/-/packet-reader-0.3.1.tgz#cd62e60af8d7fea8a705ec4ff990871c46871f27" 965 | 966 | parse-json@^2.2.0: 967 | version "2.2.0" 968 | resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-2.2.0.tgz#f480f40434ef80741f8469099f8dea18f55a4dc9" 969 | dependencies: 970 | error-ex "^1.2.0" 971 | 972 | passwordless-tokenstore-test@^0.1.4: 973 | version "0.1.4" 974 | resolved "https://registry.yarnpkg.com/passwordless-tokenstore-test/-/passwordless-tokenstore-test-0.1.4.tgz#6ea0b097414599cbf50406247de414c26d44377a" 975 | dependencies: 976 | chai "^3.5.0" 977 | chance "^1.0.1" 978 | mocha "^2.4.5" 979 | node-uuid "^1.4.7" 980 | 981 | passwordless-tokenstore@0.0.10: 982 | version "0.0.10" 983 | resolved "https://registry.yarnpkg.com/passwordless-tokenstore/-/passwordless-tokenstore-0.0.10.tgz#829c4c0b792ac2c55de54c05c118d79e946b9b2e" 984 | 985 | path-exists@^2.0.0: 986 | version "2.1.0" 987 | resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-2.1.0.tgz#0feb6c64f0fc518d9a754dd5efb62c7022761f4b" 988 | dependencies: 989 | pinkie-promise "^2.0.0" 990 | 991 | path-is-absolute@^1.0.0, path-is-absolute@~1.0.0: 992 | version "1.0.1" 993 | resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" 994 | 995 | path-type@^1.0.0: 996 | version "1.1.0" 997 | resolved "https://registry.yarnpkg.com/path-type/-/path-type-1.1.0.tgz#59c44f7ee491da704da415da5a4070ba4f8fe441" 998 | dependencies: 999 | graceful-fs "^4.1.2" 1000 | pify "^2.0.0" 1001 | pinkie-promise "^2.0.0" 1002 | 1003 | pathval@^1.0.0: 1004 | version "1.1.0" 1005 | resolved "https://registry.yarnpkg.com/pathval/-/pathval-1.1.0.tgz#b942e6d4bde653005ef6b71361def8727d0645e0" 1006 | 1007 | performance-now@^0.2.0: 1008 | version "0.2.0" 1009 | resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-0.2.0.tgz#33ef30c5c77d4ea21c5a53869d91b56d8f2555e5" 1010 | 1011 | pg-connection-string@0.1.3: 1012 | version "0.1.3" 1013 | resolved "https://registry.yarnpkg.com/pg-connection-string/-/pg-connection-string-0.1.3.tgz#da1847b20940e42ee1492beaf65d49d91b245df7" 1014 | 1015 | pg-pool@1.*: 1016 | version "1.8.0" 1017 | resolved "https://registry.yarnpkg.com/pg-pool/-/pg-pool-1.8.0.tgz#f7ec73824c37a03f076f51bfdf70e340147c4f37" 1018 | dependencies: 1019 | generic-pool "2.4.3" 1020 | object-assign "4.1.0" 1021 | 1022 | pg-types@1.*: 1023 | version "1.12.0" 1024 | resolved "https://registry.yarnpkg.com/pg-types/-/pg-types-1.12.0.tgz#8ad3b7b897e3fd463e62de241ad5fc640b4a66f0" 1025 | dependencies: 1026 | ap "~0.2.0" 1027 | postgres-array "~1.0.0" 1028 | postgres-bytea "~1.0.0" 1029 | postgres-date "~1.0.0" 1030 | postgres-interval "^1.1.0" 1031 | 1032 | pg@^6.4.0: 1033 | version "6.4.0" 1034 | resolved "https://registry.yarnpkg.com/pg/-/pg-6.4.0.tgz#cb76ba2e7c2eab89fc64bf7a9fe648ced72436dc" 1035 | dependencies: 1036 | buffer-writer "1.0.1" 1037 | packet-reader "0.3.1" 1038 | pg-connection-string "0.1.3" 1039 | pg-pool "1.*" 1040 | pg-types "1.*" 1041 | pgpass "1.x" 1042 | semver "4.3.2" 1043 | 1044 | pgpass@1.x: 1045 | version "1.0.2" 1046 | resolved "https://registry.yarnpkg.com/pgpass/-/pgpass-1.0.2.tgz#2a7bb41b6065b67907e91da1b07c1847c877b306" 1047 | dependencies: 1048 | split "^1.0.0" 1049 | 1050 | pify@^2.0.0: 1051 | version "2.3.0" 1052 | resolved "https://registry.yarnpkg.com/pify/-/pify-2.3.0.tgz#ed141a6ac043a849ea588498e7dca8b15330e90c" 1053 | 1054 | pinkie-promise@^2.0.0: 1055 | version "2.0.1" 1056 | resolved "https://registry.yarnpkg.com/pinkie-promise/-/pinkie-promise-2.0.1.tgz#2135d6dfa7a358c069ac9b178776288228450ffa" 1057 | dependencies: 1058 | pinkie "^2.0.0" 1059 | 1060 | pinkie@^2.0.0: 1061 | version "2.0.4" 1062 | resolved "https://registry.yarnpkg.com/pinkie/-/pinkie-2.0.4.tgz#72556b80cfa0d48a974e80e77248e80ed4f7f870" 1063 | 1064 | postgres-array@~1.0.0: 1065 | version "1.0.2" 1066 | resolved "https://registry.yarnpkg.com/postgres-array/-/postgres-array-1.0.2.tgz#8e0b32eb03bf77a5c0a7851e0441c169a256a238" 1067 | 1068 | postgres-bytea@~1.0.0: 1069 | version "1.0.0" 1070 | resolved "https://registry.yarnpkg.com/postgres-bytea/-/postgres-bytea-1.0.0.tgz#027b533c0aa890e26d172d47cf9ccecc521acd35" 1071 | 1072 | postgres-date@~1.0.0: 1073 | version "1.0.3" 1074 | resolved "https://registry.yarnpkg.com/postgres-date/-/postgres-date-1.0.3.tgz#e2d89702efdb258ff9d9cee0fe91bd06975257a8" 1075 | 1076 | postgres-interval@^1.1.0: 1077 | version "1.1.0" 1078 | resolved "https://registry.yarnpkg.com/postgres-interval/-/postgres-interval-1.1.0.tgz#1031e7bac34564132862adc9eb6c6d2f3aa75bb4" 1079 | dependencies: 1080 | xtend "^4.0.0" 1081 | 1082 | process-nextick-args@~1.0.6: 1083 | version "1.0.7" 1084 | resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-1.0.7.tgz#150e20b756590ad3f91093f25a4f2ad8bff30ba3" 1085 | 1086 | punycode@^1.4.1: 1087 | version "1.4.1" 1088 | resolved "https://registry.yarnpkg.com/punycode/-/punycode-1.4.1.tgz#c0d5a63b2718800ad8e1eb0fa5269c84dd41845e" 1089 | 1090 | qs@~6.4.0: 1091 | version "6.4.0" 1092 | resolved "https://registry.yarnpkg.com/qs/-/qs-6.4.0.tgz#13e26d28ad6b0ffaa91312cd3bf708ed351e7233" 1093 | 1094 | rc@~1.1.6: 1095 | version "1.1.7" 1096 | resolved "https://registry.yarnpkg.com/rc/-/rc-1.1.7.tgz#c5ea564bb07aff9fd3a5b32e906c1d3a65940fea" 1097 | dependencies: 1098 | deep-extend "~0.4.0" 1099 | ini "~1.3.0" 1100 | minimist "^1.2.0" 1101 | strip-json-comments "~2.0.1" 1102 | 1103 | read-pkg-up@^1.0.1: 1104 | version "1.0.1" 1105 | resolved "https://registry.yarnpkg.com/read-pkg-up/-/read-pkg-up-1.0.1.tgz#9d63c13276c065918d57f002a57f40a1b643fb02" 1106 | dependencies: 1107 | find-up "^1.0.0" 1108 | read-pkg "^1.0.0" 1109 | 1110 | read-pkg@^1.0.0: 1111 | version "1.1.0" 1112 | resolved "https://registry.yarnpkg.com/read-pkg/-/read-pkg-1.1.0.tgz#f5ffaa5ecd29cb31c0474bca7d756b6bb29e3f28" 1113 | dependencies: 1114 | load-json-file "^1.0.0" 1115 | normalize-package-data "^2.3.2" 1116 | path-type "^1.0.0" 1117 | 1118 | readable-stream@^2.0.6, readable-stream@~2.1.4: 1119 | version "2.1.5" 1120 | resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.1.5.tgz#66fa8b720e1438b364681f2ad1a63c618448c9d0" 1121 | dependencies: 1122 | buffer-shims "^1.0.0" 1123 | core-util-is "~1.0.0" 1124 | inherits "~2.0.1" 1125 | isarray "~1.0.0" 1126 | process-nextick-args "~1.0.6" 1127 | string_decoder "~0.10.x" 1128 | util-deprecate "~1.0.1" 1129 | 1130 | redent@^1.0.0: 1131 | version "1.0.0" 1132 | resolved "https://registry.yarnpkg.com/redent/-/redent-1.0.0.tgz#cf916ab1fd5f1f16dfb20822dd6ec7f730c2afde" 1133 | dependencies: 1134 | indent-string "^2.1.0" 1135 | strip-indent "^1.0.1" 1136 | 1137 | repeating@^2.0.0: 1138 | version "2.0.1" 1139 | resolved "https://registry.yarnpkg.com/repeating/-/repeating-2.0.1.tgz#5214c53a926d3552707527fbab415dbc08d06dda" 1140 | dependencies: 1141 | is-finite "^1.0.0" 1142 | 1143 | request@^2.79.0: 1144 | version "2.81.0" 1145 | resolved "https://registry.yarnpkg.com/request/-/request-2.81.0.tgz#c6928946a0e06c5f8d6f8a9333469ffda46298a0" 1146 | dependencies: 1147 | aws-sign2 "~0.6.0" 1148 | aws4 "^1.2.1" 1149 | caseless "~0.12.0" 1150 | combined-stream "~1.0.5" 1151 | extend "~3.0.0" 1152 | forever-agent "~0.6.1" 1153 | form-data "~2.1.1" 1154 | har-validator "~4.2.1" 1155 | hawk "~3.1.3" 1156 | http-signature "~1.1.0" 1157 | is-typedarray "~1.0.0" 1158 | isstream "~0.1.2" 1159 | json-stringify-safe "~5.0.1" 1160 | mime-types "~2.1.7" 1161 | oauth-sign "~0.8.1" 1162 | performance-now "^0.2.0" 1163 | qs "~6.4.0" 1164 | safe-buffer "^5.0.1" 1165 | stringstream "~0.0.4" 1166 | tough-cookie "~2.3.0" 1167 | tunnel-agent "^0.6.0" 1168 | uuid "^3.0.0" 1169 | 1170 | resolve@~1.1.0: 1171 | version "1.1.7" 1172 | resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.1.7.tgz#203114d82ad2c5ed9e8e0411b3932875e889e97b" 1173 | 1174 | rimraf@2, rimraf@~2.5.1, rimraf@~2.5.4: 1175 | version "2.5.4" 1176 | resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-2.5.4.tgz#96800093cbf1a0c86bd95b4625467535c29dfa04" 1177 | dependencies: 1178 | glob "^7.0.5" 1179 | 1180 | rimraf@~2.2.8: 1181 | version "2.2.8" 1182 | resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-2.2.8.tgz#e439be2aaee327321952730f99a8929e4fc50582" 1183 | 1184 | safe-buffer@^5.0.1: 1185 | version "5.1.1" 1186 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.1.tgz#893312af69b2123def71f57889001671eeb2c853" 1187 | 1188 | "semver@2 || 3 || 4 || 5", semver@~5.3.0: 1189 | version "5.3.0" 1190 | resolved "https://registry.yarnpkg.com/semver/-/semver-5.3.0.tgz#9b2ce5d3de02d17c6012ad326aa6b4d0cf54f94f" 1191 | 1192 | semver@4.3.2: 1193 | version "4.3.2" 1194 | resolved "https://registry.yarnpkg.com/semver/-/semver-4.3.2.tgz#c7a07158a80bedd052355b770d82d6640f803be7" 1195 | 1196 | set-blocking@~2.0.0: 1197 | version "2.0.0" 1198 | resolved "https://registry.yarnpkg.com/set-blocking/-/set-blocking-2.0.0.tgz#045f9782d011ae9a6803ddd382b24392b3d890f7" 1199 | 1200 | sigmund@~1.0.0: 1201 | version "1.0.1" 1202 | resolved "https://registry.yarnpkg.com/sigmund/-/sigmund-1.0.1.tgz#3ff21f198cad2175f9f3b781853fd94d0d19b590" 1203 | 1204 | signal-exit@^3.0.0: 1205 | version "3.0.2" 1206 | resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.2.tgz#b5fdc08f1287ea1178628e415e25132b73646c6d" 1207 | 1208 | sntp@1.x.x: 1209 | version "1.0.9" 1210 | resolved "https://registry.yarnpkg.com/sntp/-/sntp-1.0.9.tgz#6541184cc90aeea6c6e7b35e2659082443c66198" 1211 | dependencies: 1212 | hoek "2.x.x" 1213 | 1214 | spdx-correct@~1.0.0: 1215 | version "1.0.2" 1216 | resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-1.0.2.tgz#4b3073d933ff51f3912f03ac5519498a4150db40" 1217 | dependencies: 1218 | spdx-license-ids "^1.0.2" 1219 | 1220 | spdx-expression-parse@~1.0.0: 1221 | version "1.0.4" 1222 | resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-1.0.4.tgz#9bdf2f20e1f40ed447fbe273266191fced51626c" 1223 | 1224 | spdx-license-ids@^1.0.2: 1225 | version "1.2.2" 1226 | resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-1.2.2.tgz#c9df7a3424594ade6bd11900d596696dc06bac57" 1227 | 1228 | split@^1.0.0: 1229 | version "1.0.0" 1230 | resolved "https://registry.yarnpkg.com/split/-/split-1.0.0.tgz#c4395ce683abcd254bc28fe1dabb6e5c27dcffae" 1231 | dependencies: 1232 | through "2" 1233 | 1234 | sprintf-js@~1.0.2: 1235 | version "1.0.3" 1236 | resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c" 1237 | 1238 | sshpk@^1.7.0: 1239 | version "1.13.1" 1240 | resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.13.1.tgz#512df6da6287144316dc4c18fe1cf1d940739be3" 1241 | dependencies: 1242 | asn1 "~0.2.3" 1243 | assert-plus "^1.0.0" 1244 | dashdash "^1.12.0" 1245 | getpass "^0.1.1" 1246 | optionalDependencies: 1247 | bcrypt-pbkdf "^1.0.0" 1248 | ecc-jsbn "~0.1.1" 1249 | jsbn "~0.1.0" 1250 | tweetnacl "~0.14.0" 1251 | 1252 | string-width@^1.0.1, string-width@^1.0.2: 1253 | version "1.0.2" 1254 | resolved "https://registry.yarnpkg.com/string-width/-/string-width-1.0.2.tgz#118bdf5b8cdc51a2a7e70d211e07e2b0b9b107d3" 1255 | dependencies: 1256 | code-point-at "^1.0.0" 1257 | is-fullwidth-code-point "^1.0.0" 1258 | strip-ansi "^3.0.0" 1259 | 1260 | string_decoder@~0.10.x: 1261 | version "0.10.31" 1262 | resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-0.10.31.tgz#62e203bc41766c6c28c9fc84301dab1c5310fa94" 1263 | 1264 | stringstream@~0.0.4: 1265 | version "0.0.5" 1266 | resolved "https://registry.yarnpkg.com/stringstream/-/stringstream-0.0.5.tgz#4e484cd4de5a0bbbee18e46307710a8a81621878" 1267 | 1268 | strip-ansi@^3.0.0, strip-ansi@^3.0.1: 1269 | version "3.0.1" 1270 | resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-3.0.1.tgz#6a385fb8853d952d5ff05d0e8aaf94278dc63dcf" 1271 | dependencies: 1272 | ansi-regex "^2.0.0" 1273 | 1274 | strip-bom@^2.0.0: 1275 | version "2.0.0" 1276 | resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-2.0.0.tgz#6219a85616520491f35788bdbf1447a99c7e6b0e" 1277 | dependencies: 1278 | is-utf8 "^0.2.0" 1279 | 1280 | strip-indent@^1.0.1: 1281 | version "1.0.1" 1282 | resolved "https://registry.yarnpkg.com/strip-indent/-/strip-indent-1.0.1.tgz#0c7962a6adefa7bbd4ac366460a638552ae1a0a2" 1283 | dependencies: 1284 | get-stdin "^4.0.1" 1285 | 1286 | strip-json-comments@~2.0.1: 1287 | version "2.0.1" 1288 | resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz#3c531942e908c2697c0ec344858c286c7ca0a60a" 1289 | 1290 | supports-color@1.2.0: 1291 | version "1.2.0" 1292 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-1.2.0.tgz#ff1ed1e61169d06b3cf2d588e188b18d8847e17e" 1293 | 1294 | supports-color@3.1.2: 1295 | version "3.1.2" 1296 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-3.1.2.tgz#72a262894d9d408b956ca05ff37b2ed8a6e2a2d5" 1297 | dependencies: 1298 | has-flag "^1.0.0" 1299 | 1300 | supports-color@^2.0.0: 1301 | version "2.0.0" 1302 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-2.0.0.tgz#535d045ce6b6363fa40117084629995e9df324c7" 1303 | 1304 | tar-pack@~3.3.0: 1305 | version "3.3.0" 1306 | resolved "https://registry.yarnpkg.com/tar-pack/-/tar-pack-3.3.0.tgz#30931816418f55afc4d21775afdd6720cee45dae" 1307 | dependencies: 1308 | debug "~2.2.0" 1309 | fstream "~1.0.10" 1310 | fstream-ignore "~1.0.5" 1311 | once "~1.3.3" 1312 | readable-stream "~2.1.4" 1313 | rimraf "~2.5.1" 1314 | tar "~2.2.1" 1315 | uid-number "~0.0.6" 1316 | 1317 | tar@~2.2.1: 1318 | version "2.2.1" 1319 | resolved "https://registry.yarnpkg.com/tar/-/tar-2.2.1.tgz#8e4d2a256c0e2185c6b18ad694aec968b83cb1d1" 1320 | dependencies: 1321 | block-stream "*" 1322 | fstream "^1.0.2" 1323 | inherits "2" 1324 | 1325 | through@2: 1326 | version "2.3.8" 1327 | resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5" 1328 | 1329 | to-iso-string@0.0.2: 1330 | version "0.0.2" 1331 | resolved "https://registry.yarnpkg.com/to-iso-string/-/to-iso-string-0.0.2.tgz#4dc19e664dfccbe25bd8db508b00c6da158255d1" 1332 | 1333 | tough-cookie@~2.3.0: 1334 | version "2.3.2" 1335 | resolved "https://registry.yarnpkg.com/tough-cookie/-/tough-cookie-2.3.2.tgz#f081f76e4c85720e6c37a5faced737150d84072a" 1336 | dependencies: 1337 | punycode "^1.4.1" 1338 | 1339 | trim-newlines@^1.0.0: 1340 | version "1.0.0" 1341 | resolved "https://registry.yarnpkg.com/trim-newlines/-/trim-newlines-1.0.0.tgz#5887966bb582a4503a41eb524f7d35011815a613" 1342 | 1343 | tunnel-agent@^0.6.0: 1344 | version "0.6.0" 1345 | resolved "https://registry.yarnpkg.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz#27a5dea06b36b04a0a9966774b290868f0fc40fd" 1346 | dependencies: 1347 | safe-buffer "^5.0.1" 1348 | 1349 | tweetnacl@^0.14.3, tweetnacl@~0.14.0: 1350 | version "0.14.5" 1351 | resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-0.14.5.tgz#5ae68177f192d4456269d108afa93ff8743f4f64" 1352 | 1353 | type-detect@0.1.1: 1354 | version "0.1.1" 1355 | resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-0.1.1.tgz#0ba5ec2a885640e470ea4e8505971900dac58822" 1356 | 1357 | type-detect@^1.0.0: 1358 | version "1.0.0" 1359 | resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-1.0.0.tgz#762217cc06db258ec48908a1298e8b95121e8ea2" 1360 | 1361 | type-detect@^3.0.0: 1362 | version "3.0.0" 1363 | resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-3.0.0.tgz#46d0cc8553abb7b13a352b0d6dea2fd58f2d9b55" 1364 | 1365 | type-detect@^4.0.0: 1366 | version "4.0.3" 1367 | resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-4.0.3.tgz#0e3f2670b44099b0b46c284d136a7ef49c74c2ea" 1368 | 1369 | uid-number@~0.0.6: 1370 | version "0.0.6" 1371 | resolved "https://registry.yarnpkg.com/uid-number/-/uid-number-0.0.6.tgz#0ea10e8035e8eb5b8e4449f06da1c730663baa81" 1372 | 1373 | underscore.string@~3.2.3: 1374 | version "3.2.3" 1375 | resolved "https://registry.yarnpkg.com/underscore.string/-/underscore.string-3.2.3.tgz#806992633665d5e5fcb4db1fb3a862eb68e9e6da" 1376 | 1377 | util-deprecate@~1.0.1: 1378 | version "1.0.2" 1379 | resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" 1380 | 1381 | uuid@^3.0.0: 1382 | version "3.1.0" 1383 | resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.1.0.tgz#3dd3d3e790abc24d7b0d3a034ffababe28ebbc04" 1384 | 1385 | validate-npm-package-license@^3.0.1: 1386 | version "3.0.1" 1387 | resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.1.tgz#2804babe712ad3379459acfbe24746ab2c303fbc" 1388 | dependencies: 1389 | spdx-correct "~1.0.0" 1390 | spdx-expression-parse "~1.0.0" 1391 | 1392 | verror@1.3.6: 1393 | version "1.3.6" 1394 | resolved "https://registry.yarnpkg.com/verror/-/verror-1.3.6.tgz#cff5df12946d297d2baaefaa2689e25be01c005c" 1395 | dependencies: 1396 | extsprintf "1.0.2" 1397 | 1398 | which@~1.2.1: 1399 | version "1.2.14" 1400 | resolved "https://registry.yarnpkg.com/which/-/which-1.2.14.tgz#9a87c4378f03e827cecaf1acdf56c736c01c14e5" 1401 | dependencies: 1402 | isexe "^2.0.0" 1403 | 1404 | wide-align@^1.1.0: 1405 | version "1.1.2" 1406 | resolved "https://registry.yarnpkg.com/wide-align/-/wide-align-1.1.2.tgz#571e0f1b0604636ebc0dfc21b0339bbe31341710" 1407 | dependencies: 1408 | string-width "^1.0.2" 1409 | 1410 | wrappy@1: 1411 | version "1.0.2" 1412 | resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" 1413 | 1414 | xtend@^4.0.0: 1415 | version "4.0.1" 1416 | resolved "https://registry.yarnpkg.com/xtend/-/xtend-4.0.1.tgz#a5c6d532be656e23db820efb943a1f04998d63af" 1417 | --------------------------------------------------------------------------------