diff --git a/console2/src/components/organisms/ProcessActivity/Toolbar.tsx b/console2/src/components/organisms/ProcessActivity/Toolbar.tsx
index 1594193938..6bbd3393f1 100644
--- a/console2/src/components/organisms/ProcessActivity/Toolbar.tsx
+++ b/console2/src/components/organisms/ProcessActivity/Toolbar.tsx
@@ -146,7 +146,7 @@ const renderProcessStatus = (process?: ProcessEntry) => {
let duration;
if (process.status === ProcessStatus.RUNNING) {
- duration = formatDuration(new Date().getTime() - parseDate(process.createdAt).getTime());
+ duration = formatDuration(new Date().getTime() - parseDate(process.lastRunAt || process.createdAt).getTime());
}
return (
<>
diff --git a/server/dist/src/main/resources/concord-server.conf b/server/dist/src/main/resources/concord-server.conf
index fe0db4b7fb..e7b0b231ed 100644
--- a/server/dist/src/main/resources/concord-server.conf
+++ b/server/dist/src/main/resources/concord-server.conf
@@ -533,6 +533,17 @@ concord-server {
pfed {
enabled = false
priority = 0
+
+ bearerToken {
+ # enable bearer tokens
+ enableBearerTokens = false
+
+ # allow all clientIds
+ allowAllClientIds = false
+
+ # list of allowed pingfed clientids for bearer tokens
+ allowedClientIds = ["clientId1", "clientId2"]
+ }
}
authEndpointUrl = "http://auth.example.com/authorize"
tokenEndpointUrl = "http://auth.example.com/token"
@@ -549,7 +560,7 @@ concord-server {
# enable to validate token signature
tokenSignatureValidation = false
-
+
# JSON as a string
#tokenEncryptionKey = "{}"
diff --git a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/JwtAuthenticator.java b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/JwtAuthenticator.java
index 754400733f..63753fee7a 100644
--- a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/JwtAuthenticator.java
+++ b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/JwtAuthenticator.java
@@ -39,6 +39,8 @@
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
+import java.util.List;
+
public class JwtAuthenticator {
@@ -78,8 +80,8 @@ public JwtAuthenticator(SsoConfiguration cfg, SsoClient ssoClient) {
* @param token the JWT
* @return true
if token valid and not expired
*/
- public boolean isTokenValid(String token) {
- return isTokenValid(token, null);
+ public boolean isTokenValid(String token, boolean restrictOnClientId) {
+ return isTokenValid(token, null, restrictOnClientId);
}
/**
@@ -89,13 +91,22 @@ public boolean isTokenValid(String token) {
* @param nonce nonce
* @return true
if token valid, correct nonce and not expired
*/
- public boolean isTokenValid(String token, String nonce) {
+ public boolean isTokenValid(String token, String nonce, boolean restrictOnClientId) {
try {
Map claims = validateTokenAndGetClaims(token);
if (claims == null) {
return false;
}
+ if (restrictOnClientId) {
+ List allowedClientIds = cfg.getAllowedClientIds();
+ String clientId = (String) claims.get("client_id");
+ if(!allowedClientIds.contains(clientId)) {
+ log.warn("isTokenValid ['{}', '{}'] -> clientId not in allowed list for bearer tokens", token, clientId);
+ return false;
+ }
+ }
+
if (nonce == null) {
return true;
}
@@ -113,22 +124,7 @@ public boolean isTokenValid(String token, String nonce) {
}
}
- /**
- * Validates the token and returns the corresponding user login.
- *
- * @param token the JWT
- * @return corresponding user login or null
if the JWT is invalid
- */
- public String validateTokenAndGetLogin(String token) {
- Map claims = validateTokenAndGetClaims(token);
- if (claims == null) {
- return null;
- }
- return (String) claims.get("sub");
- }
-
- private Map validateTokenAndGetClaims(String token) {
-
+ public Map validateTokenAndGetClaims(String token) {
try {
JWT jwt = validateToken(token);
if (jwt == null) {
diff --git a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoAuthFilter.java b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoAuthFilter.java
index 2d9051dee6..ddc7c13e1f 100644
--- a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoAuthFilter.java
+++ b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoAuthFilter.java
@@ -66,7 +66,7 @@ public void doFilter(HttpServletRequest request, HttpServletResponse response, F
if (token != null) {
if (refreshToken == null){
- boolean isValid = jwtAuthenticator.isTokenValid(token);
+ boolean isValid = jwtAuthenticator.isTokenValid(token, false);
if (isValid) {
log.info("doFilter -> found valid token in cookies, redirect to '{}'", from);
redirectHelper.sendRedirect(response, from);
diff --git a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoClient.java b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoClient.java
index 5e4f83dc82..f07a303527 100644
--- a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoClient.java
+++ b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoClient.java
@@ -126,9 +126,9 @@ public String getTokenSigningKey() throws IOException {
}
}
- public Profile getUserProfile(String refreshToken) throws IOException {
+ public Profile getUserProfileByRefreshToken(String refreshToken) throws IOException {
Token token = getTokenByRefreshToken(refreshToken);
- return getProfile(token);
+ return getProfile(token.accessToken());
}
private Token getToken(String urlParameters) throws IOException {
@@ -178,7 +178,7 @@ private void postRequest(HttpURLConnection con, String urlParameters) throws IOE
}
}
- private Profile getProfile(Token token) throws IOException {
+ public Profile getProfile(String accessToken) throws IOException {
if (cfg.getUserInfoEndpointUrl() == null) {
return null;
}
@@ -186,7 +186,7 @@ private Profile getProfile(Token token) throws IOException {
try {
URL url = new URL(cfg.getUserInfoEndpointUrl());
con = (HttpURLConnection) url.openConnection();
- String authzHeaderValue = String.format("Bearer %s", token.accessToken());
+ String authzHeaderValue = String.format("Bearer %s", accessToken);
con.setRequestProperty(HttpHeaders.AUTHORIZATION, authzHeaderValue);
con.setRequestProperty(HttpHeaders.CONTENT_TYPE, CONTENT_TYPE_HEADER);
con.setRequestMethod("GET");
@@ -240,6 +240,9 @@ public interface Token {
@JsonIgnoreProperties(ignoreUnknown = true)
public interface Profile {
+ @JsonProperty("sub")
+ String sub();
+
@JsonProperty("sAMAccountName")
String userId();
diff --git a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoConfiguration.java b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoConfiguration.java
index 2514ccc426..880e3fce09 100644
--- a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoConfiguration.java
+++ b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoConfiguration.java
@@ -26,6 +26,10 @@
import javax.inject.Inject;
import java.io.Serializable;
import java.time.Duration;
+import java.util.Collections;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
public class SsoConfiguration implements Serializable {
@@ -63,6 +67,14 @@ public class SsoConfiguration implements Serializable {
@Config("sso.clientSecret")
private String clientSecret;
+ @Inject
+ @Config("sso.pfed.bearerToken.enableBearerTokens")
+ private boolean enableBearerTokens;
+
+ @Inject
+ @Config("sso.pfed.bearerToken.allowAllClientIds")
+ private boolean allowAllClientIds;
+
@Inject
@Nullable
@Config("sso.tokenSigningKey")
@@ -103,6 +115,10 @@ public class SsoConfiguration implements Serializable {
@Config("sso.autoCreateUsers")
private boolean autoCreateUsers;
+ @Inject
+ @Config("sso.pfed.bearerToken.allowedClientIds")
+ private List allowedClientIds;
+
public boolean isAutoCreateUsers() {
return autoCreateUsers;
}
@@ -135,6 +151,14 @@ public String getClientSecret() {
return clientSecret;
}
+ public boolean getEnableBearerTokens() {
+ return enableBearerTokens;
+ }
+
+ public boolean getAllowAllClientIds() {
+ return allowAllClientIds;
+ }
+
public String getTokenEncryptionKey() {
return tokenEncryptionKey;
}
@@ -170,4 +194,9 @@ public boolean isTokenSignatureValidation() {
public String getUserInfoEndpointUrl() {
return userInfoEndpointUrl;
}
+
+ public List getAllowedClientIds() {
+ return allowedClientIds;
+ }
+
}
diff --git a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoHandler.java b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoHandler.java
index f7aaa47d2a..a7136495ae 100644
--- a/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoHandler.java
+++ b/server/plugins/pfed-sso/src/main/java/com/walmartlabs/concord/server/plugins/pfedsso/SsoHandler.java
@@ -56,30 +56,34 @@ public AuthenticationToken createToken(ServletRequest request, ServletResponse r
HttpServletRequest req = WebUtils.toHttp(request);
- String token = SsoCookies.getTokenCookie(req);
+ String bearerToken = cfg.getEnableBearerTokens() ? extractTokenFromRequest(req) : null;
+ String token = bearerToken != null ? bearerToken : SsoCookies.getTokenCookie(req);
+
if (token == null) {
return null;
}
- String login = jwtAuthenticator.validateTokenAndGetLogin(token);
- if (login == null) {
+ boolean restrictOnClientId = (bearerToken != null) && (!cfg.getAllowAllClientIds());
+
+ if (!jwtAuthenticator.isTokenValid(token, restrictOnClientId)) {
return null;
}
- String[] as = parseDomain(login);
-
- String refreshToken = SsoCookies.getRefreshCookie(req);
- // get userprofile send the response as null if refreshToken is expired or used
- SsoClient.Profile profile;
try {
- profile = ssoClient.getUserProfile(refreshToken);
+ SsoClient.Profile profile = bearerToken != null ? ssoClient.getProfile(bearerToken) :
+ ssoClient.getUserProfileByRefreshToken(SsoCookies.getRefreshCookie(req));
+
+ if (profile == null) {
+ return null;
+ }
+
+ String[] as = parseDomain(profile.sub());
+
+ return new SsoToken(as[0], as[1], profile.displayName(), profile.mail(), profile.userPrincipalName(), profile.nameInNamespace(), profile.groups());
+
} catch (IOException e) {
return null;
}
- if (profile == null) {
- return null;
- }
- return new SsoToken(as[0], as[1], profile.displayName(), profile.mail(), profile.userPrincipalName(), profile.nameInNamespace(), profile.groups());
}
@Override
@@ -112,4 +116,20 @@ private String[] parseDomain(String s) {
String domain = s.substring(pos + 1);
return new String[]{username, domain};
}
+
+ private String extractTokenFromRequest(HttpServletRequest request) {
+ final String value = request.getHeader("Authorization");
+
+ if (value == null || !value.toLowerCase().startsWith("bearer")) {
+ return null;
+ }
+
+ String[] parts = value.split(" ");
+
+ if (parts.length < 2) {
+ return null;
+ }
+
+ return parts[1].trim();
+ }
}