UNPKG

bitgo

Version:
297 lines • 43.2 kB
"use strict"; // // Tests for API Access Tokens // // Copyright 2016, BitGo, Inc. All Rights Reserved. // Object.defineProperty(exports, "__esModule", { value: true }); /* eslint-disable @typescript-eslint/no-empty-function */ const assert_1 = require("assert"); const should = require("should"); const _ = require("lodash"); const TestBitGo = require('../lib/test_bitgo'); const TestUtil = require('./testutil'); describe('Access Token', function () { let INITIAL_TOKEN_COUNT; // used as an offset when checking if tokens were properly created or removed let loginAccessTokenHex; // save token hex for when we remove a token that was just set let bitgo; const someScopes = ['openid', 'profile', 'wallet_create', 'wallet_view_all']; before(function () { bitgo = new TestBitGo(); bitgo.initializeTestVars(); return bitgo .authenticateTestUser(bitgo.testUserOTP()) .then(function () { loginAccessTokenHex = bitgo._token; const filterFunc = function (tok) { return tok.label; }; return TestUtil.deleteTestTokens(bitgo, filterFunc); }) .then(function () { return bitgo.listAccessTokens(); }) .then(function (tokens) { INITIAL_TOKEN_COUNT = tokens.length; }); }); describe('authentication with access token', function () { let addedTokenHex; it('should authenticate with added access token', function () { return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes }) .then(function (res) { addedTokenHex = res.token; return bitgo.authenticateWithAccessToken({ accessToken: addedTokenHex }); }) .then(function () { bitgo._token.should.equal(addedTokenHex); // set _token back to original login value return bitgo.authenticateWithAccessToken({ accessToken: loginAccessTokenHex }); }) .then(function () { bitgo._token.should.equal(loginAccessTokenHex); return bitgo.removeAccessToken({ label: 'test token' }); }); }); }); describe('Add', function () { describe('bad args', function () { it('arguments', function () { assert_1.strict.throws(function () { bitgo.addAccessToken({}, 'invalid'); }); assert_1.strict.throws(function () { bitgo.addAccessToken({}, function () { }); }); assert_1.strict.throws(function () { bitgo.addAccessToken({ otp: bitgo.testUserOTP(), scope: ['wallet_view_all', 'openid', 'profile'], }, 'invalid'); }); }); it('fails to add without scope', function () { assert_1.strict.throws(function () { bitgo.addAccessToken({ otp: bitgo.testUserOTP(), }, function () { }); }); }); it('fails to add with empty scope', function () { assert_1.strict.throws(function () { bitgo.addAccessToken({ otp: bitgo.testUserOTP(), scope: [], }, function () { }); }); }); it('fails to add with incorrect type of scope', function () { assert_1.strict.throws(function () { bitgo.addAccessToken({ otp: bitgo.testUserOTP(), scope: 'notAnArray', }, function () { }); }); }); it('fails to add with invalid scope', function () { const promise = bitgo.addAccessToken({ otp: 'badToken', label: 'test token', scope: ['invalid'] }); return TestUtil.throws(promise, 'invalid scope'); }); it('fails to add with bad otp', function () { const promise = bitgo.addAccessToken({ otp: 'badToken', label: 'test token', scope: someScopes }); return TestUtil.throws(promise, 'invalid'); }); }); describe('success', function () { afterEach(function () { return bitgo.removeAccessToken({ label: 'test token' }); }); it('simple add', function () { return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes }) .then(function (res) { res.should.have.property('user'); res.should.have.property('scope'); res.should.have.property('created'); res.should.have.property('expires'); res.should.have.property('origin'); res.should.have.property('label'); res.should.have.property('isExtensible'); res.should.have.property('token'); res.should.not.have.property('unlock'); res.should.not.have.property('enterprise'); res.label.should.equal('test token'); res.isExtensible.should.equal(false); return bitgo.listAccessTokens(); }) .then(function (tokens) { const numTokens = tokens.length; numTokens.should.equal(INITIAL_TOKEN_COUNT + 1); }); }); it('duration', function () { const DURATION = 3600 * 10; // ten days return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', duration: DURATION, scope: someScopes }) .then(function (res) { res.label.should.equal('test token'); const created = res.created; const expires = res.expires; const createdPlusDuration = new Date(new Date(created).getTime() + DURATION * 1000).getTime(); const expiresTime = new Date(expires).getTime(); const leeway = 10; // because of the miniscule time it takes to execute a function, we give a 10 ms leeway in the time differences createdPlusDuration.should.be.greaterThan(expiresTime - leeway); createdPlusDuration.should.be.lessThan(expiresTime + leeway); }); }); it('ipRestrict', function () { const IPRESTRICT = ['0.0.0.0', '8.8.8.8']; return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', ipRestrict: IPRESTRICT, scope: someScopes }) .then(function (token) { token.should.have.property('token'); }); }); it('txValueLimit', function () { const TXVALUELIMIT = 1e8; // 1 BTC return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', txValueLimit: TXVALUELIMIT, scope: someScopes, }) .then(function (res) { res.unlock.txValueLimit.should.equal(1e8); }); }); // see some examples of Scope Values under https://www.bitgo.com/api/#partner-oauth it('scopes', function () { return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes }) .then(function (res) { res.scope.should.have.length(4); }); }); }); }); describe('List', function () { it('should list no new access token', function () { return bitgo.listAccessTokens().then(function (tokens) { tokens.length.should.equal(INITIAL_TOKEN_COUNT); }); }); it('should add and list single access token', function () { return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes }) .then(function (res) { res.label.should.equal('test token'); return bitgo.listAccessTokens(); }) .then(function (tokens) { tokens.length.should.equal(INITIAL_TOKEN_COUNT + 1); const token = _.find(tokens, function (tok) { return tok.label === 'test token'; }); should.exist(token); }); }); it('should add another and list multiple access tokens', function () { let token1; let token2; return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token 2', scope: someScopes }) .then(function (res) { res.label.should.equal('test token 2'); return bitgo.listAccessTokens(); }) .then(function (tokens) { tokens.length.should.equal(INITIAL_TOKEN_COUNT + 2); token1 = _.find(tokens, function (tok) { return tok.label === 'test token 2'; }); token2 = _.find(tokens, function (tok) { return tok.label === 'test token'; }); should.exist(token1); should.exist(token2); // cleanup access tokens for future tests return bitgo.removeAccessToken({ id: token1.id }); }) .then(function () { return bitgo.removeAccessToken({ id: token2.id }); }); }); }); describe('Update', function () { should.exist(true); // no-op // access tokens have no update API, they can only be created or removed }); describe('Remove', function () { let ambiguousTokenId; before(function () { return bitgo.addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes }); }); it('arguments', function () { assert_1.strict.throws(function () { bitgo.removeAccessToken({}, 'invalid'); }); assert_1.strict.throws(function () { bitgo.removeAccessToken({}, function () { }); }); assert_1.strict.throws(function () { bitgo.removeAccessToken({ id: 'non-existent id' }, 'invalid'); }); assert_1.strict.throws(function () { bitgo.removeAccessToken({ label: 'non-existent label' }, 'invalid'); }); }); it('should fail with ambigous remove', function () { // begin by adding second token return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes }) .then(function (token) { ambiguousTokenId = token.id; const promise = bitgo.removeAccessToken({ label: 'test token' }); return TestUtil.throws(promise, 'ambiguous call: multiple tokens matching this label'); }); }); it('should remove by label', function () { return bitgo .listAccessTokens() .then(function (tokens) { tokens.length.should.equal(INITIAL_TOKEN_COUNT + 2); // now remove return bitgo.removeAccessToken({ id: ambiguousTokenId }); }) .then(function () { return bitgo.listAccessTokens(); }) .then(function (tokens) { tokens.length.should.equal(INITIAL_TOKEN_COUNT + 1); return bitgo.removeAccessToken({ label: 'test token' }); }) .then(function () { return bitgo.listAccessTokens(); }) .then(function (tokens) { tokens.length.should.equal(INITIAL_TOKEN_COUNT); }); }); it('should remove access token by id', function () { return bitgo .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes }) .then(function (tok) { return bitgo.removeAccessToken({ id: tok.id }); }) .then(function () { return bitgo.listAccessTokens(); }) .then(function (tokens) { tokens.length.should.equal(INITIAL_TOKEN_COUNT); }); }); }); }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"accesstoken.js","sourceRoot":"","sources":["../../../test/integration/accesstoken.ts"],"names":[],"mappings":";AAAA,EAAE;AACF,8BAA8B;AAC9B,EAAE;AACF,oDAAoD;AACpD,EAAE;;AAEF,yDAAyD;AAEzD,mCAA0C;AAC1C,iCAAkC;AAClC,4BAA4B;AAE5B,MAAM,SAAS,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAAC;AAC/C,MAAM,QAAQ,GAAG,OAAO,CAAC,YAAY,CAAC,CAAC;AAEvC,QAAQ,CAAC,cAAc,EAAE;IACvB,IAAI,mBAAmB,CAAC,CAAC,6EAA6E;IACtG,IAAI,mBAAmB,CAAC,CAAC,8DAA8D;IACvF,IAAI,KAAK,CAAC;IACV,MAAM,UAAU,GAAG,CAAC,QAAQ,EAAE,SAAS,EAAE,eAAe,EAAE,iBAAiB,CAAC,CAAC;IAE7E,MAAM,CAAC;QACL,KAAK,GAAG,IAAI,SAAS,EAAE,CAAC;QACxB,KAAK,CAAC,kBAAkB,EAAE,CAAC;QAC3B,OAAO,KAAK;aACT,oBAAoB,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;aACzC,IAAI,CAAC;YACJ,mBAAmB,GAAG,KAAK,CAAC,MAAM,CAAC;YAEnC,MAAM,UAAU,GAAG,UAAU,GAAG;gBAC9B,OAAO,GAAG,CAAC,KAAK,CAAC;YACnB,CAAC,CAAC;YACF,OAAO,QAAQ,CAAC,gBAAgB,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;QACtD,CAAC,CAAC;aACD,IAAI,CAAC;YACJ,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC;QAClC,CAAC,CAAC;aACD,IAAI,CAAC,UAAU,MAAM;YACpB,mBAAmB,GAAG,MAAM,CAAC,MAAM,CAAC;QACtC,CAAC,CAAC,CAAC;IACP,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,kCAAkC,EAAE;QAC3C,IAAI,aAAa,CAAC;QAClB,EAAE,CAAC,6CAA6C,EAAE;YAChD,OAAO,KAAK;iBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;iBACpF,IAAI,CAAC,UAAU,GAAG;gBACjB,aAAa,GAAG,GAAG,CAAC,KAAK,CAAC;gBAC1B,OAAO,KAAK,CAAC,2BAA2B,CAAC,EAAE,WAAW,EAAE,aAAa,EAAE,CAAC,CAAC;YAC3E,CAAC,CAAC;iBACD,IAAI,CAAC;gBACJ,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;gBAEzC,0CAA0C;gBAC1C,OAAO,KAAK,CAAC,2BAA2B,CAAC,EAAE,WAAW,EAAE,mBAAmB,EAAE,CAAC,CAAC;YACjF,CAAC,CAAC;iBACD,IAAI,CAAC;gBACJ,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,CAAC,CAAC;gBAE/C,OAAO,KAAK,CAAC,iBAAiB,CAAC,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAC1D,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,KAAK,EAAE;QACd,QAAQ,CAAC,UAAU,EAAE;YACnB,EAAE,CAAC,WAAW,EAAE;gBACd,eAAM,CAAC,MAAM,CAAC;oBACZ,KAAK,CAAC,cAAc,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;gBACtC,CAAC,CAAC,CAAC;gBACH,eAAM,CAAC,MAAM,CAAC;oBACZ,KAAK,CAAC,cAAc,CAAC,EAAE,EAAE,cAAa,CAAC,CAAC,CAAC;gBAC3C,CAAC,CAAC,CAAC;gBAEH,eAAM,CAAC,MAAM,CAAC;oBACZ,KAAK,CAAC,cAAc,CAClB;wBACE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE;wBACxB,KAAK,EAAE,CAAC,iBAAiB,EAAE,QAAQ,EAAE,SAAS,CAAC;qBAChD,EACD,SAAS,CACV,CAAC;gBACJ,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,4BAA4B,EAAE;gBAC/B,eAAM,CAAC,MAAM,CAAC;oBACZ,KAAK,CAAC,cAAc,CAClB;wBACE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE;qBACzB,EACD,cAAa,CAAC,CACf,CAAC;gBACJ,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,+BAA+B,EAAE;gBAClC,eAAM,CAAC,MAAM,CAAC;oBACZ,KAAK,CAAC,cAAc,CAClB;wBACE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE;wBACxB,KAAK,EAAE,EAAE;qBACV,EACD,cAAa,CAAC,CACf,CAAC;gBACJ,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,2CAA2C,EAAE;gBAC9C,eAAM,CAAC,MAAM,CAAC;oBACZ,KAAK,CAAC,cAAc,CAClB;wBACE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE;wBACxB,KAAK,EAAE,YAAY;qBACpB,EACD,cAAa,CAAC,CACf,CAAC;gBACJ,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,iCAAiC,EAAE;gBACpC,MAAM,OAAO,GAAG,KAAK,CAAC,cAAc,CAAC,EAAE,GAAG,EAAE,UAAU,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;gBACnG,OAAO,QAAQ,CAAC,MAAM,CAAC,OAAO,EAAE,eAAe,CAAC,CAAC;YACnD,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,2BAA2B,EAAE;gBAC9B,MAAM,OAAO,GAAG,KAAK,CAAC,cAAc,CAAC,EAAE,GAAG,EAAE,UAAU,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC,CAAC;gBAClG,OAAO,QAAQ,CAAC,MAAM,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;YAC7C,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,SAAS,EAAE;YAClB,SAAS,CAAC;gBACR,OAAO,KAAK,CAAC,iBAAiB,CAAC,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAC1D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,YAAY,EAAE;gBACf,OAAO,KAAK;qBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;qBACpF,IAAI,CAAC,UAAU,GAAG;oBACjB,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;oBACjC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;oBAClC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;oBACpC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;oBACpC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;oBACnC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;oBAClC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;oBACzC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;oBAElC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;oBACvC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;oBAE3C,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;oBACrC,GAAG,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;oBAErC,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC;gBAClC,CAAC,CAAC;qBACD,IAAI,CAAC,UAAU,MAAM;oBACpB,MAAM,SAAS,GAAG,MAAM,CAAC,MAAM,CAAC;oBAChC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,GAAG,CAAC,CAAC,CAAC;gBAClD,CAAC,CAAC,CAAC;YACP,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,UAAU,EAAE;gBACb,MAAM,QAAQ,GAAG,IAAI,GAAG,EAAE,CAAC,CAAC,WAAW;gBACvC,OAAO,KAAK;qBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;qBACxG,IAAI,CAAC,UAAU,GAAG;oBACjB,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;oBAErC,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC;oBAC5B,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC;oBAC5B,MAAM,mBAAmB,GAAG,IAAI,IAAI,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,GAAG,QAAQ,GAAG,IAAI,CAAC,CAAC,OAAO,EAAE,CAAC;oBAC9F,MAAM,WAAW,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,CAAC;oBAChD,MAAM,MAAM,GAAG,EAAE,CAAC,CAAC,+GAA+G;oBAClI,mBAAmB,CAAC,MAAM,CAAC,EAAE,CAAC,WAAW,CAAC,WAAW,GAAG,MAAM,CAAC,CAAC;oBAChE,mBAAmB,CAAC,MAAM,CAAC,EAAE,CAAC,QAAQ,CAAC,WAAW,GAAG,MAAM,CAAC,CAAC;gBAC/D,CAAC,CAAC,CAAC;YACP,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,YAAY,EAAE;gBACf,MAAM,UAAU,GAAG,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;gBAC1C,OAAO,KAAK;qBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,UAAU,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;qBAC5G,IAAI,CAAC,UAAU,KAAK;oBACnB,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBACtC,CAAC,CAAC,CAAC;YACP,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,cAAc,EAAE;gBACjB,MAAM,YAAY,GAAG,GAAG,CAAC,CAAC,QAAQ;gBAClC,OAAO,KAAK;qBACT,cAAc,CAAC;oBACd,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE;oBACxB,KAAK,EAAE,YAAY;oBACnB,YAAY,EAAE,YAAY;oBAC1B,KAAK,EAAE,UAAU;iBAClB,CAAC;qBACD,IAAI,CAAC,UAAU,GAAG;oBACjB,GAAG,CAAC,MAAM,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBAC5C,CAAC,CAAC,CAAC;YACP,CAAC,CAAC,CAAC;YAEH,mFAAmF;YACnF,EAAE,CAAC,QAAQ,EAAE;gBACX,OAAO,KAAK;qBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;qBACpF,IAAI,CAAC,UAAU,GAAG;oBACjB,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;gBAClC,CAAC,CAAC,CAAC;YACP,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,MAAM,EAAE;QACf,EAAE,CAAC,iCAAiC,EAAE;YACpC,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC,IAAI,CAAC,UAAU,MAAM;gBACnD,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,CAAC,CAAC;YAClD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yCAAyC,EAAE;YAC5C,OAAO,KAAK;iBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;iBACpF,IAAI,CAAC,UAAU,GAAG;gBACjB,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;gBAErC,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAClC,CAAC,CAAC;iBACD,IAAI,CAAC,UAAU,MAAM;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,GAAG,CAAC,CAAC,CAAC;gBACpD,MAAM,KAAK,GAAG,CAAC,CAAC,IAAI,CAAC,MAAM,EAAE,UAAU,GAAG;oBACxC,OAAO,GAAG,CAAC,KAAK,KAAK,YAAY,CAAC;gBACpC,CAAC,CAAC,CAAC;gBACH,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACtB,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oDAAoD,EAAE;YACvD,IAAI,MAAM,CAAC;YACX,IAAI,MAAM,CAAC;YACX,OAAO,KAAK;iBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,cAAc,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;iBACtF,IAAI,CAAC,UAAU,GAAG;gBACjB,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;gBAEvC,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAClC,CAAC,CAAC;iBACD,IAAI,CAAC,UAAU,MAAM;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,GAAG,CAAC,CAAC,CAAC;gBAEpD,MAAM,GAAG,CAAC,CAAC,IAAI,CAAC,MAAM,EAAE,UAAU,GAAG;oBACnC,OAAO,GAAG,CAAC,KAAK,KAAK,cAAc,CAAC;gBACtC,CAAC,CAAC,CAAC;gBACH,MAAM,GAAG,CAAC,CAAC,IAAI,CAAC,MAAM,EAAE,UAAU,GAAG;oBACnC,OAAO,GAAG,CAAC,KAAK,KAAK,YAAY,CAAC;gBACpC,CAAC,CAAC,CAAC;gBAEH,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;gBACrB,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;gBAErB,yCAAyC;gBACzC,OAAO,KAAK,CAAC,iBAAiB,CAAC,EAAE,EAAE,EAAE,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC;YACpD,CAAC,CAAC;iBACD,IAAI,CAAC;gBACJ,OAAO,KAAK,CAAC,iBAAiB,CAAC,EAAE,EAAE,EAAE,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC;YACpD,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,QAAQ,EAAE;QACjB,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ;QAC5B,wEAAwE;IAC1E,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,QAAQ,EAAE;QACjB,IAAI,gBAAgB,CAAC;QACrB,MAAM,CAAC;YACL,OAAO,KAAK,CAAC,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC,CAAC;QACpG,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,WAAW,EAAE;YACd,eAAM,CAAC,MAAM,CAAC;gBACZ,KAAK,CAAC,iBAAiB,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;YACzC,CAAC,CAAC,CAAC;YACH,eAAM,CAAC,MAAM,CAAC;gBACZ,KAAK,CAAC,iBAAiB,CAAC,EAAE,EAAE,cAAa,CAAC,CAAC,CAAC;YAC9C,CAAC,CAAC,CAAC;YACH,eAAM,CAAC,MAAM,CAAC;gBACZ,KAAK,CAAC,iBAAiB,CAAC,EAAE,EAAE,EAAE,iBAAiB,EAAE,EAAE,SAAS,CAAC,CAAC;YAChE,CAAC,CAAC,CAAC;YACH,eAAM,CAAC,MAAM,CAAC;gBACZ,KAAK,CAAC,iBAAiB,CAAC,EAAE,KAAK,EAAE,oBAAoB,EAAE,EAAE,SAAS,CAAC,CAAC;YACtE,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kCAAkC,EAAE;YACrC,+BAA+B;YAC/B,OAAO,KAAK;iBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;iBACpF,IAAI,CAAC,UAAU,KAAK;gBACnB,gBAAgB,GAAG,KAAK,CAAC,EAAE,CAAC;gBAC5B,MAAM,OAAO,GAAG,KAAK,CAAC,iBAAiB,CAAC,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;gBACjE,OAAO,QAAQ,CAAC,MAAM,CAAC,OAAO,EAAE,qDAAqD,CAAC,CAAC;YACzF,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,wBAAwB,EAAE;YAC3B,OAAO,KAAK;iBACT,gBAAgB,EAAE;iBAClB,IAAI,CAAC,UAAU,MAAM;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,GAAG,CAAC,CAAC,CAAC;gBAEpD,aAAa;gBACb,OAAO,KAAK,CAAC,iBAAiB,CAAC,EAAE,EAAE,EAAE,gBAAgB,EAAE,CAAC,CAAC;YAC3D,CAAC,CAAC;iBACD,IAAI,CAAC;gBACJ,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAClC,CAAC,CAAC;iBACD,IAAI,CAAC,UAAU,MAAM;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,GAAG,CAAC,CAAC,CAAC;gBAEpD,OAAO,KAAK,CAAC,iBAAiB,CAAC,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAC1D,CAAC,CAAC;iBACD,IAAI,CAAC;gBACJ,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAClC,CAAC,CAAC;iBACD,IAAI,CAAC,UAAU,MAAM;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,CAAC,CAAC;YAClD,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kCAAkC,EAAE;YACrC,OAAO,KAAK;iBACT,cAAc,CAAC,EAAE,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC;iBACpF,IAAI,CAAC,UAAU,GAAG;gBACjB,OAAO,KAAK,CAAC,iBAAiB,CAAC,EAAE,EAAE,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC;YACjD,CAAC,CAAC;iBACD,IAAI,CAAC;gBACJ,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAClC,CAAC,CAAC;iBACD,IAAI,CAAC,UAAU,MAAM;gBACpB,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,CAAC,CAAC;YAClD,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["//\n// Tests for API Access Tokens\n//\n// Copyright 2016, BitGo, Inc.  All Rights Reserved.\n//\n\n/* eslint-disable @typescript-eslint/no-empty-function */\n\nimport { strict as assert } from 'assert';\nimport should = require('should');\nimport * as _ from 'lodash';\n\nconst TestBitGo = require('../lib/test_bitgo');\nconst TestUtil = require('./testutil');\n\ndescribe('Access Token', function () {\n  let INITIAL_TOKEN_COUNT; // used as an offset when checking if tokens were properly created or removed\n  let loginAccessTokenHex; // save token hex for when we remove a token that was just set\n  let bitgo;\n  const someScopes = ['openid', 'profile', 'wallet_create', 'wallet_view_all'];\n\n  before(function () {\n    bitgo = new TestBitGo();\n    bitgo.initializeTestVars();\n    return bitgo\n      .authenticateTestUser(bitgo.testUserOTP())\n      .then(function () {\n        loginAccessTokenHex = bitgo._token;\n\n        const filterFunc = function (tok) {\n          return tok.label;\n        };\n        return TestUtil.deleteTestTokens(bitgo, filterFunc);\n      })\n      .then(function () {\n        return bitgo.listAccessTokens();\n      })\n      .then(function (tokens) {\n        INITIAL_TOKEN_COUNT = tokens.length;\n      });\n  });\n\n  describe('authentication with access token', function () {\n    let addedTokenHex;\n    it('should authenticate with added access token', function () {\n      return bitgo\n        .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes })\n        .then(function (res) {\n          addedTokenHex = res.token;\n          return bitgo.authenticateWithAccessToken({ accessToken: addedTokenHex });\n        })\n        .then(function () {\n          bitgo._token.should.equal(addedTokenHex);\n\n          // set _token back to original login value\n          return bitgo.authenticateWithAccessToken({ accessToken: loginAccessTokenHex });\n        })\n        .then(function () {\n          bitgo._token.should.equal(loginAccessTokenHex);\n\n          return bitgo.removeAccessToken({ label: 'test token' });\n        });\n    });\n  });\n\n  describe('Add', function () {\n    describe('bad args', function () {\n      it('arguments', function () {\n        assert.throws(function () {\n          bitgo.addAccessToken({}, 'invalid');\n        });\n        assert.throws(function () {\n          bitgo.addAccessToken({}, function () {});\n        });\n\n        assert.throws(function () {\n          bitgo.addAccessToken(\n            {\n              otp: bitgo.testUserOTP(),\n              scope: ['wallet_view_all', 'openid', 'profile'],\n            },\n            'invalid'\n          );\n        });\n      });\n\n      it('fails to add without scope', function () {\n        assert.throws(function () {\n          bitgo.addAccessToken(\n            {\n              otp: bitgo.testUserOTP(),\n            },\n            function () {}\n          );\n        });\n      });\n\n      it('fails to add with empty scope', function () {\n        assert.throws(function () {\n          bitgo.addAccessToken(\n            {\n              otp: bitgo.testUserOTP(),\n              scope: [],\n            },\n            function () {}\n          );\n        });\n      });\n\n      it('fails to add with incorrect type of scope', function () {\n        assert.throws(function () {\n          bitgo.addAccessToken(\n            {\n              otp: bitgo.testUserOTP(),\n              scope: 'notAnArray',\n            },\n            function () {}\n          );\n        });\n      });\n\n      it('fails to add with invalid scope', function () {\n        const promise = bitgo.addAccessToken({ otp: 'badToken', label: 'test token', scope: ['invalid'] });\n        return TestUtil.throws(promise, 'invalid scope');\n      });\n\n      it('fails to add with bad otp', function () {\n        const promise = bitgo.addAccessToken({ otp: 'badToken', label: 'test token', scope: someScopes });\n        return TestUtil.throws(promise, 'invalid');\n      });\n    });\n\n    describe('success', function () {\n      afterEach(function () {\n        return bitgo.removeAccessToken({ label: 'test token' });\n      });\n\n      it('simple add', function () {\n        return bitgo\n          .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes })\n          .then(function (res) {\n            res.should.have.property('user');\n            res.should.have.property('scope');\n            res.should.have.property('created');\n            res.should.have.property('expires');\n            res.should.have.property('origin');\n            res.should.have.property('label');\n            res.should.have.property('isExtensible');\n            res.should.have.property('token');\n\n            res.should.not.have.property('unlock');\n            res.should.not.have.property('enterprise');\n\n            res.label.should.equal('test token');\n            res.isExtensible.should.equal(false);\n\n            return bitgo.listAccessTokens();\n          })\n          .then(function (tokens) {\n            const numTokens = tokens.length;\n            numTokens.should.equal(INITIAL_TOKEN_COUNT + 1);\n          });\n      });\n\n      it('duration', function () {\n        const DURATION = 3600 * 10; // ten days\n        return bitgo\n          .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', duration: DURATION, scope: someScopes })\n          .then(function (res) {\n            res.label.should.equal('test token');\n\n            const created = res.created;\n            const expires = res.expires;\n            const createdPlusDuration = new Date(new Date(created).getTime() + DURATION * 1000).getTime();\n            const expiresTime = new Date(expires).getTime();\n            const leeway = 10; // because of the miniscule time it takes to execute a function, we give a 10 ms leeway in the time differences\n            createdPlusDuration.should.be.greaterThan(expiresTime - leeway);\n            createdPlusDuration.should.be.lessThan(expiresTime + leeway);\n          });\n      });\n\n      it('ipRestrict', function () {\n        const IPRESTRICT = ['0.0.0.0', '8.8.8.8'];\n        return bitgo\n          .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', ipRestrict: IPRESTRICT, scope: someScopes })\n          .then(function (token) {\n            token.should.have.property('token');\n          });\n      });\n\n      it('txValueLimit', function () {\n        const TXVALUELIMIT = 1e8; // 1 BTC\n        return bitgo\n          .addAccessToken({\n            otp: bitgo.testUserOTP(),\n            label: 'test token',\n            txValueLimit: TXVALUELIMIT,\n            scope: someScopes,\n          })\n          .then(function (res) {\n            res.unlock.txValueLimit.should.equal(1e8);\n          });\n      });\n\n      // see some examples of Scope Values under https://www.bitgo.com/api/#partner-oauth\n      it('scopes', function () {\n        return bitgo\n          .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes })\n          .then(function (res) {\n            res.scope.should.have.length(4);\n          });\n      });\n    });\n  });\n\n  describe('List', function () {\n    it('should list no new access token', function () {\n      return bitgo.listAccessTokens().then(function (tokens) {\n        tokens.length.should.equal(INITIAL_TOKEN_COUNT);\n      });\n    });\n\n    it('should add and list single access token', function () {\n      return bitgo\n        .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes })\n        .then(function (res) {\n          res.label.should.equal('test token');\n\n          return bitgo.listAccessTokens();\n        })\n        .then(function (tokens) {\n          tokens.length.should.equal(INITIAL_TOKEN_COUNT + 1);\n          const token = _.find(tokens, function (tok) {\n            return tok.label === 'test token';\n          });\n          should.exist(token);\n        });\n    });\n\n    it('should add another and list multiple access tokens', function () {\n      let token1;\n      let token2;\n      return bitgo\n        .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token 2', scope: someScopes })\n        .then(function (res) {\n          res.label.should.equal('test token 2');\n\n          return bitgo.listAccessTokens();\n        })\n        .then(function (tokens) {\n          tokens.length.should.equal(INITIAL_TOKEN_COUNT + 2);\n\n          token1 = _.find(tokens, function (tok) {\n            return tok.label === 'test token 2';\n          });\n          token2 = _.find(tokens, function (tok) {\n            return tok.label === 'test token';\n          });\n\n          should.exist(token1);\n          should.exist(token2);\n\n          // cleanup access tokens for future tests\n          return bitgo.removeAccessToken({ id: token1.id });\n        })\n        .then(function () {\n          return bitgo.removeAccessToken({ id: token2.id });\n        });\n    });\n  });\n\n  describe('Update', function () {\n    should.exist(true); // no-op\n    // access tokens have no update API, they can only be created or removed\n  });\n\n  describe('Remove', function () {\n    let ambiguousTokenId;\n    before(function () {\n      return bitgo.addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes });\n    });\n\n    it('arguments', function () {\n      assert.throws(function () {\n        bitgo.removeAccessToken({}, 'invalid');\n      });\n      assert.throws(function () {\n        bitgo.removeAccessToken({}, function () {});\n      });\n      assert.throws(function () {\n        bitgo.removeAccessToken({ id: 'non-existent id' }, 'invalid');\n      });\n      assert.throws(function () {\n        bitgo.removeAccessToken({ label: 'non-existent label' }, 'invalid');\n      });\n    });\n\n    it('should fail with ambigous remove', function () {\n      // begin by adding second token\n      return bitgo\n        .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes })\n        .then(function (token) {\n          ambiguousTokenId = token.id;\n          const promise = bitgo.removeAccessToken({ label: 'test token' });\n          return TestUtil.throws(promise, 'ambiguous call: multiple tokens matching this label');\n        });\n    });\n\n    it('should remove by label', function () {\n      return bitgo\n        .listAccessTokens()\n        .then(function (tokens) {\n          tokens.length.should.equal(INITIAL_TOKEN_COUNT + 2);\n\n          // now remove\n          return bitgo.removeAccessToken({ id: ambiguousTokenId });\n        })\n        .then(function () {\n          return bitgo.listAccessTokens();\n        })\n        .then(function (tokens) {\n          tokens.length.should.equal(INITIAL_TOKEN_COUNT + 1);\n\n          return bitgo.removeAccessToken({ label: 'test token' });\n        })\n        .then(function () {\n          return bitgo.listAccessTokens();\n        })\n        .then(function (tokens) {\n          tokens.length.should.equal(INITIAL_TOKEN_COUNT);\n        });\n    });\n\n    it('should remove access token by id', function () {\n      return bitgo\n        .addAccessToken({ otp: bitgo.testUserOTP(), label: 'test token', scope: someScopes })\n        .then(function (tok) {\n          return bitgo.removeAccessToken({ id: tok.id });\n        })\n        .then(function () {\n          return bitgo.listAccessTokens();\n        })\n        .then(function (tokens) {\n          tokens.length.should.equal(INITIAL_TOKEN_COUNT);\n        });\n    });\n  });\n});\n"]}