2016-02-03 10:20:22 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2016 Red Hat, Inc. and/or its affiliates
|
|
|
|
* and other contributors as indicated by the @author tags.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
(function( window, undefined ) {
|
|
|
|
|
|
|
|
var Keycloak = function (config) {
|
|
|
|
if (!(this instanceof Keycloak)) {
|
|
|
|
return new Keycloak(config);
|
|
|
|
}
|
|
|
|
|
|
|
|
var kc = this;
|
|
|
|
var adapter;
|
|
|
|
var refreshQueue = [];
|
2016-07-04 18:58:15 +00:00
|
|
|
var callbackStorage;
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
var loginIframe = {
|
|
|
|
enable: true,
|
2016-10-17 12:05:36 +00:00
|
|
|
callbackList: [],
|
2015-07-17 11:45:43 +00:00
|
|
|
interval: 5
|
|
|
|
};
|
|
|
|
|
|
|
|
kc.init = function (initOptions) {
|
|
|
|
kc.authenticated = false;
|
2016-04-15 12:09:07 +00:00
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
callbackStorage = createCallbackStorage();
|
2016-04-15 12:09:07 +00:00
|
|
|
|
2016-04-14 10:00:41 +00:00
|
|
|
if (initOptions && initOptions.adapter === 'cordova') {
|
2016-07-04 11:07:17 +00:00
|
|
|
adapter = loadAdapter('cordova');
|
2016-04-14 09:59:34 +00:00
|
|
|
} else if (initOptions && initOptions.adapter === 'default') {
|
2015-07-17 11:45:43 +00:00
|
|
|
adapter = loadAdapter();
|
2016-04-14 09:59:34 +00:00
|
|
|
} else {
|
|
|
|
if (window.Cordova) {
|
|
|
|
adapter = loadAdapter('cordova');
|
|
|
|
} else {
|
|
|
|
adapter = loadAdapter();
|
|
|
|
}
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
2016-04-15 12:09:07 +00:00
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
if (initOptions) {
|
|
|
|
if (typeof initOptions.checkLoginIframe !== 'undefined') {
|
|
|
|
loginIframe.enable = initOptions.checkLoginIframe;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (initOptions.checkLoginIframeInterval) {
|
|
|
|
loginIframe.interval = initOptions.checkLoginIframeInterval;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (initOptions.onLoad === 'login-required') {
|
|
|
|
kc.loginRequired = true;
|
|
|
|
}
|
2015-11-24 13:26:41 +00:00
|
|
|
|
|
|
|
if (initOptions.responseMode) {
|
|
|
|
if (initOptions.responseMode === 'query' || initOptions.responseMode === 'fragment') {
|
|
|
|
kc.responseMode = initOptions.responseMode;
|
|
|
|
} else {
|
|
|
|
throw 'Invalid value for responseMode';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (initOptions.flow) {
|
|
|
|
switch (initOptions.flow) {
|
|
|
|
case 'standard':
|
|
|
|
kc.responseType = 'code';
|
|
|
|
break;
|
|
|
|
case 'implicit':
|
2015-11-27 07:29:50 +00:00
|
|
|
kc.responseType = 'id_token token';
|
2015-11-24 13:26:41 +00:00
|
|
|
break;
|
|
|
|
case 'hybrid':
|
2015-11-27 07:29:50 +00:00
|
|
|
kc.responseType = 'code id_token token';
|
2015-11-24 13:26:41 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw 'Invalid value for flow';
|
|
|
|
}
|
2015-11-27 07:29:50 +00:00
|
|
|
kc.flow = initOptions.flow;
|
2015-11-24 13:26:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!kc.responseMode) {
|
|
|
|
kc.responseMode = 'fragment';
|
|
|
|
}
|
|
|
|
if (!kc.responseType) {
|
|
|
|
kc.responseType = 'code';
|
2015-11-27 07:29:50 +00:00
|
|
|
kc.flow = 'standard';
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var promise = createPromise();
|
|
|
|
|
|
|
|
var initPromise = createPromise();
|
|
|
|
initPromise.promise.success(function() {
|
|
|
|
kc.onReady && kc.onReady(kc.authenticated);
|
|
|
|
promise.setSuccess(kc.authenticated);
|
2016-07-14 21:55:27 +00:00
|
|
|
}).error(function(errorData) {
|
|
|
|
promise.setError(errorData);
|
2015-07-17 11:45:43 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
var configPromise = loadConfig(config);
|
|
|
|
|
|
|
|
function onLoad() {
|
|
|
|
var doLogin = function(prompt) {
|
|
|
|
if (!prompt) {
|
|
|
|
options.prompt = 'none';
|
|
|
|
}
|
|
|
|
kc.login(options).success(function () {
|
|
|
|
initPromise.setSuccess();
|
|
|
|
}).error(function () {
|
|
|
|
initPromise.setError();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
var options = {};
|
|
|
|
switch (initOptions.onLoad) {
|
|
|
|
case 'check-sso':
|
|
|
|
if (loginIframe.enable) {
|
|
|
|
setupCheckLoginIframe().success(function() {
|
|
|
|
checkLoginIframe().success(function () {
|
|
|
|
doLogin(false);
|
|
|
|
}).error(function () {
|
|
|
|
initPromise.setSuccess();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
doLogin(false);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'login-required':
|
|
|
|
doLogin(true);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw 'Invalid value for onLoad';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function processInit() {
|
|
|
|
var callback = parseCallback(window.location.href);
|
|
|
|
|
|
|
|
if (callback) {
|
|
|
|
setupCheckLoginIframe();
|
|
|
|
window.history.replaceState({}, null, callback.newUrl);
|
|
|
|
processCallback(callback, initPromise);
|
|
|
|
return;
|
|
|
|
} else if (initOptions) {
|
|
|
|
if (initOptions.token || initOptions.refreshToken) {
|
2016-09-06 11:00:37 +00:00
|
|
|
setToken(initOptions.token, initOptions.refreshToken, initOptions.idToken);
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
if (loginIframe.enable) {
|
|
|
|
setupCheckLoginIframe().success(function() {
|
|
|
|
checkLoginIframe().success(function () {
|
2016-09-06 06:59:58 +00:00
|
|
|
kc.onAuthSuccess && kc.onAuthSuccess();
|
2015-07-17 11:45:43 +00:00
|
|
|
initPromise.setSuccess();
|
|
|
|
}).error(function () {
|
2016-09-06 06:59:58 +00:00
|
|
|
kc.onAuthError && kc.onAuthError();
|
2015-07-17 11:45:43 +00:00
|
|
|
if (initOptions.onLoad) {
|
|
|
|
onLoad();
|
2016-09-15 21:25:29 +00:00
|
|
|
} else {
|
|
|
|
initPromise.setError();
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} else {
|
2016-09-06 06:59:58 +00:00
|
|
|
kc.updateToken(-1).success(function() {
|
|
|
|
kc.onAuthSuccess && kc.onAuthSuccess();
|
|
|
|
initPromise.setSuccess();
|
|
|
|
}).error(function() {
|
|
|
|
kc.onAuthError && kc.onAuthError();
|
|
|
|
if (initOptions.onLoad) {
|
|
|
|
onLoad();
|
2016-09-15 21:25:29 +00:00
|
|
|
} else {
|
|
|
|
initPromise.setError();
|
2016-09-06 06:59:58 +00:00
|
|
|
}
|
|
|
|
});
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
} else if (initOptions.onLoad) {
|
|
|
|
onLoad();
|
2016-03-18 11:06:28 +00:00
|
|
|
} else {
|
|
|
|
initPromise.setSuccess();
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
initPromise.setSuccess();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
configPromise.success(processInit);
|
|
|
|
configPromise.error(function() {
|
|
|
|
promise.setError();
|
|
|
|
});
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.login = function (options) {
|
|
|
|
return adapter.login(options);
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.createLoginUrl = function(options) {
|
|
|
|
var state = createUUID();
|
2015-11-24 13:26:41 +00:00
|
|
|
var nonce = createUUID();
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
var redirectUri = adapter.redirectUri(options);
|
|
|
|
if (options && options.prompt) {
|
|
|
|
redirectUri += (redirectUri.indexOf('?') == -1 ? '?' : '&') + 'prompt=' + options.prompt;
|
|
|
|
}
|
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
callbackStorage.add({ state: state, nonce: nonce, redirectUri: encodeURIComponent(redirectUri) });
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
var action = 'auth';
|
|
|
|
if (options && options.action == 'register') {
|
|
|
|
action = 'registrations';
|
|
|
|
}
|
|
|
|
|
2016-07-14 17:31:07 +00:00
|
|
|
var scope = (options && options.scope) ? "openid " + options.scope : "openid";
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
var url = getRealmUrl()
|
|
|
|
+ '/protocol/openid-connect/' + action
|
|
|
|
+ '?client_id=' + encodeURIComponent(kc.clientId)
|
|
|
|
+ '&redirect_uri=' + encodeURIComponent(redirectUri)
|
|
|
|
+ '&state=' + encodeURIComponent(state)
|
2015-11-24 13:26:41 +00:00
|
|
|
+ '&nonce=' + encodeURIComponent(nonce)
|
|
|
|
+ '&response_mode=' + encodeURIComponent(kc.responseMode)
|
2016-07-14 17:31:07 +00:00
|
|
|
+ '&response_type=' + encodeURIComponent(kc.responseType)
|
|
|
|
+ '&scope=' + encodeURIComponent(scope);
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
if (options && options.prompt) {
|
2015-11-09 15:27:33 +00:00
|
|
|
url += '&prompt=' + encodeURIComponent(options.prompt);
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
|
2016-07-21 16:18:05 +00:00
|
|
|
if (options && options.maxAge) {
|
|
|
|
url += '&max_age=' + encodeURIComponent(options.maxAge);
|
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
if (options && options.loginHint) {
|
2015-11-09 15:27:33 +00:00
|
|
|
url += '&login_hint=' + encodeURIComponent(options.loginHint);
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (options && options.idpHint) {
|
2015-11-09 15:27:33 +00:00
|
|
|
url += '&kc_idp_hint=' + encodeURIComponent(options.idpHint);
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
|
2015-10-23 10:23:24 +00:00
|
|
|
if (options && options.locale) {
|
2015-11-09 15:27:33 +00:00
|
|
|
url += '&ui_locales=' + encodeURIComponent(options.locale);
|
2015-10-23 10:23:24 +00:00
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
return url;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.logout = function(options) {
|
|
|
|
return adapter.logout(options);
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.createLogoutUrl = function(options) {
|
|
|
|
var url = getRealmUrl()
|
|
|
|
+ '/protocol/openid-connect/logout'
|
2016-02-06 16:43:38 +00:00
|
|
|
+ '?redirect_uri=' + encodeURIComponent(adapter.redirectUri(options, false));
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
return url;
|
|
|
|
}
|
|
|
|
|
2015-10-07 11:53:25 +00:00
|
|
|
kc.register = function (options) {
|
|
|
|
return adapter.register(options);
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.createRegisterUrl = function(options) {
|
|
|
|
if (!options) {
|
|
|
|
options = {};
|
|
|
|
}
|
|
|
|
options.action = 'register';
|
|
|
|
return kc.createLoginUrl(options);
|
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
kc.createAccountUrl = function(options) {
|
|
|
|
var url = getRealmUrl()
|
|
|
|
+ '/account'
|
|
|
|
+ '?referrer=' + encodeURIComponent(kc.clientId)
|
|
|
|
+ '&referrer_uri=' + encodeURIComponent(adapter.redirectUri(options));
|
|
|
|
|
|
|
|
return url;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.accountManagement = function() {
|
|
|
|
return adapter.accountManagement();
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.hasRealmRole = function (role) {
|
|
|
|
var access = kc.realmAccess;
|
|
|
|
return !!access && access.roles.indexOf(role) >= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.hasResourceRole = function(role, resource) {
|
|
|
|
if (!kc.resourceAccess) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
var access = kc.resourceAccess[resource || kc.clientId];
|
|
|
|
return !!access && access.roles.indexOf(role) >= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.loadUserProfile = function() {
|
|
|
|
var url = getRealmUrl() + '/account';
|
|
|
|
var req = new XMLHttpRequest();
|
|
|
|
req.open('GET', url, true);
|
|
|
|
req.setRequestHeader('Accept', 'application/json');
|
|
|
|
req.setRequestHeader('Authorization', 'bearer ' + kc.token);
|
|
|
|
|
|
|
|
var promise = createPromise();
|
|
|
|
|
|
|
|
req.onreadystatechange = function () {
|
|
|
|
if (req.readyState == 4) {
|
|
|
|
if (req.status == 200) {
|
|
|
|
kc.profile = JSON.parse(req.responseText);
|
|
|
|
promise.setSuccess(kc.profile);
|
|
|
|
} else {
|
|
|
|
promise.setError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
req.send();
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.loadUserInfo = function() {
|
|
|
|
var url = getRealmUrl() + '/protocol/openid-connect/userinfo';
|
|
|
|
var req = new XMLHttpRequest();
|
|
|
|
req.open('GET', url, true);
|
|
|
|
req.setRequestHeader('Accept', 'application/json');
|
|
|
|
req.setRequestHeader('Authorization', 'bearer ' + kc.token);
|
|
|
|
|
|
|
|
var promise = createPromise();
|
|
|
|
|
|
|
|
req.onreadystatechange = function () {
|
|
|
|
if (req.readyState == 4) {
|
|
|
|
if (req.status == 200) {
|
|
|
|
kc.userInfo = JSON.parse(req.responseText);
|
|
|
|
promise.setSuccess(kc.userInfo);
|
|
|
|
} else {
|
|
|
|
promise.setError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
req.send();
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.isTokenExpired = function(minValidity) {
|
2015-11-27 07:29:50 +00:00
|
|
|
if (!kc.tokenParsed || (!kc.refreshToken && kc.flow != 'implicit' )) {
|
2015-07-17 11:45:43 +00:00
|
|
|
throw 'Not authenticated';
|
|
|
|
}
|
|
|
|
|
2016-09-21 07:13:05 +00:00
|
|
|
var expiresIn = kc.tokenParsed['exp'] - Math.ceil(new Date().getTime() / 1000) + kc.timeSkew;
|
2015-07-17 11:45:43 +00:00
|
|
|
if (minValidity) {
|
|
|
|
expiresIn -= minValidity;
|
|
|
|
}
|
|
|
|
return expiresIn < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.updateToken = function(minValidity) {
|
|
|
|
var promise = createPromise();
|
|
|
|
|
|
|
|
if (!kc.tokenParsed || !kc.refreshToken) {
|
|
|
|
promise.setError();
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
minValidity = minValidity || 5;
|
|
|
|
|
|
|
|
var exec = function() {
|
2016-09-21 07:13:05 +00:00
|
|
|
var refreshToken = false;
|
|
|
|
if (kc.timeSkew == -1) {
|
|
|
|
console.info('Skew ' + kc.timeSkew);
|
|
|
|
refreshToken = true;
|
|
|
|
console.info('[KEYCLOAK] Refreshing token: time skew not set');
|
|
|
|
} else if (minValidity == -1) {
|
|
|
|
refreshToken = true;
|
|
|
|
console.info('[KEYCLOAK] Refreshing token: forced refresh');
|
|
|
|
} else if (kc.isTokenExpired(minValidity)) {
|
|
|
|
refreshToken = true;
|
|
|
|
console.info('[KEYCLOAK] Refreshing token: token expired');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!refreshToken) {
|
2015-07-17 11:45:43 +00:00
|
|
|
promise.setSuccess(false);
|
|
|
|
} else {
|
|
|
|
var params = 'grant_type=refresh_token&' + 'refresh_token=' + kc.refreshToken;
|
|
|
|
var url = getRealmUrl() + '/protocol/openid-connect/token';
|
|
|
|
|
|
|
|
refreshQueue.push(promise);
|
|
|
|
|
|
|
|
if (refreshQueue.length == 1) {
|
|
|
|
var req = new XMLHttpRequest();
|
|
|
|
req.open('POST', url, true);
|
|
|
|
req.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
|
2016-08-26 15:23:44 +00:00
|
|
|
req.withCredentials = true;
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
if (kc.clientId && kc.clientSecret) {
|
|
|
|
req.setRequestHeader('Authorization', 'Basic ' + btoa(kc.clientId + ':' + kc.clientSecret));
|
|
|
|
} else {
|
|
|
|
params += '&client_id=' + encodeURIComponent(kc.clientId);
|
|
|
|
}
|
|
|
|
|
2015-08-20 11:29:32 +00:00
|
|
|
var timeLocal = new Date().getTime();
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
req.onreadystatechange = function () {
|
|
|
|
if (req.readyState == 4) {
|
|
|
|
if (req.status == 200) {
|
2016-09-21 07:13:05 +00:00
|
|
|
console.info('[KEYCLOAK] Token refreshed');
|
|
|
|
|
2015-08-20 11:29:32 +00:00
|
|
|
timeLocal = (timeLocal + new Date().getTime()) / 2;
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
var tokenResponse = JSON.parse(req.responseText);
|
2015-08-20 11:29:32 +00:00
|
|
|
|
2016-09-21 07:13:05 +00:00
|
|
|
setToken(tokenResponse['access_token'], tokenResponse['refresh_token'], tokenResponse['id_token'], timeLocal);
|
2016-09-06 11:00:37 +00:00
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
kc.onAuthRefreshSuccess && kc.onAuthRefreshSuccess();
|
|
|
|
for (var p = refreshQueue.pop(); p != null; p = refreshQueue.pop()) {
|
|
|
|
p.setSuccess(true);
|
|
|
|
}
|
|
|
|
} else {
|
2016-09-21 07:13:05 +00:00
|
|
|
console.warn('[KEYCLOAK] Failed to refresh token');
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
kc.onAuthRefreshError && kc.onAuthRefreshError();
|
|
|
|
for (var p = refreshQueue.pop(); p != null; p = refreshQueue.pop()) {
|
|
|
|
p.setError(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
req.send(params);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (loginIframe.enable) {
|
|
|
|
var iframePromise = checkLoginIframe();
|
|
|
|
iframePromise.success(function() {
|
|
|
|
exec();
|
|
|
|
}).error(function() {
|
|
|
|
promise.setError();
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.clearToken = function() {
|
|
|
|
if (kc.token) {
|
2016-09-06 11:00:37 +00:00
|
|
|
setToken(null, null, null);
|
2015-07-17 11:45:43 +00:00
|
|
|
kc.onAuthLogout && kc.onAuthLogout();
|
|
|
|
if (kc.loginRequired) {
|
|
|
|
kc.login();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getRealmUrl() {
|
|
|
|
if (kc.authServerUrl.charAt(kc.authServerUrl.length - 1) == '/') {
|
|
|
|
return kc.authServerUrl + 'realms/' + encodeURIComponent(kc.realm);
|
|
|
|
} else {
|
|
|
|
return kc.authServerUrl + '/realms/' + encodeURIComponent(kc.realm);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getOrigin() {
|
|
|
|
if (!window.location.origin) {
|
|
|
|
return window.location.protocol + "//" + window.location.hostname + (window.location.port ? ':' + window.location.port: '');
|
|
|
|
} else {
|
|
|
|
return window.location.origin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function processCallback(oauth, promise) {
|
|
|
|
var code = oauth.code;
|
|
|
|
var error = oauth.error;
|
|
|
|
var prompt = oauth.prompt;
|
|
|
|
|
2015-11-24 13:26:41 +00:00
|
|
|
var timeLocal = new Date().getTime();
|
|
|
|
|
2015-11-27 07:29:50 +00:00
|
|
|
if (error) {
|
|
|
|
if (prompt != 'none') {
|
2016-07-14 21:55:27 +00:00
|
|
|
var errorData = { error: error, error_description: oauth.error_description };
|
|
|
|
kc.onAuthError && kc.onAuthError(errorData);
|
|
|
|
promise && promise.setError(errorData);
|
2015-11-27 07:29:50 +00:00
|
|
|
} else {
|
|
|
|
promise && promise.setSuccess();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
} else if ((kc.flow != 'standard') && (oauth.access_token || oauth.id_token)) {
|
|
|
|
authSuccess(oauth.access_token, null, oauth.id_token, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((kc.flow != 'implicit') && code) {
|
2015-07-17 11:45:43 +00:00
|
|
|
var params = 'code=' + code + '&grant_type=authorization_code';
|
|
|
|
var url = getRealmUrl() + '/protocol/openid-connect/token';
|
|
|
|
|
|
|
|
var req = new XMLHttpRequest();
|
|
|
|
req.open('POST', url, true);
|
|
|
|
req.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
|
|
|
|
|
|
|
|
if (kc.clientId && kc.clientSecret) {
|
|
|
|
req.setRequestHeader('Authorization', 'Basic ' + btoa(kc.clientId + ':' + kc.clientSecret));
|
|
|
|
} else {
|
|
|
|
params += '&client_id=' + encodeURIComponent(kc.clientId);
|
|
|
|
}
|
|
|
|
|
|
|
|
params += '&redirect_uri=' + oauth.redirectUri;
|
|
|
|
|
|
|
|
req.withCredentials = true;
|
|
|
|
|
|
|
|
req.onreadystatechange = function() {
|
|
|
|
if (req.readyState == 4) {
|
|
|
|
if (req.status == 200) {
|
2015-08-20 11:29:32 +00:00
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
var tokenResponse = JSON.parse(req.responseText);
|
2015-11-27 07:29:50 +00:00
|
|
|
authSuccess(tokenResponse['access_token'], tokenResponse['refresh_token'], tokenResponse['id_token'], kc.flow === 'standard');
|
2015-07-17 11:45:43 +00:00
|
|
|
} else {
|
|
|
|
kc.onAuthError && kc.onAuthError();
|
|
|
|
promise && promise.setError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
req.send(params);
|
|
|
|
}
|
2015-11-24 13:26:41 +00:00
|
|
|
|
2015-11-27 07:29:50 +00:00
|
|
|
function authSuccess(accessToken, refreshToken, idToken, fulfillPromise) {
|
2015-11-24 13:26:41 +00:00
|
|
|
timeLocal = (timeLocal + new Date().getTime()) / 2;
|
|
|
|
|
2016-09-21 07:13:05 +00:00
|
|
|
setToken(accessToken, refreshToken, idToken, timeLocal);
|
2015-11-24 13:26:41 +00:00
|
|
|
|
|
|
|
if ((kc.tokenParsed && kc.tokenParsed.nonce != oauth.storedNonce) ||
|
|
|
|
(kc.refreshTokenParsed && kc.refreshTokenParsed.nonce != oauth.storedNonce) ||
|
|
|
|
(kc.idTokenParsed && kc.idTokenParsed.nonce != oauth.storedNonce)) {
|
|
|
|
|
2016-09-21 07:13:05 +00:00
|
|
|
console.info('[KEYCLOAK] Invalid nonce, clearing token');
|
2015-11-24 13:26:41 +00:00
|
|
|
kc.clearToken();
|
|
|
|
promise && promise.setError();
|
|
|
|
} else {
|
2015-11-27 07:29:50 +00:00
|
|
|
if (fulfillPromise) {
|
|
|
|
kc.onAuthSuccess && kc.onAuthSuccess();
|
|
|
|
promise && promise.setSuccess();
|
|
|
|
}
|
2015-11-24 13:26:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function loadConfig(url) {
|
|
|
|
var promise = createPromise();
|
|
|
|
var configUrl;
|
|
|
|
|
|
|
|
if (!config) {
|
|
|
|
configUrl = 'keycloak.json';
|
|
|
|
} else if (typeof config === 'string') {
|
|
|
|
configUrl = config;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (configUrl) {
|
|
|
|
var req = new XMLHttpRequest();
|
|
|
|
req.open('GET', configUrl, true);
|
|
|
|
req.setRequestHeader('Accept', 'application/json');
|
|
|
|
|
|
|
|
req.onreadystatechange = function () {
|
|
|
|
if (req.readyState == 4) {
|
|
|
|
if (req.status == 200) {
|
|
|
|
var config = JSON.parse(req.responseText);
|
|
|
|
|
|
|
|
kc.authServerUrl = config['auth-server-url'];
|
|
|
|
kc.realm = config['realm'];
|
|
|
|
kc.clientId = config['resource'];
|
|
|
|
kc.clientSecret = (config['credentials'] || {})['secret'];
|
|
|
|
|
|
|
|
promise.setSuccess();
|
|
|
|
} else {
|
|
|
|
promise.setError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
req.send();
|
|
|
|
} else {
|
|
|
|
if (!config['url']) {
|
|
|
|
var scripts = document.getElementsByTagName('script');
|
|
|
|
for (var i = 0; i < scripts.length; i++) {
|
|
|
|
if (scripts[i].src.match(/.*keycloak\.js/)) {
|
|
|
|
config.url = scripts[i].src.substr(0, scripts[i].src.indexOf('/js/keycloak.js'));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!config.realm) {
|
|
|
|
throw 'realm missing';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!config.clientId) {
|
|
|
|
throw 'clientId missing';
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.authServerUrl = config.url;
|
|
|
|
kc.realm = config.realm;
|
|
|
|
kc.clientId = config.clientId;
|
|
|
|
kc.clientSecret = (config.credentials || {}).secret;
|
|
|
|
|
|
|
|
promise.setSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
2016-09-21 07:13:05 +00:00
|
|
|
function setToken(token, refreshToken, idToken, timeLocal) {
|
2015-11-27 07:29:50 +00:00
|
|
|
if (kc.tokenTimeoutHandle) {
|
|
|
|
clearTimeout(kc.tokenTimeoutHandle);
|
|
|
|
kc.tokenTimeoutHandle = null;
|
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
if (token) {
|
|
|
|
kc.token = token;
|
|
|
|
kc.tokenParsed = decodeToken(token);
|
|
|
|
var sessionId = kc.realm + '/' + kc.tokenParsed.sub;
|
|
|
|
if (kc.tokenParsed.session_state) {
|
|
|
|
sessionId = sessionId + '/' + kc.tokenParsed.session_state;
|
|
|
|
}
|
|
|
|
kc.sessionId = sessionId;
|
|
|
|
kc.authenticated = true;
|
|
|
|
kc.subject = kc.tokenParsed.sub;
|
|
|
|
kc.realmAccess = kc.tokenParsed.realm_access;
|
|
|
|
kc.resourceAccess = kc.tokenParsed.resource_access;
|
2015-11-27 07:29:50 +00:00
|
|
|
|
2016-09-21 07:13:05 +00:00
|
|
|
if (timeLocal) {
|
|
|
|
kc.timeSkew = Math.floor(timeLocal / 1000) - kc.tokenParsed.iat;
|
|
|
|
console.info('[KEYCLOAK] Estimated time difference between browser and server is ' + kc.timeSkew + ' seconds');
|
|
|
|
} else {
|
|
|
|
kc.timeSkew = -1;
|
|
|
|
}
|
|
|
|
|
2015-11-27 07:29:50 +00:00
|
|
|
if (kc.onTokenExpired) {
|
2016-09-21 07:13:05 +00:00
|
|
|
if (kc.timeSkew == -1) {
|
2016-09-06 11:00:37 +00:00
|
|
|
kc.onTokenExpired();
|
|
|
|
} else {
|
2016-09-21 07:13:05 +00:00
|
|
|
var expiresIn = (kc.tokenParsed['exp'] - (new Date().getTime() / 1000) + kc.timeSkew) * 1000;
|
|
|
|
if (expiresIn <= 0) {
|
|
|
|
kc.onTokenExpired();
|
|
|
|
} else {
|
|
|
|
kc.tokenTimeoutHandle = setTimeout(kc.onTokenExpired, expiresIn);
|
|
|
|
}
|
2016-09-06 11:00:37 +00:00
|
|
|
}
|
2015-11-27 07:29:50 +00:00
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
} else {
|
|
|
|
delete kc.token;
|
|
|
|
delete kc.tokenParsed;
|
|
|
|
delete kc.subject;
|
|
|
|
delete kc.realmAccess;
|
|
|
|
delete kc.resourceAccess;
|
|
|
|
|
|
|
|
kc.authenticated = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (refreshToken) {
|
|
|
|
kc.refreshToken = refreshToken;
|
|
|
|
kc.refreshTokenParsed = decodeToken(refreshToken);
|
|
|
|
} else {
|
|
|
|
delete kc.refreshToken;
|
|
|
|
delete kc.refreshTokenParsed;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (idToken) {
|
|
|
|
kc.idToken = idToken;
|
|
|
|
kc.idTokenParsed = decodeToken(idToken);
|
|
|
|
} else {
|
|
|
|
delete kc.idToken;
|
|
|
|
delete kc.idTokenParsed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function decodeToken(str) {
|
|
|
|
str = str.split('.')[1];
|
|
|
|
|
|
|
|
str = str.replace('/-/g', '+');
|
|
|
|
str = str.replace('/_/g', '/');
|
|
|
|
switch (str.length % 4)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
str += '==';
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
str += '=';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw 'Invalid token';
|
|
|
|
}
|
|
|
|
|
|
|
|
str = (str + '===').slice(0, str.length + (str.length % 4));
|
|
|
|
str = str.replace(/-/g, '+').replace(/_/g, '/');
|
|
|
|
|
|
|
|
str = decodeURIComponent(escape(atob(str)));
|
|
|
|
|
|
|
|
str = JSON.parse(str);
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
function createUUID() {
|
|
|
|
var s = [];
|
|
|
|
var hexDigits = '0123456789abcdef';
|
|
|
|
for (var i = 0; i < 36; i++) {
|
|
|
|
s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1);
|
|
|
|
}
|
|
|
|
s[14] = '4';
|
|
|
|
s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1);
|
|
|
|
s[8] = s[13] = s[18] = s[23] = '-';
|
|
|
|
var uuid = s.join('');
|
|
|
|
return uuid;
|
|
|
|
}
|
|
|
|
|
|
|
|
kc.callback_id = 0;
|
|
|
|
|
|
|
|
function createCallbackId() {
|
|
|
|
var id = '<id: ' + (kc.callback_id++) + (Math.random()) + '>';
|
|
|
|
return id;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseCallback(url) {
|
2015-11-24 13:26:41 +00:00
|
|
|
var oauth = new CallbackParser(url, kc.responseMode).parseUri();
|
2016-07-04 18:58:15 +00:00
|
|
|
var oauthState = callbackStorage.get(oauth.state);
|
2015-07-17 11:45:43 +00:00
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
if (oauthState && (oauth.code || oauth.error || oauth.access_token || oauth.id_token)) {
|
|
|
|
oauth.redirectUri = oauthState.redirectUri;
|
|
|
|
oauth.storedNonce = oauthState.nonce;
|
2015-07-17 11:45:43 +00:00
|
|
|
|
2015-11-24 13:26:41 +00:00
|
|
|
if (oauth.fragment) {
|
|
|
|
oauth.newUrl += '#' + oauth.fragment;
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
2015-11-24 13:26:41 +00:00
|
|
|
|
|
|
|
return oauth;
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function createPromise() {
|
|
|
|
var p = {
|
|
|
|
setSuccess: function(result) {
|
|
|
|
p.success = true;
|
|
|
|
p.result = result;
|
|
|
|
if (p.successCallback) {
|
|
|
|
p.successCallback(result);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
setError: function(result) {
|
|
|
|
p.error = true;
|
|
|
|
p.result = result;
|
|
|
|
if (p.errorCallback) {
|
|
|
|
p.errorCallback(result);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
promise: {
|
|
|
|
success: function(callback) {
|
|
|
|
if (p.success) {
|
|
|
|
callback(p.result);
|
|
|
|
} else if (!p.error) {
|
|
|
|
p.successCallback = callback;
|
|
|
|
}
|
|
|
|
return p.promise;
|
|
|
|
},
|
|
|
|
error: function(callback) {
|
|
|
|
if (p.error) {
|
|
|
|
callback(p.result);
|
|
|
|
} else if (!p.success) {
|
|
|
|
p.errorCallback = callback;
|
|
|
|
}
|
|
|
|
return p.promise;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
function setupCheckLoginIframe() {
|
|
|
|
var promise = createPromise();
|
|
|
|
|
|
|
|
if (!loginIframe.enable) {
|
|
|
|
promise.setSuccess();
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (loginIframe.iframe) {
|
|
|
|
promise.setSuccess();
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
var iframe = document.createElement('iframe');
|
|
|
|
loginIframe.iframe = iframe;
|
|
|
|
|
|
|
|
iframe.onload = function() {
|
|
|
|
var realmUrl = getRealmUrl();
|
|
|
|
if (realmUrl.charAt(0) === '/') {
|
|
|
|
loginIframe.iframeOrigin = getOrigin();
|
|
|
|
} else {
|
|
|
|
loginIframe.iframeOrigin = realmUrl.substring(0, realmUrl.indexOf('/', 8));
|
|
|
|
}
|
|
|
|
promise.setSuccess();
|
|
|
|
|
|
|
|
setTimeout(check, loginIframe.interval * 1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
var src = getRealmUrl() + '/protocol/openid-connect/login-status-iframe.html?client_id=' + encodeURIComponent(kc.clientId) + '&origin=' + getOrigin();
|
|
|
|
iframe.setAttribute('src', src );
|
|
|
|
iframe.style.display = 'none';
|
|
|
|
document.body.appendChild(iframe);
|
|
|
|
|
|
|
|
var messageCallback = function(event) {
|
2016-07-04 11:07:17 +00:00
|
|
|
|
|
|
|
|
2016-10-17 12:05:36 +00:00
|
|
|
if (event.origin !== loginIframe.iframeOrigin) {
|
2016-07-04 11:07:17 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-10-17 12:05:36 +00:00
|
|
|
|
|
|
|
for (i = loginIframe.callbackList.length - 1; i >= 0; --i) {
|
|
|
|
var promise = loginIframe.callbackList[i];
|
|
|
|
if (event.data == "unchanged") {
|
|
|
|
promise.setSuccess();
|
|
|
|
} else {
|
|
|
|
kc.clearToken();
|
|
|
|
promise.setError();
|
|
|
|
}
|
|
|
|
loginIframe.callbackList.splice(i, 1);
|
2016-07-04 11:07:17 +00:00
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
};
|
|
|
|
window.addEventListener('message', messageCallback, false);
|
|
|
|
|
|
|
|
var check = function() {
|
|
|
|
checkLoginIframe();
|
|
|
|
if (kc.token) {
|
|
|
|
setTimeout(check, loginIframe.interval * 1000);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
function checkLoginIframe() {
|
|
|
|
var promise = createPromise();
|
|
|
|
|
2016-10-18 09:50:51 +00:00
|
|
|
if (loginIframe.iframe && loginIframe.iframeOrigin ) {
|
2015-07-17 11:45:43 +00:00
|
|
|
var msg = {};
|
|
|
|
msg.callbackId = createCallbackId();
|
2016-10-17 12:05:36 +00:00
|
|
|
msg.sessionId = kc.sessionId;
|
|
|
|
loginIframe.callbackList.push(promise);
|
2015-07-17 11:45:43 +00:00
|
|
|
var origin = loginIframe.iframeOrigin;
|
2016-10-18 09:50:51 +00:00
|
|
|
if(loginIframe.callbackList.length == 1) {
|
|
|
|
loginIframe.iframe.contentWindow.postMessage(JSON.stringify(msg), origin);
|
|
|
|
}
|
2015-07-17 11:45:43 +00:00
|
|
|
} else {
|
|
|
|
promise.setSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
function loadAdapter(type) {
|
|
|
|
if (!type || type == 'default') {
|
|
|
|
return {
|
|
|
|
login: function(options) {
|
|
|
|
window.location.href = kc.createLoginUrl(options);
|
|
|
|
return createPromise().promise;
|
|
|
|
},
|
|
|
|
|
|
|
|
logout: function(options) {
|
|
|
|
window.location.href = kc.createLogoutUrl(options);
|
|
|
|
return createPromise().promise;
|
|
|
|
},
|
|
|
|
|
2015-10-07 11:53:25 +00:00
|
|
|
register: function(options) {
|
|
|
|
window.location.href = kc.createRegisterUrl(options);
|
|
|
|
return createPromise().promise;
|
|
|
|
},
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
accountManagement : function() {
|
|
|
|
window.location.href = kc.createAccountUrl();
|
|
|
|
return createPromise().promise;
|
|
|
|
},
|
|
|
|
|
2016-02-06 16:43:38 +00:00
|
|
|
redirectUri: function(options, encodeHash) {
|
|
|
|
if (arguments.length == 1) {
|
|
|
|
encodeHash = true;
|
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
if (options && options.redirectUri) {
|
|
|
|
return options.redirectUri;
|
|
|
|
} else if (kc.redirectUri) {
|
|
|
|
return kc.redirectUri;
|
|
|
|
} else {
|
|
|
|
var redirectUri = location.href;
|
2016-02-06 16:43:38 +00:00
|
|
|
if (location.hash && encodeHash) {
|
2015-07-17 11:45:43 +00:00
|
|
|
redirectUri = redirectUri.substring(0, location.href.indexOf('#'));
|
|
|
|
redirectUri += (redirectUri.indexOf('?') == -1 ? '?' : '&') + 'redirect_fragment=' + encodeURIComponent(location.hash.substring(1));
|
|
|
|
}
|
|
|
|
return redirectUri;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type == 'cordova') {
|
|
|
|
loginIframe.enable = false;
|
|
|
|
|
|
|
|
return {
|
|
|
|
login: function(options) {
|
|
|
|
var promise = createPromise();
|
|
|
|
|
|
|
|
var o = 'location=no';
|
|
|
|
if (options && options.prompt == 'none') {
|
|
|
|
o += ',hidden=yes';
|
|
|
|
}
|
|
|
|
|
|
|
|
var loginUrl = kc.createLoginUrl(options);
|
|
|
|
var ref = window.open(loginUrl, '_blank', o);
|
|
|
|
|
2016-02-23 12:31:31 +00:00
|
|
|
var completed = false;
|
2015-07-17 11:45:43 +00:00
|
|
|
|
|
|
|
ref.addEventListener('loadstart', function(event) {
|
|
|
|
if (event.url.indexOf('http://localhost') == 0) {
|
2016-02-23 12:31:31 +00:00
|
|
|
var callback = parseCallback(event.url);
|
|
|
|
processCallback(callback, promise);
|
2015-07-17 11:45:43 +00:00
|
|
|
ref.close();
|
2016-02-23 12:31:31 +00:00
|
|
|
completed = true;
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
ref.addEventListener('loaderror', function(event) {
|
2016-02-23 12:31:31 +00:00
|
|
|
if (!completed) {
|
|
|
|
if (event.url.indexOf('http://localhost') == 0) {
|
|
|
|
var callback = parseCallback(event.url);
|
|
|
|
processCallback(callback, promise);
|
|
|
|
ref.close();
|
|
|
|
completed = true;
|
|
|
|
} else {
|
|
|
|
promise.setError();
|
|
|
|
ref.close();
|
|
|
|
}
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
},
|
|
|
|
|
|
|
|
logout: function(options) {
|
|
|
|
var promise = createPromise();
|
|
|
|
|
|
|
|
var logoutUrl = kc.createLogoutUrl(options);
|
|
|
|
var ref = window.open(logoutUrl, '_blank', 'location=no,hidden=yes');
|
|
|
|
|
|
|
|
var error;
|
|
|
|
|
|
|
|
ref.addEventListener('loadstart', function(event) {
|
|
|
|
if (event.url.indexOf('http://localhost') == 0) {
|
|
|
|
ref.close();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
ref.addEventListener('loaderror', function(event) {
|
2016-02-04 08:47:25 +00:00
|
|
|
if (event.url.indexOf('http://localhost') == 0) {
|
|
|
|
ref.close();
|
|
|
|
} else {
|
2015-07-17 11:45:43 +00:00
|
|
|
error = true;
|
|
|
|
ref.close();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
ref.addEventListener('exit', function(event) {
|
|
|
|
if (error) {
|
|
|
|
promise.setError();
|
|
|
|
} else {
|
|
|
|
kc.clearToken();
|
|
|
|
promise.setSuccess();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return promise.promise;
|
|
|
|
},
|
|
|
|
|
2015-10-07 11:53:25 +00:00
|
|
|
register : function() {
|
|
|
|
var registerUrl = kc.createRegisterUrl();
|
|
|
|
var ref = window.open(registerUrl, '_blank', 'location=no');
|
|
|
|
ref.addEventListener('loadstart', function(event) {
|
|
|
|
if (event.url.indexOf('http://localhost') == 0) {
|
|
|
|
ref.close();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
accountManagement : function() {
|
|
|
|
var accountUrl = kc.createAccountUrl();
|
|
|
|
var ref = window.open(accountUrl, '_blank', 'location=no');
|
|
|
|
ref.addEventListener('loadstart', function(event) {
|
|
|
|
if (event.url.indexOf('http://localhost') == 0) {
|
|
|
|
ref.close();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
redirectUri: function(options) {
|
|
|
|
return 'http://localhost';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
throw 'invalid adapter type: ' + type;
|
|
|
|
}
|
2015-11-24 13:26:41 +00:00
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
var LocalStorage = function() {
|
|
|
|
if (!(this instanceof LocalStorage)) {
|
|
|
|
return new LocalStorage();
|
|
|
|
}
|
2015-11-24 13:26:41 +00:00
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
localStorage.setItem('kc-test', 'test');
|
|
|
|
localStorage.removeItem('kc-test');
|
|
|
|
|
|
|
|
var cs = this;
|
|
|
|
|
|
|
|
function clearExpired() {
|
|
|
|
var time = new Date().getTime();
|
|
|
|
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 {
|
|
|
|
var expires = JSON.parse(value).expires;
|
|
|
|
if (!expires || expires < time) {
|
|
|
|
localStorage.removeItem(key);
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
localStorage.removeItem(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-04-15 12:09:07 +00:00
|
|
|
}
|
2016-07-04 18:58:15 +00:00
|
|
|
|
|
|
|
cs.get = function(state) {
|
|
|
|
if (!state) {
|
|
|
|
return;
|
2016-04-15 12:09:07 +00:00
|
|
|
}
|
2016-07-04 18:58:15 +00:00
|
|
|
|
|
|
|
var key = 'kc-callback-' + state;
|
|
|
|
var value = localStorage.getItem(key);
|
|
|
|
if (value) {
|
2016-04-15 12:09:07 +00:00
|
|
|
localStorage.removeItem(key);
|
2016-07-04 18:58:15 +00:00
|
|
|
value = JSON.parse(value);
|
2016-04-15 12:09:07 +00:00
|
|
|
}
|
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
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();
|
2016-04-15 12:09:07 +00:00
|
|
|
}
|
2016-09-06 06:59:58 +00:00
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
var cs = this;
|
2016-04-15 12:09:07 +00:00
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
cs.get = function(state) {
|
|
|
|
if (!state) {
|
|
|
|
return;
|
2016-04-15 12:09:07 +00:00
|
|
|
}
|
|
|
|
|
2016-07-04 18:58:15 +00:00
|
|
|
var value = getCookie('kc-callback-' + state);
|
|
|
|
setCookie('kc-callback-' + state, '', cookieExpiration(-100));
|
|
|
|
if (value) {
|
|
|
|
return JSON.parse(value);
|
2016-04-15 12:09:07 +00:00
|
|
|
}
|
2016-07-04 18:58:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
cs.add = function(state) {
|
|
|
|
setCookie('kc-callback-' + state.state, JSON.stringify(state), cookieExpiration(60));
|
|
|
|
};
|
|
|
|
|
|
|
|
cs.removeItem = function(key) {
|
2016-04-15 12:09:07 +00:00
|
|
|
setCookie(key, '', cookieExpiration(-100));
|
2016-07-04 18:58:15 +00:00
|
|
|
};
|
2016-04-15 12:09:07 +00:00
|
|
|
|
|
|
|
var cookieExpiration = function (minutes) {
|
|
|
|
var exp = new Date();
|
|
|
|
exp.setTime(exp.getTime() + (minutes*60*1000));
|
|
|
|
return exp;
|
2016-07-04 18:58:15 +00:00
|
|
|
};
|
2016-04-15 12:09:07 +00:00
|
|
|
|
|
|
|
var getCookie = function (key) {
|
|
|
|
var name = key + '=';
|
|
|
|
var ca = document.cookie.split(';');
|
|
|
|
for (var i = 0; i < ca.length; i++) {
|
|
|
|
var c = ca[i];
|
|
|
|
while (c.charAt(0) == ' ') {
|
|
|
|
c = c.substring(1);
|
|
|
|
}
|
|
|
|
if (c.indexOf(name) == 0) {
|
|
|
|
return c.substring(name.length, c.length);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return '';
|
2016-07-04 18:58:15 +00:00
|
|
|
};
|
2016-04-15 12:09:07 +00:00
|
|
|
|
|
|
|
var setCookie = function (key, value, expirationDate) {
|
|
|
|
var cookie = key + '=' + value + '; '
|
|
|
|
+ 'expires=' + expirationDate.toUTCString() + '; ';
|
|
|
|
document.cookie = cookie;
|
|
|
|
}
|
2016-07-04 18:58:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
function createCallbackStorage() {
|
|
|
|
try {
|
|
|
|
return new LocalStorage();
|
|
|
|
} catch (err) {
|
|
|
|
}
|
|
|
|
|
|
|
|
return new CookieStorage();
|
2016-04-15 12:09:07 +00:00
|
|
|
}
|
|
|
|
|
2015-11-24 13:26:41 +00:00
|
|
|
var CallbackParser = function(uriToParse, responseMode) {
|
|
|
|
if (!(this instanceof CallbackParser)) {
|
|
|
|
return new CallbackParser(uriToParse, responseMode);
|
|
|
|
}
|
|
|
|
var parser = this;
|
|
|
|
|
|
|
|
var initialParse = function() {
|
|
|
|
var baseUri = null;
|
|
|
|
var queryString = null;
|
|
|
|
var fragmentString = null;
|
|
|
|
|
|
|
|
var questionMarkIndex = uriToParse.indexOf("?");
|
|
|
|
var fragmentIndex = uriToParse.indexOf("#", questionMarkIndex + 1);
|
|
|
|
if (questionMarkIndex == -1 && fragmentIndex == -1) {
|
|
|
|
baseUri = uriToParse;
|
|
|
|
} else if (questionMarkIndex != -1) {
|
|
|
|
baseUri = uriToParse.substring(0, questionMarkIndex);
|
|
|
|
queryString = uriToParse.substring(questionMarkIndex + 1);
|
|
|
|
if (fragmentIndex != -1) {
|
|
|
|
fragmentIndex = queryString.indexOf("#");
|
|
|
|
fragmentString = queryString.substring(fragmentIndex + 1);
|
|
|
|
queryString = queryString.substring(0, fragmentIndex);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
baseUri = uriToParse.substring(0, fragmentIndex);
|
|
|
|
fragmentString = uriToParse.substring(fragmentIndex + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return { baseUri: baseUri, queryString: queryString, fragmentString: fragmentString };
|
|
|
|
}
|
|
|
|
|
|
|
|
var parseParams = function(paramString) {
|
|
|
|
var result = {};
|
|
|
|
var params = paramString.split('&');
|
|
|
|
for (var i = 0; i < params.length; i++) {
|
|
|
|
var p = params[i].split('=');
|
|
|
|
var paramName = decodeURIComponent(p[0]);
|
|
|
|
var paramValue = decodeURIComponent(p[1]);
|
|
|
|
result[paramName] = paramValue;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
var handleQueryParam = function(paramName, paramValue, oauth) {
|
2016-07-14 21:55:27 +00:00
|
|
|
var supportedOAuthParams = [ 'code', 'state', 'error', 'error_description' ];
|
2015-11-24 13:26:41 +00:00
|
|
|
|
|
|
|
for (var i = 0 ; i< supportedOAuthParams.length ; i++) {
|
|
|
|
if (paramName === supportedOAuthParams[i]) {
|
|
|
|
oauth[paramName] = paramValue;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
parser.parseUri = function() {
|
|
|
|
var parsedUri = initialParse();
|
|
|
|
|
|
|
|
var queryParams = {};
|
|
|
|
if (parsedUri.queryString) {
|
|
|
|
queryParams = parseParams(parsedUri.queryString);
|
|
|
|
}
|
|
|
|
|
|
|
|
var oauth = { newUrl: parsedUri.baseUri };
|
|
|
|
for (var param in queryParams) {
|
|
|
|
switch (param) {
|
|
|
|
case 'redirect_fragment':
|
|
|
|
oauth.fragment = queryParams[param];
|
|
|
|
break;
|
|
|
|
case 'prompt':
|
|
|
|
oauth.prompt = queryParams[param];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (responseMode != 'query' || !handleQueryParam(param, queryParams[param], oauth)) {
|
|
|
|
oauth.newUrl += (oauth.newUrl.indexOf('?') == -1 ? '?' : '&') + param + '=' + queryParams[param];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (responseMode === 'fragment') {
|
|
|
|
var fragmentParams = {};
|
|
|
|
if (parsedUri.fragmentString) {
|
|
|
|
fragmentParams = parseParams(parsedUri.fragmentString);
|
|
|
|
}
|
|
|
|
for (var param in fragmentParams) {
|
|
|
|
oauth[param] = fragmentParams[param];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return oauth;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-17 11:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( typeof module === "object" && module && typeof module.exports === "object" ) {
|
|
|
|
module.exports = Keycloak;
|
|
|
|
} else {
|
|
|
|
window.Keycloak = Keycloak;
|
|
|
|
|
|
|
|
if ( typeof define === "function" && define.amd ) {
|
|
|
|
define( "keycloak", [], function () { return Keycloak; } );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})( window );
|