adal-node
Version:
Windows Azure Active Directory Client Library for node
522 lines (465 loc) • 18.3 kB
JavaScript
/*
* @copyright
* Copyright © Microsoft Open Technologies, Inc.
*
* All Rights Reserved
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http: *www.apache.org/licenses/LICENSE-2.0
*
* THIS CODE IS PROVIDED *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS
* OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION
* ANY IMPLIED WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A
* PARTICULAR PURPOSE, MERCHANTABILITY OR NON-INFRINGEMENT.
*
* See the Apache License, Version 2.0 for the specific language
* governing permissions and limitations under the License.
*/
'use strict';
var _ = require('underscore');
var crypto = require('crypto');
require('date-utils'); // Adds a number of convenience methods to the builtin Date object.
var Logger = require('./log').Logger;
var constants = require('./constants');
var cacheConstants = constants.Cache;
var TokenResponseFields = constants.TokenResponseFields;
// TODO: remove this.
// There is a PM requirement that developers be able to look in to the cache and manipulate the cache based on
// the parameters (authority, resource, clientId, userId), in any combination. They must be able find, add, and remove
// tokens based on those parameters. Any default cache that the API supplies must allow for this query pattern.
// This has the following implications:
// The developer must not be required to calculate any special fields, such as hashes or unique keys.
//
// The default cache implementation can not include optimizations that break the previous requirement.
// This means that we can only do complete scans of the data and equality can only be calculated based on
// equality of all of the individual fields.
//
// The cache interface can not make any assumption about the query efficency of the cache nor can
// it help in optimizing those queries.
//
// There is no simple sorting optimization, rather a series of indexes, and index intersection would
// be necessary.
//
// If for some reason the developer tries to update the cache with a new entry that may be a refresh
// token, they will not know that they need to update all of the refresh tokens or they may get strange
// behavior.
//
// Related to the above, there is no definition of a coherent cache. And if there was there would be
// no way for our API to enforce it. What about duplicates?
//
// there be a single cache entry per (authority, resource, clientId)
// tuple, with no special tokens (i.e. MRRT tokens)
// Required cache operations
//
// Constants
var METADATA_CLIENTID = '_clientId';
var METADATA_AUTHORITY = '_authority';
function nop(placeHolder, callback) {
callback();
}
/*
* This is a place holder cache that does nothing.
*/
var nopCache = {
add : nop,
addMany : nop,
remove : nop,
removeMany : nop,
find : nop
};
function createTokenHash(token) {
var hashAlg = crypto.createHash(cacheConstants.HASH_ALGORITHM);
hashAlg.update(token, 'utf8');
return hashAlg.digest('base64');
}
function createTokenIdMessage(entry) {
var accessTokenHash = createTokenHash(entry[TokenResponseFields.ACCESS_TOKEN]);
var message = 'AccessTokenId: ' + accessTokenHash;
if (entry[TokenResponseFields.REFRESH_TOKEN]) {
var refreshTokenHash = createTokenHash(entry[TokenResponseFields.REFRESH_TOKEN]);
message += ', RefreshTokenId: ' + refreshTokenHash;
}
return message;
}
/**
* This is the callback that is passed to all acquireToken variants below.
* @callback RefreshEntryFunction
* @memberOf CacheDriver
* @param {object} tokenResponse A token response to refresh.
* @param {string} [resource] The resource for which to obtain the token if it is different from the original token.
* @param {AcquireTokenCallback} callback Called on completion with an error or a new entry to add to the cache.
*/
/**
* Constructs a new CacheDriver object.
* @constructor
* @private
* @param {object} callContext Contains any context information that applies to the request.
* @param {string} authority
* @param {TokenCache} [cache] A token cache to use. If none is passed then the CacheDriver instance
* will not cache.
* @param {RefreshEntryFunction} refreshFunction
*/
function CacheDriver(callContext, authority, resource, clientId, cache, refreshFunction) {
this._callContext = callContext;
this._log = new Logger('CacheDriver', callContext._logContext);
this._authority = authority;
this._resource = resource;
this._clientId = clientId;
this._cache = cache || nopCache;
this._refreshFunction = refreshFunction;
}
/**
* This is the callback that is passed to all acquireToken variants below.
* @callback QueryCallback
* @memberOf CacheDriver
* @param {Error} [error] If the request fails this parameter will contain an Error object.
* @param {Array} [response] On a succesful request returns an array of matched entries.
*/
/**
* The cache driver query function. Ensures that all queries are authority specific.
* @param {object} query A query object. Can contain a clientId or userId or both.
* @param {QueryCallback} callback
*/
CacheDriver.prototype._find = function(query, callback) {
this._cache.find(query, callback);
};
/**
* Queries for all entries that might satisfy a request for a cached token.
* @param {object} query A query object. Can contain a clientId or userId or both.
* @param {QueryCallback} callback
*/
CacheDriver.prototype._getPotentialEntries = function(query, callback) {
var self = this;
var potentialEntriesQuery = {};
if (query.clientId) {
potentialEntriesQuery[METADATA_CLIENTID] = query.clientId;
}
if (query.userId) {
potentialEntriesQuery[TokenResponseFields.USER_ID] = query.userId;
}
this._log.verbose('Looking for potential cache entries:');
this._log.verbose(JSON.stringify(potentialEntriesQuery), true);
this._find(potentialEntriesQuery, function(err, entries) {
self._log.verbose('Found ' + entries.length + ' potential entries.');
callback(err, entries);
return;
});
};
/**
* Finds all multi resource refresh tokens in the cache.
* Refresh token is bound to userId, clientId.
* @param {QueryCallback} callback
*/
CacheDriver.prototype._findMRRTTokensForUser = function(user, callback) {
this._find({ isMRRT : true, userId : user, _clientId : this._clientId}, callback);
};
/**
* This is the callback that is passed to all acquireToken variants below.
* @callback SingleEntryCallback
* @memberOf CacheDriver
* @param {Error} [error] If the request fails this parameter will contain an Error object.
* @param {object} [response] On a succesful request returns a single cache entry.
*/
/**
* Finds a single entry that matches the query. If multiple entries are found that satisfy the query
* then an error will be returned.
* @param {object} query A query object.
* @param {SingleEntryCallback} callback
*/
CacheDriver.prototype._loadSingleEntryFromCache = function(query, callback) {
var self = this;
this._getPotentialEntries(query, function(err, potentialEntries) {
if (err) {
callback(err);
return;
}
var returnVal;
var isResourceTenantSpecific;
if (potentialEntries && 0 < potentialEntries.length) {
var resourceTenantSpecificEntries = _.where(potentialEntries, { resource : self._resource, _authority : self._authority });
if (!resourceTenantSpecificEntries || 0 === resourceTenantSpecificEntries.length) {
self._log.verbose('No resource specific cache entries found.');
// There are no resource specific entries. Find an MRRT token.
var mrrtTokens = _.where(potentialEntries, { isMRRT : true });
if (mrrtTokens && mrrtTokens.length > 0) {
self._log.verbose('Found an MRRT token.');
returnVal = mrrtTokens[0];
} else {
self._log.verbose('No MRRT tokens found.');
}
} else if (resourceTenantSpecificEntries.length === 1) {
self._log.verbose('Resource specific token found.');
returnVal = resourceTenantSpecificEntries[0];
isResourceTenantSpecific = true;
}else {
callback(self._log.createError('More than one token matches the criteria. The result is ambiguous.'));
return;
}
}
if (returnVal) {
self._log.verbose('Returning token from cache lookup');
self._log.verbose('Returning token from cache lookup, ' + createTokenIdMessage(returnVal), true);
}
callback(null, returnVal, isResourceTenantSpecific);
});
};
/**
* The response from a token refresh request never contains an id_token and therefore no
* userInfo can be created from the response. This function creates a new cache entry
* combining the id_token based info and cache metadata from the cache entry that was refreshed with the
* new tokens in the refresh response.
* @param {object} entry A cache entry corresponding to the resfreshResponse.
* @param {object} refreshResponse The response from a token refresh request for the entry parameter.
* @return {object} A new cache entry.
*/
CacheDriver.prototype._createEntryFromRefresh = function(entry, refreshResponse) {
var newEntry = _.clone(entry);
newEntry = _.extend(newEntry, refreshResponse);
if (entry.isMRRT && this._authority !== entry[METADATA_AUTHORITY]) {
newEntry[METADATA_AUTHORITY] = this._authority;
}
this._log.verbose('Created new cache entry from refresh response.');
return newEntry;
};
CacheDriver.prototype._replaceEntry = function(entryToReplace, newEntry, callback) {
var self = this;
this.remove(entryToReplace, function(err) {
if (err) {
callback(err);
return;
}
self.add(newEntry, callback);
});
};
/**
* Given an expired cache entry refreshes it and updates the cache.
* @param {object} entry A cache entry with an MRRT to refresh for another resource.
* @param {SingleEntryCallback} callback
*/
CacheDriver.prototype._refreshExpiredEntry = function(entry, callback) {
var self = this;
this._refreshFunction(entry, null, function(err, tokenResponse) {
if (err) {
callback(err);
return;
}
var newEntry = self._createEntryFromRefresh(entry, tokenResponse);
self._replaceEntry(entry, newEntry, function(err) {
if (err) {
self._log.error('error refreshing expired token', err, true);
} else {
self._log.info('Returning token refreshed after expiry.');
}
callback(err, newEntry);
});
});
};
/**
* Given a cache entry with an MRRT will acquire a new token for a new resource via the MRRT, and cache it.
* @param {object} entry A cache entry with an MRRT to refresh for another resource.
* @param {SingleEntryCallback} callback
*/
CacheDriver.prototype._acquireNewTokenFromMrrt = function(entry, callback) {
var self = this;
this._refreshFunction(entry, this._resource, function(err, tokenResponse) {
if (err) {
callback(err);
return;
}
var newEntry = self._createEntryFromRefresh(entry, tokenResponse);
self.add(newEntry, function(err) {
if (err) {
self._log.error('error refreshing mrrt', err, true);
} else {
self._log.info('Returning token derived from mrrt refresh.');
}
callback(err, newEntry);
});
});
};
/**
* Given a token this function will refresh it if it is either expired, or an MRRT.
* @param {object} entry A cache entry to refresh if necessary.
* @param {Boolean} isResourceSpecific Indicates whether this token is appropriate for the resource for which
* it was requested or whether it is possibly an MRRT token for which
* a resource specific access token should be acquired.
* @param {SingleEntryCallback} callback
*/
CacheDriver.prototype._refreshEntryIfNecessary = function(entry, isResourceSpecific, callback) {
var expiryDate = entry[TokenResponseFields.EXPIRES_ON];
// Add some buffer in to the time comparison to account for clock skew or latency.
var nowPlusBuffer = (new Date()).addMinutes(constants.Misc.CLOCK_BUFFER);
if (isResourceSpecific && nowPlusBuffer.isAfter(expiryDate)) {
this._log.info('Cached token is expired. Refreshing: ' + expiryDate);
this._refreshExpiredEntry(entry, callback);
return;
} else if (!isResourceSpecific && entry.isMRRT) {
this._log.info('Acquiring new access token from MRRT token.');
this._acquireNewTokenFromMrrt(entry, callback);
return;
} else {
callback(null, entry);
}
};
/**
* Finds a single entry in the cache that matches the query or fails if more than one match is found.
* @param {object} query A query object
* @param {SingleEntryCallback} callback
*/
CacheDriver.prototype.find = function(query, callback) {
var self = this;
query = query || {};
this._log.verbose('finding using query');
this._log.verbose('finding with query:' + JSON.stringify(query), true);
this._loadSingleEntryFromCache(query, function(err, entry, isResourceTenantSpecific) {
if (err) {
callback(err);
return;
}
if (!entry) {
callback();
return;
}
self._refreshEntryIfNecessary(entry, isResourceTenantSpecific, function(err, newEntry) {
callback(err, newEntry);
return;
});
});
};
/**
* Removes a single entry from the cache.
* @param {object} entry The entry to remove.
* @param {Function} callback Called on completion. The first parameter may contain an error.
*/
CacheDriver.prototype.remove = function(entry, callback) {
this._log.verbose('Removing entry.');
return this._cache.remove([entry], function(err) {
callback(err);
return;
});
};
/**
* Removes a collection of entries from the cache in a single batch operation.
* @param {Array} entries An array of cache entries to remove.
* @param {Function} callback This function is called when the operation is complete. Any error is provided as the
* first parameter.
*/
CacheDriver.prototype._removeMany = function(entries, callback) {
this._log.verbose('Remove many: ' + entries.length);
this._cache.remove(entries, function(err) {
callback(err);
return;
});
};
/**
* Adds a collection of entries to the cache in a single batch operation.
* @param {Array} entries An array of entries to add to the cache.
* @param {Function} callback This function is called when the operation is complete. Any error is provided as the
* first parameter.
*/
CacheDriver.prototype._addMany = function(entries, callback) {
this._log.verbose('Add many: ' + entries.length);
this._cache.add(entries, function(err) {
callback(err);
return;
});
};
/*
* Tests whether the passed entry is a multi resource refresh token.
* Somewhat mysteriously the presense of a resource field in a returned
* token response indicates that the response is an MRRT.
* @param {object} entry
* @return {Boolean} true if the entry is an MRRT.
*/
function isMRRT(entry) {
return entry.resource ? true : false;
}
/**
* Given an cache entry this function finds all of the MRRT tokens already in the cache
* and updates them with the refresh_token of the passed in entry.
* @param {object} entry The entry from which to get an updated refresh_token
* @param {Function} callback Called back on completion. The first parameter may contain an error.
*/
CacheDriver.prototype._updateRefreshTokens = function(entry, callback) {
var self = this;
if (isMRRT(entry)) {
this._findMRRTTokensForUser(entry.userId, function(err, mrrtTokens) {
if (err) {
callback(err);
return;
}
if (!mrrtTokens || 0 === mrrtTokens.length) {
callback();
return;
}
self._log.verbose('Updating ' + mrrtTokens.length + ' cached refresh tokens.');
self._removeMany(mrrtTokens, function(err) {
if (err) {
callback(err);
return;
}
for (var i = 0; i < mrrtTokens.length; i++) {
mrrtTokens[i][TokenResponseFields.REFRESH_TOKEN] = entry[TokenResponseFields.REFRESH_TOKEN];
}
self._addMany(mrrtTokens, function(err) {
callback(err);
return;
});
});
});
} else {
callback();
return;
}
};
/**
* Checks to see if the entry has cache metadata already. If it does
* then it probably came from a refresh operation and the metadata
* was copied from the originating entry.
* @param {object} entry The entry to check
* @return {bool} Returns true if the entry has already been augmented
* with cache metadata.
*/
CacheDriver.prototype._entryHasMetadata = function(entry) {
return (_.has(entry, METADATA_CLIENTID) && _.has(entry, METADATA_AUTHORITY));
};
CacheDriver.prototype._augmentEntryWithCacheMetadata = function(entry) {
if (this._entryHasMetadata(entry)) {
return;
}
if (isMRRT(entry)) {
this._log.verbose('Added entry is MRRT');
entry.isMRRT = true;
} else {
entry.resource = this._resource;
}
entry[METADATA_CLIENTID] = this._clientId;
entry[METADATA_AUTHORITY] = this._authority;
};
/**
* Adds a single entry to the cache.
* @param {object} entry The entry to add.
* @param {string} clientId The id of this client app.
* @param {string} resource The id of the resource for which the cached token was obtained.
* @param {Function} callback Called back on completion. The first parameter may contain an error.
*/
CacheDriver.prototype.add = function(entry, callback) {
var self = this;
this._log.verbose('Adding entry');
this._log.verbose('Adding entry, ' + createTokenIdMessage(entry));
this._augmentEntryWithCacheMetadata(entry);
this._updateRefreshTokens(entry, function(err) {
if (err) {
callback(err);
return;
}
self._cache.add([entry], function(err) {
callback(err);
return;
});
});
};
module.exports = CacheDriver;