KEYCLOAK-3262 Add support to save multiple state in keycloak.js

This commit is contained in:
Stian Thorgersen 2016-07-04 20:58:15 +02:00
parent 435cdb6180
commit f52504ac41

View file

@ -25,7 +25,7 @@
var kc = this; var kc = this;
var adapter; var adapter;
var refreshQueue = []; var refreshQueue = [];
var storage; var callbackStorage;
var loginIframe = { var loginIframe = {
enable: true, enable: true,
@ -36,7 +36,7 @@
kc.init = function (initOptions) { kc.init = function (initOptions) {
kc.authenticated = false; kc.authenticated = false;
storage = new PersistentStorage(); callbackStorage = createCallbackStorage();
if (initOptions && initOptions.adapter === 'cordova') { if (initOptions && initOptions.adapter === 'cordova') {
adapter = loadAdapter('cordova'); adapter = loadAdapter('cordova');
@ -201,7 +201,7 @@
redirectUri += (redirectUri.indexOf('?') == -1 ? '?' : '&') + 'prompt=' + options.prompt; redirectUri += (redirectUri.indexOf('?') == -1 ? '?' : '&') + 'prompt=' + options.prompt;
} }
storage.setItem('oauthState', JSON.stringify({ state: state, nonce: nonce, redirectUri: encodeURIComponent(redirectUri) })); callbackStorage.add({ state: state, nonce: nonce, redirectUri: encodeURIComponent(redirectUri) });
var action = 'auth'; var action = 'auth';
if (options && options.action == 'register') { if (options && options.action == 'register') {
@ -697,15 +697,11 @@
function parseCallback(url) { function parseCallback(url) {
var oauth = new CallbackParser(url, kc.responseMode).parseUri(); var oauth = new CallbackParser(url, kc.responseMode).parseUri();
var oauthState = callbackStorage.get(oauth.state);
var oauthState = storage.getItem('oauthState'); if (oauthState && (oauth.code || oauth.error || oauth.access_token || oauth.id_token)) {
var sessionState = oauthState && JSON.parse(oauthState); oauth.redirectUri = oauthState.redirectUri;
oauth.storedNonce = oauthState.nonce;
if (sessionState && (oauth.code || oauth.error || oauth.access_token || oauth.id_token) && oauth.state && oauth.state == sessionState.state) {
storage.removeItem('oauthState');
oauth.redirectUri = sessionState.redirectUri;
oauth.storedNonce = sessionState.nonce;
if (oauth.fragment) { if (oauth.fragment) {
oauth.newUrl += '#' + oauth.fragment; oauth.newUrl += '#' + oauth.fragment;
@ -996,60 +992,93 @@
throw 'invalid adapter type: ' + type; throw 'invalid adapter type: ' + type;
} }
var LocalStorage = function() {
if (!(this instanceof LocalStorage)) {
return new LocalStorage();
}
var PersistentStorage = function() { localStorage.setItem('kc-test', 'test');
if (!(this instanceof PersistentStorage)) { localStorage.removeItem('kc-test');
return new PersistentStorage();
} var cs = this;
var ps = this;
var useCookieStorage = function () { function clearExpired() {
if (typeof localStorage === "undefined") { var time = new Date().getTime();
return true; for (var i = 1; i <= localStorage.length; i++) {
} var key = localStorage.key(i);
if (key && key.indexOf('kc-callback-') == 0) {
var value = localStorage.getItem(key);
if (value) {
try { try {
var key = '@@keycloak-session-storage/test'; var expires = JSON.parse(value).expires;
localStorage.setItem(key, 'test'); if (!expires || expires < time) {
localStorage.removeItem(key); localStorage.removeItem(key);
return false; }
} catch (err) { } catch (err) {
// Probably in Safari "private mode" where localStorage
// quota is 0, or quota exceeded. Switching to cookie
// storage.
return true;
}
}
ps.setItem = function(key, value) {
if (useCookieStorage()) {
setCookie(key, value, cookieExpiration(5));
} else {
localStorage.setItem(key, value);
}
}
ps.getItem = function(key) {
if (useCookieStorage()) {
return getCookie(key);
}
return localStorage.getItem(key);
}
ps.removeItem = function(key) {
if (typeof localStorage !== "undefined") {
try {
// Always try to delete from localStorage.
localStorage.removeItem(key); localStorage.removeItem(key);
} catch (err) { }
} }
// Always remove the cookie. }
}
}
}
cs.get = function(state) {
if (!state) {
return;
}
var key = 'kc-callback-' + state;
var value = localStorage.getItem(key);
if (value) {
localStorage.removeItem(key);
value = JSON.parse(value);
}
clearExpired();
return value;
};
cs.add = function(state) {
clearExpired();
var key = 'kc-callback-' + state.state;
state.expires = new Date().getTime() + (60 * 60 * 1000);
localStorage.setItem(key, JSON.stringify(state));
};
};
var CookieStorage = function() {
if (!(this instanceof CookieStorage)) {
return new CookieStorage();
}
var cs = this;
cs.get = function(state) {
if (!state) {
return;
}
var value = getCookie('kc-callback-' + state);
setCookie('kc-callback-' + state, '', cookieExpiration(-100));
if (value) {
return JSON.parse(value);
}
};
cs.add = function(state) {
setCookie('kc-callback-' + state.state, JSON.stringify(state), cookieExpiration(60));
};
cs.removeItem = function(key) {
setCookie(key, '', cookieExpiration(-100)); setCookie(key, '', cookieExpiration(-100));
} };
var cookieExpiration = function (minutes) { var cookieExpiration = function (minutes) {
var exp = new Date(); var exp = new Date();
exp.setTime(exp.getTime() + (minutes*60*1000)); exp.setTime(exp.getTime() + (minutes*60*1000));
return exp; return exp;
} };
var getCookie = function (key) { var getCookie = function (key) {
var name = key + '='; var name = key + '=';
@ -1064,13 +1093,22 @@
} }
} }
return ''; return '';
} };
var setCookie = function (key, value, expirationDate) { var setCookie = function (key, value, expirationDate) {
var cookie = key + '=' + value + '; ' var cookie = key + '=' + value + '; '
+ 'expires=' + expirationDate.toUTCString() + '; '; + 'expires=' + expirationDate.toUTCString() + '; ';
document.cookie = cookie; document.cookie = cookie;
} }
};
function createCallbackStorage() {
try {
return new LocalStorage();
} catch (err) {
}
return new CookieStorage();
} }
var CallbackParser = function(uriToParse, responseMode) { var CallbackParser = function(uriToParse, responseMode) {