[KEYCLOAK-5015] - Support for Elytron capabilities in subsystem
This commit is contained in:
parent
d3e559453b
commit
a6dfb4ccdd
20 changed files with 885 additions and 176 deletions
|
@ -68,6 +68,10 @@
|
||||||
<artifactId>wildfly-web-common</artifactId>
|
<artifactId>wildfly-web-common</artifactId>
|
||||||
<scope>provided</scope>
|
<scope>provided</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.wildfly.security</groupId>
|
||||||
|
<artifactId>wildfly-elytron</artifactId>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.jboss.logging</groupId>
|
<groupId>org.jboss.logging</groupId>
|
||||||
<artifactId>jboss-logging-annotations</artifactId>
|
<artifactId>jboss-logging-annotations</artifactId>
|
||||||
|
@ -101,5 +105,9 @@
|
||||||
<artifactId>keycloak-wildfly-adapter</artifactId>
|
<artifactId>keycloak-wildfly-adapter</artifactId>
|
||||||
<version>${project.version}</version>
|
<version>${project.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.keycloak</groupId>
|
||||||
|
<artifactId>keycloak-wildfly-elytron-oidc-adapter</artifactId>
|
||||||
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
</project>
|
</project>
|
||||||
|
|
|
@ -17,41 +17,32 @@
|
||||||
|
|
||||||
package org.keycloak.subsystem.adapter.extension;
|
package org.keycloak.subsystem.adapter.extension;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import org.jboss.as.controller.AbstractAddStepHandler;
|
import org.jboss.as.controller.AbstractAddStepHandler;
|
||||||
import org.jboss.as.controller.AttributeDefinition;
|
|
||||||
import org.jboss.as.controller.OperationContext;
|
import org.jboss.as.controller.OperationContext;
|
||||||
import org.jboss.as.controller.OperationFailedException;
|
import org.jboss.as.controller.OperationFailedException;
|
||||||
|
import org.jboss.as.controller.SimpleAttributeDefinition;
|
||||||
|
import org.jboss.as.controller.capability.RuntimeCapability;
|
||||||
import org.jboss.dmr.ModelNode;
|
import org.jboss.dmr.ModelNode;
|
||||||
|
|
||||||
import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.ADD;
|
|
||||||
import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.OP;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add a deployment to a realm.
|
* Add a deployment to a realm.
|
||||||
*
|
*
|
||||||
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
*/
|
*/
|
||||||
public final class SecureDeploymentAddHandler extends AbstractAddStepHandler {
|
abstract class AbstractAdapterConfigurationAddHandler extends AbstractAddStepHandler {
|
||||||
|
|
||||||
public static SecureDeploymentAddHandler INSTANCE = new SecureDeploymentAddHandler();
|
private final boolean elytronEnabled;
|
||||||
|
|
||||||
private SecureDeploymentAddHandler() {}
|
AbstractAdapterConfigurationAddHandler(RuntimeCapability<Void> runtimeCapability, List<SimpleAttributeDefinition> attributes) {
|
||||||
|
super(runtimeCapability, attributes);
|
||||||
@Override
|
elytronEnabled = runtimeCapability != null;
|
||||||
protected void populateModel(ModelNode operation, ModelNode model) throws OperationFailedException {
|
|
||||||
// TODO: localize exception. get id number
|
|
||||||
if (!operation.get(OP).asString().equals(ADD)) {
|
|
||||||
throw new OperationFailedException("Unexpected operation for add secure deployment. operation=" + operation.toString());
|
|
||||||
}
|
|
||||||
|
|
||||||
for (AttributeDefinition attr : SecureDeploymentDefinition.ALL_ATTRIBUTES) {
|
|
||||||
attr.validateAndSet(operation, model);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
||||||
KeycloakAdapterConfigService ckService = KeycloakAdapterConfigService.getInstance();
|
KeycloakAdapterConfigService ckService = KeycloakAdapterConfigService.getInstance();
|
||||||
ckService.addSecureDeployment(operation, context.resolveExpressions(model));
|
ckService.addSecureDeployment(operation, context.resolveExpressions(model), elytronEnabled);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -0,0 +1,164 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source.
|
||||||
|
* Copyright 2016 Red Hat, Inc., and individual 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.
|
||||||
|
*/
|
||||||
|
package org.keycloak.subsystem.adapter.extension;
|
||||||
|
|
||||||
|
import org.jboss.as.controller.AttributeDefinition;
|
||||||
|
import org.jboss.as.controller.PathElement;
|
||||||
|
import org.jboss.as.controller.SimpleAttributeDefinition;
|
||||||
|
import org.jboss.as.controller.SimpleAttributeDefinitionBuilder;
|
||||||
|
import org.jboss.as.controller.SimpleResourceDefinition;
|
||||||
|
import org.jboss.as.controller.operations.common.GenericSubsystemDescribeHandler;
|
||||||
|
import org.jboss.as.controller.operations.validation.IntRangeValidator;
|
||||||
|
import org.jboss.as.controller.operations.validation.StringLengthValidator;
|
||||||
|
import org.jboss.as.controller.registry.ManagementResourceRegistration;
|
||||||
|
import org.jboss.dmr.ModelNode;
|
||||||
|
import org.jboss.dmr.ModelType;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines attributes and operations for a secure-deployment.
|
||||||
|
*
|
||||||
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
|
*/
|
||||||
|
abstract class AbstractAdapterConfigurationDefinition extends SimpleResourceDefinition {
|
||||||
|
|
||||||
|
protected static final SimpleAttributeDefinition REALM =
|
||||||
|
new SimpleAttributeDefinitionBuilder("realm", ModelType.STRING, true)
|
||||||
|
.setXmlName("realm")
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.setValidator(new StringLengthValidator(1, Integer.MAX_VALUE, true, true))
|
||||||
|
.build();
|
||||||
|
protected static final SimpleAttributeDefinition RESOURCE =
|
||||||
|
new SimpleAttributeDefinitionBuilder("resource", ModelType.STRING, true)
|
||||||
|
.setXmlName("resource")
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.setValidator(new StringLengthValidator(1, Integer.MAX_VALUE, true, true))
|
||||||
|
.build();
|
||||||
|
protected static final SimpleAttributeDefinition USE_RESOURCE_ROLE_MAPPINGS =
|
||||||
|
new SimpleAttributeDefinitionBuilder("use-resource-role-mappings", ModelType.BOOLEAN, true)
|
||||||
|
.setXmlName("use-resource-role-mappings")
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.setDefaultValue(new ModelNode(false))
|
||||||
|
.build();
|
||||||
|
protected static final SimpleAttributeDefinition BEARER_ONLY =
|
||||||
|
new SimpleAttributeDefinitionBuilder("bearer-only", ModelType.BOOLEAN, true)
|
||||||
|
.setXmlName("bearer-only")
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.setDefaultValue(new ModelNode(false))
|
||||||
|
.build();
|
||||||
|
protected static final SimpleAttributeDefinition ENABLE_BASIC_AUTH =
|
||||||
|
new SimpleAttributeDefinitionBuilder("enable-basic-auth", ModelType.BOOLEAN, true)
|
||||||
|
.setXmlName("enable-basic-auth")
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.setDefaultValue(new ModelNode(false))
|
||||||
|
.build();
|
||||||
|
protected static final SimpleAttributeDefinition PUBLIC_CLIENT =
|
||||||
|
new SimpleAttributeDefinitionBuilder("public-client", ModelType.BOOLEAN, true)
|
||||||
|
.setXmlName("public-client")
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.setDefaultValue(new ModelNode(false))
|
||||||
|
.build();
|
||||||
|
protected static final SimpleAttributeDefinition TURN_OFF_CHANGE_SESSION =
|
||||||
|
new SimpleAttributeDefinitionBuilder("turn-off-change-session-id-on-login", ModelType.BOOLEAN, true)
|
||||||
|
.setXmlName("turn-off-change-session-id-on-login")
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.setDefaultValue(new ModelNode(false))
|
||||||
|
.build();
|
||||||
|
protected static final SimpleAttributeDefinition TOKEN_MINIMUM_TIME_TO_LIVE =
|
||||||
|
new SimpleAttributeDefinitionBuilder("token-minimum-time-to-live", ModelType.INT, true)
|
||||||
|
.setXmlName("token-minimum-time-to-live")
|
||||||
|
.setValidator(new IntRangeValidator(-1, true))
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.build();
|
||||||
|
protected static final SimpleAttributeDefinition MIN_TIME_BETWEEN_JWKS_REQUESTS =
|
||||||
|
new SimpleAttributeDefinitionBuilder("min-time-between-jwks-requests", ModelType.INT, true)
|
||||||
|
.setXmlName("min-time-between-jwks-requests")
|
||||||
|
.setValidator(new IntRangeValidator(-1, true))
|
||||||
|
.setAllowExpression(true)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
static final List<SimpleAttributeDefinition> DEPLOYMENT_ONLY_ATTRIBUTES = new ArrayList<SimpleAttributeDefinition>();
|
||||||
|
|
||||||
|
static {
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(REALM);
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(RESOURCE);
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(USE_RESOURCE_ROLE_MAPPINGS);
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(BEARER_ONLY);
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(ENABLE_BASIC_AUTH);
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(PUBLIC_CLIENT);
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(TURN_OFF_CHANGE_SESSION);
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(TOKEN_MINIMUM_TIME_TO_LIVE);
|
||||||
|
DEPLOYMENT_ONLY_ATTRIBUTES.add(MIN_TIME_BETWEEN_JWKS_REQUESTS);
|
||||||
|
}
|
||||||
|
|
||||||
|
static final List<SimpleAttributeDefinition> ALL_ATTRIBUTES = new ArrayList();
|
||||||
|
|
||||||
|
static {
|
||||||
|
ALL_ATTRIBUTES.addAll(DEPLOYMENT_ONLY_ATTRIBUTES);
|
||||||
|
ALL_ATTRIBUTES.addAll(SharedAttributeDefinitons.ATTRIBUTES);
|
||||||
|
}
|
||||||
|
|
||||||
|
static final Map<String, SimpleAttributeDefinition> XML_ATTRIBUTES = new HashMap<String, SimpleAttributeDefinition>();
|
||||||
|
|
||||||
|
static {
|
||||||
|
for (SimpleAttributeDefinition def : ALL_ATTRIBUTES) {
|
||||||
|
XML_ATTRIBUTES.put(def.getXmlName(), def);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final Map<String, SimpleAttributeDefinition> DEFINITION_LOOKUP = new HashMap<String, SimpleAttributeDefinition>();
|
||||||
|
static {
|
||||||
|
for (SimpleAttributeDefinition def : ALL_ATTRIBUTES) {
|
||||||
|
DEFINITION_LOOKUP.put(def.getXmlName(), def);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private final AbstractAdapterConfigurationWriteAttributeHandler attrWriteHandler;
|
||||||
|
private final List<SimpleAttributeDefinition> attributes;
|
||||||
|
|
||||||
|
protected AbstractAdapterConfigurationDefinition(String name, List<SimpleAttributeDefinition> attributes, AbstractAdapterConfigurationAddHandler addHandler, AbstractAdapterConfigurationRemoveHandler removeHandler, AbstractAdapterConfigurationWriteAttributeHandler attrWriteHandler) {
|
||||||
|
super(PathElement.pathElement(name),
|
||||||
|
KeycloakExtension.getResourceDescriptionResolver(name),
|
||||||
|
addHandler,
|
||||||
|
removeHandler);
|
||||||
|
this.attributes = attributes;
|
||||||
|
this.attrWriteHandler = attrWriteHandler;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void registerOperations(ManagementResourceRegistration resourceRegistration) {
|
||||||
|
super.registerOperations(resourceRegistration);
|
||||||
|
resourceRegistration.registerOperationHandler(GenericSubsystemDescribeHandler.DEFINITION, GenericSubsystemDescribeHandler.INSTANCE);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void registerAttributes(ManagementResourceRegistration resourceRegistration) {
|
||||||
|
super.registerAttributes(resourceRegistration);
|
||||||
|
for (AttributeDefinition attrDef : this.attributes) {
|
||||||
|
resourceRegistration.registerReadWriteAttribute(attrDef, null, this.attrWriteHandler);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static SimpleAttributeDefinition lookup(String name) {
|
||||||
|
return DEFINITION_LOOKUP.get(name);
|
||||||
|
}
|
||||||
|
}
|
|
@ -27,11 +27,7 @@ import org.jboss.dmr.ModelNode;
|
||||||
*
|
*
|
||||||
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
*/
|
*/
|
||||||
public final class SecureDeploymentRemoveHandler extends AbstractRemoveStepHandler {
|
abstract class AbstractAdapterConfigurationRemoveHandler extends AbstractRemoveStepHandler {
|
||||||
|
|
||||||
public static SecureDeploymentRemoveHandler INSTANCE = new SecureDeploymentRemoveHandler();
|
|
||||||
|
|
||||||
private SecureDeploymentRemoveHandler() {}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
|
@ -31,14 +31,10 @@ import java.util.List;
|
||||||
*
|
*
|
||||||
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
*/
|
*/
|
||||||
public class SecureDeploymentWriteAttributeHandler extends AbstractWriteAttributeHandler<KeycloakAdapterConfigService> {
|
abstract class AbstractAdapterConfigurationWriteAttributeHandler extends AbstractWriteAttributeHandler<KeycloakAdapterConfigService> {
|
||||||
|
|
||||||
public SecureDeploymentWriteAttributeHandler(List<SimpleAttributeDefinition> definitions) {
|
AbstractAdapterConfigurationWriteAttributeHandler(List<SimpleAttributeDefinition> definitions) {
|
||||||
this(definitions.toArray(new AttributeDefinition[definitions.size()]));
|
super(definitions.toArray(new AttributeDefinition[definitions.size()]));
|
||||||
}
|
|
||||||
|
|
||||||
public SecureDeploymentWriteAttributeHandler(AttributeDefinition... definitions) {
|
|
||||||
super(definitions);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
|
@ -25,7 +25,9 @@ import org.jboss.as.web.common.WarMetaData;
|
||||||
import org.jboss.logging.Logger;
|
import org.jboss.logging.Logger;
|
||||||
import org.jboss.metadata.javaee.spec.ParamValueMetaData;
|
import org.jboss.metadata.javaee.spec.ParamValueMetaData;
|
||||||
import org.jboss.metadata.web.jboss.JBossWebMetaData;
|
import org.jboss.metadata.web.jboss.JBossWebMetaData;
|
||||||
|
import org.jboss.metadata.web.spec.ListenerMetaData;
|
||||||
import org.jboss.metadata.web.spec.LoginConfigMetaData;
|
import org.jboss.metadata.web.spec.LoginConfigMetaData;
|
||||||
|
import org.keycloak.adapters.elytron.KeycloakConfigurationServletListener;
|
||||||
import org.keycloak.subsystem.adapter.logging.KeycloakLogger;
|
import org.keycloak.subsystem.adapter.logging.KeycloakLogger;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
@ -69,6 +71,9 @@ public class KeycloakAdapterConfigDeploymentProcessor implements DeploymentUnitP
|
||||||
KeycloakAdapterConfigService service = KeycloakAdapterConfigService.getInstance();
|
KeycloakAdapterConfigService service = KeycloakAdapterConfigService.getInstance();
|
||||||
if (service.isSecureDeployment(deploymentUnit) && service.isDeploymentConfigured(deploymentUnit)) {
|
if (service.isSecureDeployment(deploymentUnit) && service.isDeploymentConfigured(deploymentUnit)) {
|
||||||
addKeycloakAuthData(phaseContext, service);
|
addKeycloakAuthData(phaseContext, service);
|
||||||
|
} else if (service.isElytronEnabled(deploymentUnit)) {
|
||||||
|
WarMetaData warMetaData = deploymentUnit.getAttachment(WarMetaData.ATTACHMENT_KEY);
|
||||||
|
addConfigurationListener(warMetaData);
|
||||||
}
|
}
|
||||||
|
|
||||||
// FYI, Undertow Extension will find deployments that have auth-method set to KEYCLOAK
|
// FYI, Undertow Extension will find deployments that have auth-method set to KEYCLOAK
|
||||||
|
@ -99,6 +104,10 @@ public class KeycloakAdapterConfigDeploymentProcessor implements DeploymentUnitP
|
||||||
loginConfig.setAuthMethod("KEYCLOAK");
|
loginConfig.setAuthMethod("KEYCLOAK");
|
||||||
loginConfig.setRealmName(service.getRealmName(deploymentUnit));
|
loginConfig.setRealmName(service.getRealmName(deploymentUnit));
|
||||||
KeycloakLogger.ROOT_LOGGER.deploymentSecured(deploymentUnit.getName());
|
KeycloakLogger.ROOT_LOGGER.deploymentSecured(deploymentUnit.getName());
|
||||||
|
|
||||||
|
if (service.isElytronEnabled(deploymentUnit)) {
|
||||||
|
addConfigurationListener(warMetaData);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void addJSONData(String json, WarMetaData warMetaData) {
|
private void addJSONData(String json, WarMetaData warMetaData) {
|
||||||
|
@ -121,6 +130,31 @@ public class KeycloakAdapterConfigDeploymentProcessor implements DeploymentUnitP
|
||||||
webMetaData.setContextParams(contextParams);
|
webMetaData.setContextParams(contextParams);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void addConfigurationListener(WarMetaData warMetaData) {
|
||||||
|
if (warMetaData == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
JBossWebMetaData webMetaData = warMetaData.getMergedJBossWebMetaData();
|
||||||
|
if (webMetaData == null) {
|
||||||
|
webMetaData = new JBossWebMetaData();
|
||||||
|
warMetaData.setMergedJBossWebMetaData(webMetaData);
|
||||||
|
}
|
||||||
|
|
||||||
|
LoginConfigMetaData loginConfig = webMetaData.getLoginConfig();
|
||||||
|
if (loginConfig == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!loginConfig.getAuthMethod().equals("KEYCLOAK")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ListenerMetaData listenerMetaData = new ListenerMetaData();
|
||||||
|
|
||||||
|
listenerMetaData.setListenerClass(KeycloakConfigurationServletListener.class.getName());
|
||||||
|
|
||||||
|
webMetaData.getListeners().add(listenerMetaData);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void undeploy(DeploymentUnit du) {
|
public void undeploy(DeploymentUnit du) {
|
||||||
|
|
||||||
|
|
|
@ -23,8 +23,11 @@ import org.jboss.dmr.ModelNode;
|
||||||
import org.jboss.dmr.Property;
|
import org.jboss.dmr.Property;
|
||||||
import org.jboss.metadata.web.jboss.JBossWebMetaData;
|
import org.jboss.metadata.web.jboss.JBossWebMetaData;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.ADDRESS;
|
import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.ADDRESS;
|
||||||
|
|
||||||
|
@ -50,6 +53,7 @@ public final class KeycloakAdapterConfigService {
|
||||||
|
|
||||||
// keycloak-secured deployments
|
// keycloak-secured deployments
|
||||||
private final Map<String, ModelNode> secureDeployments = new HashMap<String, ModelNode>();
|
private final Map<String, ModelNode> secureDeployments = new HashMap<String, ModelNode>();
|
||||||
|
private final Set<String> elytronEnabledDeployments = new HashSet<>();
|
||||||
|
|
||||||
|
|
||||||
private KeycloakAdapterConfigService() {
|
private KeycloakAdapterConfigService() {
|
||||||
|
@ -68,9 +72,13 @@ public final class KeycloakAdapterConfigService {
|
||||||
this.realms.remove(realmNameFromOp(operation));
|
this.realms.remove(realmNameFromOp(operation));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void addSecureDeployment(ModelNode operation, ModelNode model) {
|
public void addSecureDeployment(ModelNode operation, ModelNode model, boolean elytronEnabled) {
|
||||||
ModelNode deployment = model.clone();
|
ModelNode deployment = model.clone();
|
||||||
this.secureDeployments.put(deploymentNameFromOp(operation), deployment);
|
String name = deploymentNameFromOp(operation);
|
||||||
|
this.secureDeployments.put(name, deployment);
|
||||||
|
if (elytronEnabled) {
|
||||||
|
elytronEnabledDeployments.add(name);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void updateSecureDeployment(ModelNode operation, String attrName, ModelNode resolvedValue) {
|
public void updateSecureDeployment(ModelNode operation, String attrName, ModelNode resolvedValue) {
|
||||||
|
@ -79,7 +87,9 @@ public final class KeycloakAdapterConfigService {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void removeSecureDeployment(ModelNode operation) {
|
public void removeSecureDeployment(ModelNode operation) {
|
||||||
this.secureDeployments.remove(deploymentNameFromOp(operation));
|
String name = deploymentNameFromOp(operation);
|
||||||
|
this.secureDeployments.remove(name);
|
||||||
|
elytronEnabledDeployments.remove(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void addCredential(ModelNode operation, ModelNode model) {
|
public void addCredential(ModelNode operation, ModelNode model) {
|
||||||
|
@ -187,7 +197,19 @@ public final class KeycloakAdapterConfigService {
|
||||||
}
|
}
|
||||||
|
|
||||||
private String deploymentNameFromOp(ModelNode operation) {
|
private String deploymentNameFromOp(ModelNode operation) {
|
||||||
return valueFromOpAddress(SecureDeploymentDefinition.TAG_NAME, operation);
|
String deploymentName = valueFromOpAddress(SecureDeploymentDefinition.TAG_NAME, operation);
|
||||||
|
|
||||||
|
if (deploymentName == null) {
|
||||||
|
deploymentName = valueFromOpAddress(KeycloakHttpServerAuthenticationMechanismFactoryDefinition.TAG_NAME, operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deploymentName == null) {
|
||||||
|
deploymentName = valueFromOpAddress(SecureServerDefinition.TAG_NAME, operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deploymentName == null) throw new RuntimeException("Can't find deployment name in address " + operation);
|
||||||
|
|
||||||
|
return deploymentName;
|
||||||
}
|
}
|
||||||
|
|
||||||
private String credentialNameFromOp(ModelNode operation) {
|
private String credentialNameFromOp(ModelNode operation) {
|
||||||
|
@ -199,9 +221,7 @@ public final class KeycloakAdapterConfigService {
|
||||||
}
|
}
|
||||||
|
|
||||||
private String valueFromOpAddress(String addrElement, ModelNode operation) {
|
private String valueFromOpAddress(String addrElement, ModelNode operation) {
|
||||||
String deploymentName = getValueOfAddrElement(operation.get(ADDRESS), addrElement);
|
return getValueOfAddrElement(operation.get(ADDRESS), addrElement);
|
||||||
if (deploymentName == null) throw new RuntimeException("Can't find '" + addrElement + "' in address " + operation.toString());
|
|
||||||
return deploymentName;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private String getValueOfAddrElement(ModelNode address, String elementName) {
|
private String getValueOfAddrElement(ModelNode address, String elementName) {
|
||||||
|
@ -241,8 +261,22 @@ public final class KeycloakAdapterConfigService {
|
||||||
return json.toJSONString(true);
|
return json.toJSONString(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getJSON(String deploymentName) {
|
||||||
|
ModelNode deployment = this.secureDeployments.get(deploymentName);
|
||||||
|
String realmName = deployment.get(RealmDefinition.TAG_NAME).asString();
|
||||||
|
ModelNode realm = this.realms.get(realmName);
|
||||||
|
|
||||||
|
ModelNode json = new ModelNode();
|
||||||
|
json.get(RealmDefinition.TAG_NAME).set(realmName);
|
||||||
|
|
||||||
|
// Realm values set first. Some can be overridden by deployment values.
|
||||||
|
if (realm != null) setJSONValues(json, realm);
|
||||||
|
setJSONValues(json, deployment);
|
||||||
|
return json.toJSONString(true);
|
||||||
|
}
|
||||||
|
|
||||||
private void setJSONValues(ModelNode json, ModelNode values) {
|
private void setJSONValues(ModelNode json, ModelNode values) {
|
||||||
for (Property prop : values.asPropertyList()) {
|
for (Property prop : new ArrayList<>(values.asPropertyList())) {
|
||||||
String name = prop.getName();
|
String name = prop.getName();
|
||||||
ModelNode value = prop.getValue();
|
ModelNode value = prop.getValue();
|
||||||
if (value.isDefined()) {
|
if (value.isDefined()) {
|
||||||
|
@ -258,6 +292,10 @@ public final class KeycloakAdapterConfigService {
|
||||||
return this.secureDeployments.containsKey(deploymentName);
|
return this.secureDeployments.containsKey(deploymentName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean isElytronEnabled(DeploymentUnit deploymentUnit) {
|
||||||
|
return elytronEnabledDeployments.contains(preferredDeploymentName(deploymentUnit));
|
||||||
|
}
|
||||||
|
|
||||||
private ModelNode getSecureDeployment(DeploymentUnit deploymentUnit) {
|
private ModelNode getSecureDeployment(DeploymentUnit deploymentUnit) {
|
||||||
String deploymentName = preferredDeploymentName(deploymentUnit);
|
String deploymentName = preferredDeploymentName(deploymentUnit);
|
||||||
return this.secureDeployments.containsKey(deploymentName)
|
return this.secureDeployments.containsKey(deploymentName)
|
||||||
|
|
|
@ -29,13 +29,14 @@ import org.jboss.modules.ModuleLoader;
|
||||||
*/
|
*/
|
||||||
public class KeycloakDependencyProcessorWildFly extends KeycloakDependencyProcessor {
|
public class KeycloakDependencyProcessorWildFly extends KeycloakDependencyProcessor {
|
||||||
|
|
||||||
|
private static final ModuleIdentifier KEYCLOAK_ELYTRON_ADAPTER = ModuleIdentifier.create("org.keycloak.keycloak-wildfly-elytron-oidc-adapter");
|
||||||
private static final ModuleIdentifier KEYCLOAK_WILDFLY_ADAPTER = ModuleIdentifier.create("org.keycloak.keycloak-wildfly-adapter");
|
private static final ModuleIdentifier KEYCLOAK_WILDFLY_ADAPTER = ModuleIdentifier.create("org.keycloak.keycloak-wildfly-adapter");
|
||||||
private static final ModuleIdentifier KEYCLOAK_UNDERTOW_ADAPTER = ModuleIdentifier.create("org.keycloak.keycloak-undertow-adapter");
|
private static final ModuleIdentifier KEYCLOAK_UNDERTOW_ADAPTER = ModuleIdentifier.create("org.keycloak.keycloak-undertow-adapter");
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void addPlatformSpecificModules(ModuleSpecification moduleSpecification, ModuleLoader moduleLoader) {
|
protected void addPlatformSpecificModules(ModuleSpecification moduleSpecification, ModuleLoader moduleLoader) {
|
||||||
// ModuleDependency(ModuleLoader moduleLoader, ModuleIdentifier identifier, boolean optional, boolean export, boolean importServices, boolean userSpecified)
|
|
||||||
moduleSpecification.addSystemDependency(new ModuleDependency(moduleLoader, KEYCLOAK_WILDFLY_ADAPTER, false, false, true, false));
|
moduleSpecification.addSystemDependency(new ModuleDependency(moduleLoader, KEYCLOAK_WILDFLY_ADAPTER, false, false, true, false));
|
||||||
moduleSpecification.addSystemDependency(new ModuleDependency(moduleLoader, KEYCLOAK_UNDERTOW_ADAPTER, false, false, false, false));
|
moduleSpecification.addSystemDependency(new ModuleDependency(moduleLoader, KEYCLOAK_UNDERTOW_ADAPTER, false, false, false, false));
|
||||||
|
moduleSpecification.addSystemDependency(new ModuleDependency(moduleLoader, KEYCLOAK_ELYTRON_ADAPTER, true, false, false, false));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,8 +47,9 @@ public class KeycloakExtension implements Extension {
|
||||||
private static final ResourceDefinition KEYCLOAK_SUBSYSTEM_RESOURCE = new KeycloakSubsystemDefinition();
|
private static final ResourceDefinition KEYCLOAK_SUBSYSTEM_RESOURCE = new KeycloakSubsystemDefinition();
|
||||||
static final RealmDefinition REALM_DEFINITION = new RealmDefinition();
|
static final RealmDefinition REALM_DEFINITION = new RealmDefinition();
|
||||||
static final SecureDeploymentDefinition SECURE_DEPLOYMENT_DEFINITION = new SecureDeploymentDefinition();
|
static final SecureDeploymentDefinition SECURE_DEPLOYMENT_DEFINITION = new SecureDeploymentDefinition();
|
||||||
|
static final SecureServerDefinition SECURE_SERVER_DEFINITION = new SecureServerDefinition();
|
||||||
static final CredentialDefinition CREDENTIAL_DEFINITION = new CredentialDefinition();
|
static final CredentialDefinition CREDENTIAL_DEFINITION = new CredentialDefinition();
|
||||||
static final RedirecRewritetRuleDefinition REDIRECT_RULE_DEFINITON = new RedirecRewritetRuleDefinition();
|
static final RedirecRewritetRuleDefinition REDIRECT_RULE_DEFINITON = new RedirecRewritetRuleDefinition();
|
||||||
|
|
||||||
public static StandardResourceDescriptionResolver getResourceDescriptionResolver(final String... keyPrefix) {
|
public static StandardResourceDescriptionResolver getResourceDescriptionResolver(final String... keyPrefix) {
|
||||||
StringBuilder prefix = new StringBuilder(SUBSYSTEM_NAME);
|
StringBuilder prefix = new StringBuilder(SUBSYSTEM_NAME);
|
||||||
|
@ -80,6 +81,10 @@ public class KeycloakExtension implements Extension {
|
||||||
secureDeploymentRegistration.registerSubModel(CREDENTIAL_DEFINITION);
|
secureDeploymentRegistration.registerSubModel(CREDENTIAL_DEFINITION);
|
||||||
secureDeploymentRegistration.registerSubModel(REDIRECT_RULE_DEFINITON);
|
secureDeploymentRegistration.registerSubModel(REDIRECT_RULE_DEFINITON);
|
||||||
|
|
||||||
|
ManagementResourceRegistration secureServerRegistration = registration.registerSubModel(SECURE_SERVER_DEFINITION);
|
||||||
|
secureServerRegistration.registerSubModel(CREDENTIAL_DEFINITION);
|
||||||
|
secureServerRegistration.registerSubModel(REDIRECT_RULE_DEFINITON);
|
||||||
|
|
||||||
subsystem.registerXMLElementWriter(PARSER);
|
subsystem.registerXMLElementWriter(PARSER);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,67 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source.
|
||||||
|
* Copyright 2016 Red Hat, Inc., and individual 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.keycloak.subsystem.adapter.extension;
|
||||||
|
|
||||||
|
import org.jboss.msc.service.Service;
|
||||||
|
import org.jboss.msc.service.StartContext;
|
||||||
|
import org.jboss.msc.service.StartException;
|
||||||
|
import org.jboss.msc.service.StopContext;
|
||||||
|
import org.jboss.msc.value.InjectedValue;
|
||||||
|
import org.keycloak.adapters.AdapterDeploymentContext;
|
||||||
|
import org.keycloak.adapters.KeycloakDeploymentBuilder;
|
||||||
|
import org.keycloak.adapters.elytron.KeycloakHttpServerAuthenticationMechanismFactory;
|
||||||
|
import org.wildfly.security.auth.server.SecurityDomain;
|
||||||
|
import org.wildfly.security.http.HttpServerAuthenticationMechanismFactory;
|
||||||
|
import org.wildfly.security.http.util.SetMechanismInformationMechanismFactory;
|
||||||
|
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author <a href="mailto:psilva@redhat.com">Pedro Igor</a>
|
||||||
|
*/
|
||||||
|
public class KeycloakHttpAuthenticationFactoryService implements Service<HttpServerAuthenticationMechanismFactory> {
|
||||||
|
|
||||||
|
private final String factoryName;
|
||||||
|
private HttpServerAuthenticationMechanismFactory httpAuthenticationFactory;
|
||||||
|
|
||||||
|
public KeycloakHttpAuthenticationFactoryService(String factoryName) {
|
||||||
|
this.factoryName = factoryName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void start(StartContext context) throws StartException {
|
||||||
|
KeycloakAdapterConfigService adapterConfigService = KeycloakAdapterConfigService.getInstance();
|
||||||
|
String config = adapterConfigService.getJSON(this.factoryName);
|
||||||
|
this.httpAuthenticationFactory = new KeycloakHttpServerAuthenticationMechanismFactory(createDeploymentContext(config.getBytes()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void stop(StopContext context) {
|
||||||
|
this.httpAuthenticationFactory = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public HttpServerAuthenticationMechanismFactory getValue() throws IllegalStateException, IllegalArgumentException {
|
||||||
|
return new SetMechanismInformationMechanismFactory(this.httpAuthenticationFactory);
|
||||||
|
}
|
||||||
|
|
||||||
|
private AdapterDeploymentContext createDeploymentContext(byte[] config) {
|
||||||
|
return new AdapterDeploymentContext(KeycloakDeploymentBuilder.build(new ByteArrayInputStream(config)));
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,117 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source.
|
||||||
|
* Copyright 2016 Red Hat, Inc., and individual 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.keycloak.subsystem.adapter.extension;
|
||||||
|
|
||||||
|
import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.OP_ADDR;
|
||||||
|
import static org.keycloak.subsystem.adapter.extension.KeycloakHttpServerAuthenticationMechanismFactoryDefinition.KeycloakHttpServerAuthenticationMechanismFactoryAddHandler.HTTP_SERVER_AUTHENTICATION_CAPABILITY;
|
||||||
|
|
||||||
|
import org.jboss.as.controller.OperationContext;
|
||||||
|
import org.jboss.as.controller.OperationFailedException;
|
||||||
|
import org.jboss.as.controller.PathAddress;
|
||||||
|
import org.jboss.as.controller.SimpleResourceDefinition;
|
||||||
|
import org.jboss.as.controller.capability.RuntimeCapability;
|
||||||
|
import org.jboss.dmr.ModelNode;
|
||||||
|
import org.jboss.msc.service.ServiceController;
|
||||||
|
import org.jboss.msc.service.ServiceName;
|
||||||
|
import org.wildfly.security.http.HttpServerAuthenticationMechanismFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link SimpleResourceDefinition} that can be used to configure a {@link org.keycloak.adapters.elytron.KeycloakHttpServerAuthenticationMechanismFactory}
|
||||||
|
* and expose it as a capability for other subsystems.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:psilva@redhat.com">Pedro Igor</a>
|
||||||
|
*/
|
||||||
|
class KeycloakHttpServerAuthenticationMechanismFactoryDefinition extends AbstractAdapterConfigurationDefinition {
|
||||||
|
|
||||||
|
static final String TAG_NAME = "http-server-mechanism-factory";
|
||||||
|
|
||||||
|
KeycloakHttpServerAuthenticationMechanismFactoryDefinition() {
|
||||||
|
this(TAG_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
KeycloakHttpServerAuthenticationMechanismFactoryDefinition(String tagName) {
|
||||||
|
super(tagName, ALL_ATTRIBUTES, new KeycloakHttpServerAuthenticationMechanismFactoryAddHandler(), new KeycloakHttpServerAuthenticationMechanismFactoryRemoveHandler(), new KeycloakHttpServerAuthenticationMechanismFactoryWriteHandler());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link AbstractAdapterConfigurationAddHandler} that exposes a {@link KeycloakHttpServerAuthenticationMechanismFactoryDefinition}
|
||||||
|
* as a capability through the installation of a {@link KeycloakHttpAuthenticationFactoryService}.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:psilva@redhat.com">Pedro Igor</a>
|
||||||
|
*/
|
||||||
|
static final class KeycloakHttpServerAuthenticationMechanismFactoryAddHandler extends AbstractAdapterConfigurationAddHandler {
|
||||||
|
|
||||||
|
static final String HTTP_SERVER_AUTHENTICATION_CAPABILITY = "org.wildfly.security.http-server-mechanism-factory";
|
||||||
|
static final RuntimeCapability<Void> HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY = RuntimeCapability
|
||||||
|
.Builder.of(HTTP_SERVER_AUTHENTICATION_CAPABILITY, true, HttpServerAuthenticationMechanismFactory.class)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
KeycloakHttpServerAuthenticationMechanismFactoryAddHandler() {
|
||||||
|
super(HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY, ALL_ATTRIBUTES);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
||||||
|
super.performRuntime(context, operation, model);
|
||||||
|
installCapability(context, operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void installCapability(OperationContext context, ModelNode operation) {
|
||||||
|
PathAddress pathAddress = PathAddress.pathAddress(operation.get(OP_ADDR));
|
||||||
|
String factoryName = pathAddress.getLastElement().getValue();
|
||||||
|
ServiceName serviceName = context.getCapabilityServiceName(HTTP_SERVER_AUTHENTICATION_CAPABILITY, factoryName, HttpServerAuthenticationMechanismFactory.class);
|
||||||
|
KeycloakHttpAuthenticationFactoryService service = new KeycloakHttpAuthenticationFactoryService(factoryName);
|
||||||
|
context.getServiceTarget().addService(serviceName, service).setInitialMode(ServiceController.Mode.ACTIVE).install();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link AbstractAdapterConfigurationRemoveHandler} that handles the removal of {@link KeycloakHttpServerAuthenticationMechanismFactoryDefinition}.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:psilva@redhat.com">Pedro Igor</a>
|
||||||
|
*/
|
||||||
|
static final class KeycloakHttpServerAuthenticationMechanismFactoryRemoveHandler extends AbstractAdapterConfigurationRemoveHandler {
|
||||||
|
@Override
|
||||||
|
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
||||||
|
super.performRuntime(context, operation, model);
|
||||||
|
PathAddress pathAddress = PathAddress.pathAddress(operation.get(OP_ADDR));
|
||||||
|
String factoryName = pathAddress.getLastElement().getValue();
|
||||||
|
ServiceName serviceName = context.getCapabilityServiceName(HTTP_SERVER_AUTHENTICATION_CAPABILITY, factoryName, HttpServerAuthenticationMechanismFactory.class);
|
||||||
|
|
||||||
|
context.removeService(serviceName);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void recoverServices(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
||||||
|
super.recoverServices(context, operation, model);
|
||||||
|
KeycloakHttpServerAuthenticationMechanismFactoryAddHandler.installCapability(context, operation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link AbstractAdapterConfigurationWriteAttributeHandler} that updates attributes on a {@link KeycloakHttpServerAuthenticationMechanismFactoryDefinition}.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:psilva@redhat.com">Pedro Igor</a>
|
||||||
|
*/
|
||||||
|
static final class KeycloakHttpServerAuthenticationMechanismFactoryWriteHandler extends AbstractAdapterConfigurationWriteAttributeHandler {
|
||||||
|
KeycloakHttpServerAuthenticationMechanismFactoryWriteHandler() {
|
||||||
|
super(ALL_ATTRIBUTES);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -62,6 +62,9 @@ class KeycloakSubsystemParser implements XMLStreamConstants, XMLElementReader<Li
|
||||||
else if (reader.getLocalName().equals(SecureDeploymentDefinition.TAG_NAME)) {
|
else if (reader.getLocalName().equals(SecureDeploymentDefinition.TAG_NAME)) {
|
||||||
readDeployment(reader, list);
|
readDeployment(reader, list);
|
||||||
}
|
}
|
||||||
|
else if (reader.getLocalName().equals(SecureServerDefinition.TAG_NAME)) {
|
||||||
|
readSecureServer(reader, list);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -89,27 +92,35 @@ class KeycloakSubsystemParser implements XMLStreamConstants, XMLElementReader<Li
|
||||||
}
|
}
|
||||||
|
|
||||||
private void readDeployment(XMLExtendedStreamReader reader, List<ModelNode> resourcesToAdd) throws XMLStreamException {
|
private void readDeployment(XMLExtendedStreamReader reader, List<ModelNode> resourcesToAdd) throws XMLStreamException {
|
||||||
|
readSecureResource(KeycloakExtension.SECURE_DEPLOYMENT_DEFINITION.TAG_NAME, KeycloakExtension.SECURE_DEPLOYMENT_DEFINITION, reader, resourcesToAdd);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void readSecureServer(XMLExtendedStreamReader reader, List<ModelNode> resourcesToAdd) throws XMLStreamException {
|
||||||
|
readSecureResource(KeycloakExtension.SECURE_SERVER_DEFINITION.TAG_NAME, KeycloakExtension.SECURE_SERVER_DEFINITION, reader, resourcesToAdd);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void readSecureResource(String tagName, AbstractAdapterConfigurationDefinition resource, XMLExtendedStreamReader reader, List<ModelNode> resourcesToAdd) throws XMLStreamException {
|
||||||
String name = readNameAttribute(reader);
|
String name = readNameAttribute(reader);
|
||||||
ModelNode addSecureDeployment = new ModelNode();
|
ModelNode addSecureDeployment = new ModelNode();
|
||||||
addSecureDeployment.get(ModelDescriptionConstants.OP).set(ModelDescriptionConstants.ADD);
|
addSecureDeployment.get(ModelDescriptionConstants.OP).set(ModelDescriptionConstants.ADD);
|
||||||
PathAddress addr = PathAddress.pathAddress(PathElement.pathElement(ModelDescriptionConstants.SUBSYSTEM, KeycloakExtension.SUBSYSTEM_NAME),
|
PathAddress addr = PathAddress.pathAddress(PathElement.pathElement(ModelDescriptionConstants.SUBSYSTEM, KeycloakExtension.SUBSYSTEM_NAME),
|
||||||
PathElement.pathElement(SecureDeploymentDefinition.TAG_NAME, name));
|
PathElement.pathElement(tagName, name));
|
||||||
addSecureDeployment.get(ModelDescriptionConstants.OP_ADDR).set(addr.toModelNode());
|
addSecureDeployment.get(ModelDescriptionConstants.OP_ADDR).set(addr.toModelNode());
|
||||||
List<ModelNode> credentialsToAdd = new ArrayList<ModelNode>();
|
List<ModelNode> credentialsToAdd = new ArrayList<ModelNode>();
|
||||||
List<ModelNode> redirectRulesToAdd = new ArrayList<ModelNode>();
|
List<ModelNode> redirectRulesToAdd = new ArrayList<ModelNode>();
|
||||||
while (reader.hasNext() && nextTag(reader) != END_ELEMENT) {
|
while (reader.hasNext() && nextTag(reader) != END_ELEMENT) {
|
||||||
String tagName = reader.getLocalName();
|
String localName = reader.getLocalName();
|
||||||
if (tagName.equals(CredentialDefinition.TAG_NAME)) {
|
if (localName.equals(CredentialDefinition.TAG_NAME)) {
|
||||||
readCredential(reader, addr, credentialsToAdd);
|
readCredential(reader, addr, credentialsToAdd);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (tagName.equals(RedirecRewritetRuleDefinition.TAG_NAME)) {
|
if (localName.equals(RedirecRewritetRuleDefinition.TAG_NAME)) {
|
||||||
readRewriteRule(reader, addr, redirectRulesToAdd);
|
readRewriteRule(reader, addr, redirectRulesToAdd);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
SimpleAttributeDefinition def = SecureDeploymentDefinition.lookup(tagName);
|
SimpleAttributeDefinition def = resource.lookup(localName);
|
||||||
if (def == null) throw new XMLStreamException("Unknown secure-deployment tag " + tagName);
|
if (def == null) throw new XMLStreamException("Unknown secure-deployment tag " + localName);
|
||||||
def.parseAndSetParameter(reader.getElementText(), addSecureDeployment, reader);
|
def.parseAndSetParameter(reader.getElementText(), addSecureDeployment, reader);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -236,6 +247,7 @@ class KeycloakSubsystemParser implements XMLStreamConstants, XMLElementReader<Li
|
||||||
context.startSubsystemElement(KeycloakExtension.NAMESPACE, false);
|
context.startSubsystemElement(KeycloakExtension.NAMESPACE, false);
|
||||||
writeRealms(writer, context);
|
writeRealms(writer, context);
|
||||||
writeSecureDeployments(writer, context);
|
writeSecureDeployments(writer, context);
|
||||||
|
writeSecureServers(writer, context);
|
||||||
writer.writeEndElement();
|
writer.writeEndElement();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -256,14 +268,22 @@ class KeycloakSubsystemParser implements XMLStreamConstants, XMLElementReader<Li
|
||||||
}
|
}
|
||||||
|
|
||||||
private void writeSecureDeployments(XMLExtendedStreamWriter writer, SubsystemMarshallingContext context) throws XMLStreamException {
|
private void writeSecureDeployments(XMLExtendedStreamWriter writer, SubsystemMarshallingContext context) throws XMLStreamException {
|
||||||
if (!context.getModelNode().get(SecureDeploymentDefinition.TAG_NAME).isDefined()) {
|
writeSecureResource(SecureDeploymentDefinition.TAG_NAME, SecureDeploymentDefinition.ALL_ATTRIBUTES, writer, context);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void writeSecureServers(XMLExtendedStreamWriter writer, SubsystemMarshallingContext context) throws XMLStreamException {
|
||||||
|
writeSecureResource(SecureServerDefinition.TAG_NAME, SecureServerDefinition.ALL_ATTRIBUTES, writer, context);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void writeSecureResource(String tagName, List<SimpleAttributeDefinition> attributes, XMLExtendedStreamWriter writer, SubsystemMarshallingContext context) throws XMLStreamException {
|
||||||
|
if (!context.getModelNode().get(tagName).isDefined()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
for (Property deployment : context.getModelNode().get(SecureDeploymentDefinition.TAG_NAME).asPropertyList()) {
|
for (Property deployment : context.getModelNode().get(tagName).asPropertyList()) {
|
||||||
writer.writeStartElement(SecureDeploymentDefinition.TAG_NAME);
|
writer.writeStartElement(tagName);
|
||||||
writer.writeAttribute("name", deployment.getName());
|
writer.writeAttribute("name", deployment.getName());
|
||||||
ModelNode deploymentElements = deployment.getValue();
|
ModelNode deploymentElements = deployment.getValue();
|
||||||
for (AttributeDefinition element : SecureDeploymentDefinition.ALL_ATTRIBUTES) {
|
for (AttributeDefinition element : attributes) {
|
||||||
element.marshallAsElement(deploymentElements, writer);
|
element.marshallAsElement(deploymentElements, writer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -271,7 +291,7 @@ class KeycloakSubsystemParser implements XMLStreamConstants, XMLElementReader<Li
|
||||||
if (credentials.isDefined()) {
|
if (credentials.isDefined()) {
|
||||||
writeCredentials(writer, credentials);
|
writeCredentials(writer, credentials);
|
||||||
}
|
}
|
||||||
|
|
||||||
ModelNode redirectRewriteRule = deploymentElements.get(RedirecRewritetRuleDefinition.TAG_NAME);
|
ModelNode redirectRewriteRule = deploymentElements.get(RedirecRewritetRuleDefinition.TAG_NAME);
|
||||||
if (redirectRewriteRule.isDefined()) {
|
if (redirectRewriteRule.isDefined()) {
|
||||||
writeRedirectRules(writer, redirectRewriteRule);
|
writeRedirectRules(writer, redirectRewriteRule);
|
||||||
|
|
|
@ -16,138 +16,88 @@
|
||||||
*/
|
*/
|
||||||
package org.keycloak.subsystem.adapter.extension;
|
package org.keycloak.subsystem.adapter.extension;
|
||||||
|
|
||||||
import org.jboss.as.controller.AttributeDefinition;
|
import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.OP_ADDR;
|
||||||
import org.jboss.as.controller.PathElement;
|
|
||||||
import org.jboss.as.controller.SimpleAttributeDefinition;
|
|
||||||
import org.jboss.as.controller.SimpleAttributeDefinitionBuilder;
|
|
||||||
import org.jboss.as.controller.SimpleResourceDefinition;
|
|
||||||
import org.jboss.as.controller.operations.common.GenericSubsystemDescribeHandler;
|
|
||||||
import org.jboss.as.controller.operations.validation.IntRangeValidator;
|
|
||||||
import org.jboss.as.controller.operations.validation.StringLengthValidator;
|
|
||||||
import org.jboss.as.controller.registry.ManagementResourceRegistration;
|
|
||||||
import org.jboss.dmr.ModelNode;
|
|
||||||
import org.jboss.dmr.ModelType;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import org.jboss.as.controller.OperationContext;
|
||||||
import java.util.HashMap;
|
import org.jboss.as.controller.OperationFailedException;
|
||||||
import java.util.List;
|
import org.jboss.as.controller.PathAddress;
|
||||||
import java.util.Map;
|
import org.jboss.as.controller.capability.RuntimeCapability;
|
||||||
|
import org.jboss.dmr.ModelNode;
|
||||||
|
import org.jboss.msc.service.ServiceController;
|
||||||
|
import org.jboss.msc.service.ServiceName;
|
||||||
|
import org.jboss.msc.service.ServiceTarget;
|
||||||
|
import org.wildfly.security.http.HttpServerAuthenticationMechanismFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines attributes and operations for a secure-deployment.
|
* Defines attributes and operations for a secure-deployment.
|
||||||
*
|
*
|
||||||
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
*/
|
*/
|
||||||
public class SecureDeploymentDefinition extends SimpleResourceDefinition {
|
final class SecureDeploymentDefinition extends AbstractAdapterConfigurationDefinition {
|
||||||
|
|
||||||
public static final String TAG_NAME = "secure-deployment";
|
static final String TAG_NAME = "secure-deployment";
|
||||||
|
|
||||||
protected static final SimpleAttributeDefinition REALM =
|
|
||||||
new SimpleAttributeDefinitionBuilder("realm", ModelType.STRING, true)
|
|
||||||
.setXmlName("realm")
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.setValidator(new StringLengthValidator(1, Integer.MAX_VALUE, true, true))
|
|
||||||
.build();
|
|
||||||
protected static final SimpleAttributeDefinition RESOURCE =
|
|
||||||
new SimpleAttributeDefinitionBuilder("resource", ModelType.STRING, true)
|
|
||||||
.setXmlName("resource")
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.setValidator(new StringLengthValidator(1, Integer.MAX_VALUE, true, true))
|
|
||||||
.build();
|
|
||||||
protected static final SimpleAttributeDefinition USE_RESOURCE_ROLE_MAPPINGS =
|
|
||||||
new SimpleAttributeDefinitionBuilder("use-resource-role-mappings", ModelType.BOOLEAN, true)
|
|
||||||
.setXmlName("use-resource-role-mappings")
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.setDefaultValue(new ModelNode(false))
|
|
||||||
.build();
|
|
||||||
protected static final SimpleAttributeDefinition BEARER_ONLY =
|
|
||||||
new SimpleAttributeDefinitionBuilder("bearer-only", ModelType.BOOLEAN, true)
|
|
||||||
.setXmlName("bearer-only")
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.setDefaultValue(new ModelNode(false))
|
|
||||||
.build();
|
|
||||||
protected static final SimpleAttributeDefinition ENABLE_BASIC_AUTH =
|
|
||||||
new SimpleAttributeDefinitionBuilder("enable-basic-auth", ModelType.BOOLEAN, true)
|
|
||||||
.setXmlName("enable-basic-auth")
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.setDefaultValue(new ModelNode(false))
|
|
||||||
.build();
|
|
||||||
protected static final SimpleAttributeDefinition PUBLIC_CLIENT =
|
|
||||||
new SimpleAttributeDefinitionBuilder("public-client", ModelType.BOOLEAN, true)
|
|
||||||
.setXmlName("public-client")
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.setDefaultValue(new ModelNode(false))
|
|
||||||
.build();
|
|
||||||
protected static final SimpleAttributeDefinition TURN_OFF_CHANGE_SESSION =
|
|
||||||
new SimpleAttributeDefinitionBuilder("turn-off-change-session-id-on-login", ModelType.BOOLEAN, true)
|
|
||||||
.setXmlName("turn-off-change-session-id-on-login")
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.setDefaultValue(new ModelNode(false))
|
|
||||||
.build();
|
|
||||||
protected static final SimpleAttributeDefinition TOKEN_MINIMUM_TIME_TO_LIVE =
|
|
||||||
new SimpleAttributeDefinitionBuilder("token-minimum-time-to-live", ModelType.INT, true)
|
|
||||||
.setXmlName("token-minimum-time-to-live")
|
|
||||||
.setValidator(new IntRangeValidator(-1, true))
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.build();
|
|
||||||
protected static final SimpleAttributeDefinition MIN_TIME_BETWEEN_JWKS_REQUESTS =
|
|
||||||
new SimpleAttributeDefinitionBuilder("min-time-between-jwks-requests", ModelType.INT, true)
|
|
||||||
.setXmlName("min-time-between-jwks-requests")
|
|
||||||
.setValidator(new IntRangeValidator(-1, true))
|
|
||||||
.setAllowExpression(true)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
|
|
||||||
protected static final List<SimpleAttributeDefinition> DEPLOYMENT_ONLY_ATTRIBUTES = new ArrayList<SimpleAttributeDefinition>();
|
|
||||||
static {
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(REALM);
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(RESOURCE);
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(USE_RESOURCE_ROLE_MAPPINGS);
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(BEARER_ONLY);
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(ENABLE_BASIC_AUTH);
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(PUBLIC_CLIENT);
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(TURN_OFF_CHANGE_SESSION);
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(TOKEN_MINIMUM_TIME_TO_LIVE);
|
|
||||||
DEPLOYMENT_ONLY_ATTRIBUTES.add(MIN_TIME_BETWEEN_JWKS_REQUESTS);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected static final List<SimpleAttributeDefinition> ALL_ATTRIBUTES = new ArrayList<SimpleAttributeDefinition>();
|
|
||||||
static {
|
|
||||||
ALL_ATTRIBUTES.addAll(DEPLOYMENT_ONLY_ATTRIBUTES);
|
|
||||||
ALL_ATTRIBUTES.addAll(SharedAttributeDefinitons.ATTRIBUTES);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static final Map<String, SimpleAttributeDefinition> DEFINITION_LOOKUP = new HashMap<String, SimpleAttributeDefinition>();
|
|
||||||
static {
|
|
||||||
for (SimpleAttributeDefinition def : ALL_ATTRIBUTES) {
|
|
||||||
DEFINITION_LOOKUP.put(def.getXmlName(), def);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static SecureDeploymentWriteAttributeHandler attrHandler = new SecureDeploymentWriteAttributeHandler(ALL_ATTRIBUTES);
|
|
||||||
|
|
||||||
public SecureDeploymentDefinition() {
|
public SecureDeploymentDefinition() {
|
||||||
super(PathElement.pathElement(TAG_NAME),
|
super(TAG_NAME, ALL_ATTRIBUTES, new SecureDeploymentAddHandler(), new SecureDeploymentRemoveHandler(), new SecureDeploymentWriteAttributeHandler());
|
||||||
KeycloakExtension.getResourceDescriptionResolver(TAG_NAME),
|
|
||||||
SecureDeploymentAddHandler.INSTANCE,
|
|
||||||
SecureDeploymentRemoveHandler.INSTANCE);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
/**
|
||||||
public void registerOperations(ManagementResourceRegistration resourceRegistration) {
|
* Add a deployment to a realm.
|
||||||
super.registerOperations(resourceRegistration);
|
*
|
||||||
resourceRegistration.registerOperationHandler(GenericSubsystemDescribeHandler.DEFINITION, GenericSubsystemDescribeHandler.INSTANCE);
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
}
|
*/
|
||||||
|
static final class SecureDeploymentAddHandler extends AbstractAdapterConfigurationAddHandler {
|
||||||
|
|
||||||
@Override
|
static final String HTTP_SERVER_AUTHENTICATION_CAPABILITY = "org.wildfly.security.http-server-mechanism-factory";
|
||||||
public void registerAttributes(ManagementResourceRegistration resourceRegistration) {
|
static RuntimeCapability<Void> HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY;
|
||||||
super.registerAttributes(resourceRegistration);
|
|
||||||
for (AttributeDefinition attrDef : ALL_ATTRIBUTES) {
|
static {
|
||||||
resourceRegistration.registerReadWriteAttribute(attrDef, null, attrHandler);
|
try {
|
||||||
|
HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY = RuntimeCapability
|
||||||
|
.Builder.of(HTTP_SERVER_AUTHENTICATION_CAPABILITY, true, HttpServerAuthenticationMechanismFactory.class)
|
||||||
|
.build();
|
||||||
|
} catch (NoClassDefFoundError ncfe) {
|
||||||
|
// ignore, Elytron not present thus no capability will be published by this resource definition
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SecureDeploymentAddHandler() {
|
||||||
|
super(HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY, ALL_ATTRIBUTES);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
||||||
|
super.performRuntime(context, operation, model);
|
||||||
|
if (HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY != null) {
|
||||||
|
installCapability(context, operation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void installCapability(OperationContext context, ModelNode operation) {
|
||||||
|
PathAddress pathAddress = PathAddress.pathAddress(operation.get(OP_ADDR));
|
||||||
|
String factoryName = pathAddress.getLastElement().getValue();
|
||||||
|
ServiceName serviceName = context.getCapabilityServiceName(HTTP_SERVER_AUTHENTICATION_CAPABILITY, factoryName, HttpServerAuthenticationMechanismFactory.class);
|
||||||
|
KeycloakHttpAuthenticationFactoryService service = new KeycloakHttpAuthenticationFactoryService(factoryName);
|
||||||
|
ServiceTarget serviceTarget = context.getServiceTarget();
|
||||||
|
serviceTarget.addService(serviceName, service).setInitialMode(ServiceController.Mode.ACTIVE).install();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static SimpleAttributeDefinition lookup(String name) {
|
/**
|
||||||
return DEFINITION_LOOKUP.get(name);
|
* Remove a secure-deployment from a realm.
|
||||||
|
*
|
||||||
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
|
*/
|
||||||
|
static final class SecureDeploymentRemoveHandler extends AbstractAdapterConfigurationRemoveHandler {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update an attribute on a secure-deployment.
|
||||||
|
*
|
||||||
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
|
*/
|
||||||
|
static final class SecureDeploymentWriteAttributeHandler extends AbstractAdapterConfigurationWriteAttributeHandler {
|
||||||
|
SecureDeploymentWriteAttributeHandler() {
|
||||||
|
super(ALL_ATTRIBUTES);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,258 @@
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
package org.keycloak.subsystem.adapter.extension;
|
||||||
|
|
||||||
|
import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.OP_ADDR;
|
||||||
|
import static org.keycloak.subsystem.adapter.extension.KeycloakHttpServerAuthenticationMechanismFactoryDefinition.KeycloakHttpServerAuthenticationMechanismFactoryAddHandler.HTTP_SERVER_AUTHENTICATION_CAPABILITY;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import io.undertow.io.IoCallback;
|
||||||
|
import io.undertow.io.Sender;
|
||||||
|
import io.undertow.server.HttpServerExchange;
|
||||||
|
import io.undertow.server.handlers.resource.Resource;
|
||||||
|
import io.undertow.server.handlers.resource.ResourceChangeListener;
|
||||||
|
import io.undertow.server.handlers.resource.ResourceManager;
|
||||||
|
import io.undertow.util.ETag;
|
||||||
|
import io.undertow.util.MimeMappings;
|
||||||
|
import org.jboss.as.controller.OperationContext;
|
||||||
|
import org.jboss.as.controller.OperationFailedException;
|
||||||
|
import org.jboss.as.controller.PathAddress;
|
||||||
|
import org.jboss.as.controller.capability.RuntimeCapability;
|
||||||
|
import org.jboss.as.server.mgmt.domain.ExtensibleHttpManagement;
|
||||||
|
import org.jboss.dmr.ModelNode;
|
||||||
|
import org.jboss.msc.service.Service;
|
||||||
|
import org.jboss.msc.service.ServiceController.Mode;
|
||||||
|
import org.jboss.msc.service.ServiceName;
|
||||||
|
import org.jboss.msc.service.ServiceTarget;
|
||||||
|
import org.jboss.msc.service.StartContext;
|
||||||
|
import org.jboss.msc.service.StartException;
|
||||||
|
import org.jboss.msc.service.StopContext;
|
||||||
|
import org.jboss.msc.value.InjectedValue;
|
||||||
|
import org.wildfly.security.http.HttpServerAuthenticationMechanismFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines attributes and operations for a secure-deployment.
|
||||||
|
*
|
||||||
|
* @author Stan Silvert ssilvert@redhat.com (C) 2013 Red Hat Inc.
|
||||||
|
*/
|
||||||
|
final class SecureServerDefinition extends AbstractAdapterConfigurationDefinition {
|
||||||
|
|
||||||
|
public static final String TAG_NAME = "secure-server";
|
||||||
|
|
||||||
|
SecureServerDefinition() {
|
||||||
|
super(TAG_NAME, ALL_ATTRIBUTES, new SecureServerAddHandler(), new SecureServerRemoveHandler(), new SecureServerWriteHandler());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link AbstractAdapterConfigurationAddHandler} that exposes a {@link SecureServerDefinition}
|
||||||
|
* as a capability through the installation of a {@link KeycloakHttpAuthenticationFactoryService}.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:psilva@redhat.com">Pedro Igor</a>
|
||||||
|
*/
|
||||||
|
static final class SecureServerAddHandler extends AbstractAdapterConfigurationAddHandler {
|
||||||
|
|
||||||
|
static final String HTTP_SERVER_AUTHENTICATION_CAPABILITY = "org.wildfly.security.http-server-mechanism-factory";
|
||||||
|
static final String HTTP_MANAGEMENT_HTTP_EXTENSIBLE_CAPABILITY = "org.wildfly.management.http.extensible";
|
||||||
|
static RuntimeCapability<Void> HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY;
|
||||||
|
|
||||||
|
static {
|
||||||
|
try {
|
||||||
|
HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY = RuntimeCapability
|
||||||
|
.Builder.of(HTTP_SERVER_AUTHENTICATION_CAPABILITY, true, HttpServerAuthenticationMechanismFactory.class)
|
||||||
|
.build();
|
||||||
|
} catch (NoClassDefFoundError ncfe) {
|
||||||
|
// ignore, Elytron not present thus no capability will be published by this resource definition
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SecureServerAddHandler() {
|
||||||
|
super(HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY, ALL_ATTRIBUTES);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
||||||
|
super.performRuntime(context, operation, model);
|
||||||
|
if (HTTP_SERVER_AUTHENTICATION_RUNTIME_CAPABILITY != null) {
|
||||||
|
installCapability(context, operation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void installCapability(OperationContext context, ModelNode operation) throws OperationFailedException {
|
||||||
|
PathAddress pathAddress = PathAddress.pathAddress(operation.get(OP_ADDR));
|
||||||
|
String factoryName = pathAddress.getLastElement().getValue();
|
||||||
|
ServiceName serviceName = context.getCapabilityServiceName(HTTP_SERVER_AUTHENTICATION_CAPABILITY, factoryName, HttpServerAuthenticationMechanismFactory.class);
|
||||||
|
boolean publicClient = SecureServerDefinition.PUBLIC_CLIENT.resolveModelAttribute(context, operation).asBoolean(false);
|
||||||
|
|
||||||
|
if (!publicClient) {
|
||||||
|
throw new OperationFailedException("Only public clients are allowed to have their configuration exposed through the management interface");
|
||||||
|
}
|
||||||
|
|
||||||
|
KeycloakHttpAuthenticationFactoryService service = new KeycloakHttpAuthenticationFactoryService(factoryName);
|
||||||
|
ServiceTarget serviceTarget = context.getServiceTarget();
|
||||||
|
InjectedValue<ExtensibleHttpManagement> injectedValue = new InjectedValue<>();
|
||||||
|
serviceTarget.addService(serviceName.append("http-management-context"), createHttpManagementConfigContextService(factoryName, injectedValue))
|
||||||
|
.addDependency(context.getCapabilityServiceName(HTTP_MANAGEMENT_HTTP_EXTENSIBLE_CAPABILITY, ExtensibleHttpManagement.class), ExtensibleHttpManagement.class, injectedValue).setInitialMode(Mode.ACTIVE).install();
|
||||||
|
serviceTarget.addService(serviceName, service).setInitialMode(Mode.ACTIVE).install();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link AbstractAdapterConfigurationRemoveHandler} that handles the removal of {@link SecureServerDefinition}.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:psilva@redhat.com">Pedro Igor</a>
|
||||||
|
*/
|
||||||
|
static final class SecureServerRemoveHandler extends AbstractAdapterConfigurationRemoveHandler {
|
||||||
|
@Override
|
||||||
|
protected void performRuntime(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
||||||
|
super.performRuntime(context, operation, model);
|
||||||
|
PathAddress pathAddress = PathAddress.pathAddress(operation.get(OP_ADDR));
|
||||||
|
String factoryName = pathAddress.getLastElement().getValue();
|
||||||
|
ServiceName serviceName = context.getCapabilityServiceName(HTTP_SERVER_AUTHENTICATION_CAPABILITY, factoryName, HttpServerAuthenticationMechanismFactory.class);
|
||||||
|
context.removeService(serviceName);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void recoverServices(OperationContext context, ModelNode operation, ModelNode model) throws OperationFailedException {
|
||||||
|
super.recoverServices(context, operation, model);
|
||||||
|
SecureServerDefinition.SecureServerAddHandler.installCapability(context, operation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link AbstractAdapterConfigurationWriteAttributeHandler} that updates attributes on a {@link SecureServerDefinition}.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:psilva@redhat.com">Pedro Igor</a>
|
||||||
|
*/
|
||||||
|
static final class SecureServerWriteHandler extends AbstractAdapterConfigurationWriteAttributeHandler {
|
||||||
|
SecureServerWriteHandler() {
|
||||||
|
super(ALL_ATTRIBUTES);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Service<Void> createHttpManagementConfigContextService(final String factoryName, final InjectedValue<ExtensibleHttpManagement> httpConfigContext) {
|
||||||
|
final String contextName = "/keycloak/adapter/" + factoryName + "/";
|
||||||
|
return new Service<Void>() {
|
||||||
|
public void start(StartContext startContext) throws StartException {
|
||||||
|
ExtensibleHttpManagement extensibleHttpManagement = (ExtensibleHttpManagement)httpConfigContext.getValue();
|
||||||
|
extensibleHttpManagement.addStaticContext(contextName, new ResourceManager() {
|
||||||
|
public Resource getResource(final String path) throws IOException {
|
||||||
|
KeycloakAdapterConfigService adapterConfigService = KeycloakAdapterConfigService.getInstance();
|
||||||
|
final String config = adapterConfigService.getJSON(factoryName);
|
||||||
|
|
||||||
|
if (config == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Resource() {
|
||||||
|
public String getPath() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Date getLastModified() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getLastModifiedString() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ETag getETag() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isDirectory() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Resource> list() {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getContentType(MimeMappings mimeMappings) {
|
||||||
|
return "application/json";
|
||||||
|
}
|
||||||
|
|
||||||
|
public void serve(Sender sender, HttpServerExchange exchange, IoCallback completionCallback) {
|
||||||
|
sender.send(config);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getContentLength() {
|
||||||
|
return Long.valueOf((long)config.length());
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCacheKey() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public File getFile() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Path getFilePath() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public File getResourceManagerRoot() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Path getResourceManagerRootPath() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public URL getUrl() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isResourceChangeListenerSupported() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void registerResourceChangeListener(ResourceChangeListener listener) {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void removeResourceChangeListener(ResourceChangeListener listener) {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void close() throws IOException {
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void stop(StopContext stopContext) {
|
||||||
|
((ExtensibleHttpManagement)httpConfigContext.getValue()).removeContext(contextName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Void getValue() throws IllegalStateException, IllegalArgumentException {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
|
@ -20,6 +20,8 @@ keycloak.subsystem.add=Operation Adds Keycloak adapter subsystem
|
||||||
keycloak.subsystem.remove=Operation removes Keycloak adapter subsystem
|
keycloak.subsystem.remove=Operation removes Keycloak adapter subsystem
|
||||||
keycloak.subsystem.realm=A Keycloak realm.
|
keycloak.subsystem.realm=A Keycloak realm.
|
||||||
keycloak.subsystem.secure-deployment=A deployment secured by Keycloak.
|
keycloak.subsystem.secure-deployment=A deployment secured by Keycloak.
|
||||||
|
keycloak.subsystem.secure-server=A configuration exposed to the server.
|
||||||
|
keycloak.subsystem.http-server-mechanism-factory=A http-server-mechanism-factory exposed to the server.
|
||||||
|
|
||||||
keycloak.realm=A Keycloak realm.
|
keycloak.realm=A Keycloak realm.
|
||||||
keycloak.realm.add=Add a realm definition to the subsystem.
|
keycloak.realm.add=Add a realm definition to the subsystem.
|
||||||
|
@ -90,7 +92,48 @@ keycloak.secure-deployment.token-minimum-time-to-live=The adapter will refresh t
|
||||||
keycloak.secure-deployment.min-time-between-jwks-requests=If adapter recognize token signed by unknown public key, it will try to download new public key from keycloak server. However it won't try to download if already tried it in less than 'min-time-between-jwks-requests' seconds
|
keycloak.secure-deployment.min-time-between-jwks-requests=If adapter recognize token signed by unknown public key, it will try to download new public key from keycloak server. However it won't try to download if already tried it in less than 'min-time-between-jwks-requests' seconds
|
||||||
keycloak.secure-deployment.ignore-oauth-query-parameter=disable query parameter parsing for access_token
|
keycloak.secure-deployment.ignore-oauth-query-parameter=disable query parameter parsing for access_token
|
||||||
|
|
||||||
|
keycloak.secure-server=A deployment secured by Keycloak
|
||||||
|
keycloak.secure-server.add=Add a deployment to be secured by Keycloak
|
||||||
|
keycloak.secure-server.realm=Keycloak realm
|
||||||
|
keycloak.secure-server.remove=Remove a deployment to be secured by Keycloak
|
||||||
|
keycloak.secure-server.realm-public-key=Public key of the realm
|
||||||
|
keycloak.secure-server.auth-server-url=Base URL of the Realm Auth Server
|
||||||
|
keycloak.secure-server.disable-trust-manager=Adapter will not use a trust manager when making adapter HTTPS requests
|
||||||
|
keycloak.secure-server.ssl-required=Specify if SSL is required (valid values are all, external and none)
|
||||||
|
keycloak.secure-server.allow-any-hostname=SSL Setting
|
||||||
|
keycloak.secure-server.truststore=Truststore used for adapter client HTTPS requests
|
||||||
|
keycloak.secure-server.truststore-password=Password of the Truststore
|
||||||
|
keycloak.secure-server.connection-pool-size=Connection pool size for the client used by the adapter
|
||||||
|
keycloak.secure-server.resource=Application name
|
||||||
|
keycloak.secure-server.use-resource-role-mappings=Use resource level permissions from token
|
||||||
|
keycloak.secure-server.credentials=Adapter credentials
|
||||||
|
keycloak.secure-server.redirect-rewrite-rule=Apply a rewrite rule for the redirect URI
|
||||||
|
keycloak.secure-server.bearer-only=Bearer Token Auth only
|
||||||
|
keycloak.secure-server.enable-basic-auth=Enable Basic Authentication
|
||||||
|
keycloak.secure-server.public-client=Public client
|
||||||
|
keycloak.secure-server.enable-cors=Enable Keycloak CORS support
|
||||||
|
keycloak.secure-server.autodetect-bearer-only=autodetect bearer-only requests
|
||||||
|
keycloak.secure-server.client-keystore=n/a
|
||||||
|
keycloak.secure-server.client-keystore-password=n/a
|
||||||
|
keycloak.secure-server.client-key-password=n/a
|
||||||
|
keycloak.secure-server.cors-max-age=CORS max-age header
|
||||||
|
keycloak.secure-server.cors-allowed-headers=CORS allowed headers
|
||||||
|
keycloak.secure-server.cors-allowed-methods=CORS allowed methods
|
||||||
|
keycloak.secure-server.cors-exposed-headers=CORS exposed headers
|
||||||
|
keycloak.secure-server.expose-token=Enable secure URL that exposes access token
|
||||||
|
keycloak.secure-server.auth-server-url-for-backend-requests=URL to use to make background calls to auth server
|
||||||
|
keycloak.secure-server.always-refresh-token=Refresh token on every single web request
|
||||||
|
keycloak.secure-server.register-node-at-startup=Cluster setting
|
||||||
|
keycloak.secure-server.register-node-period=how often to re-register node
|
||||||
|
keycloak.secure-server.token-store=cookie or session storage for auth session data
|
||||||
|
keycloak.secure-server.principal-attribute=token attribute to use to set Principal name
|
||||||
|
keycloak.secure-server.turn-off-change-session-id-on-login=The session id is changed by default on a successful login. Change this to true if you want to turn this off
|
||||||
|
keycloak.secure-server.token-minimum-time-to-live=The adapter will refresh the token if the current token is expired OR will expire in 'token-minimum-time-to-live' seconds or less
|
||||||
|
keycloak.secure-server.min-time-between-jwks-requests=If adapter recognize token signed by unknown public key, it will try to download new public key from keycloak server. However it won't try to download if already tried it in less than 'min-time-between-jwks-requests' seconds
|
||||||
|
keycloak.secure-server.ignore-oauth-query-parameter=disable query parameter parsing for access_token
|
||||||
|
|
||||||
keycloak.secure-deployment.credential=Credential value
|
keycloak.secure-deployment.credential=Credential value
|
||||||
|
keycloak.secure-server.credential=Credential value
|
||||||
|
|
||||||
keycloak.credential=Credential
|
keycloak.credential=Credential
|
||||||
keycloak.credential.value=Credential value
|
keycloak.credential.value=Credential value
|
||||||
|
|
|
@ -38,6 +38,7 @@
|
||||||
<xs:choice minOccurs="0" maxOccurs="unbounded">
|
<xs:choice minOccurs="0" maxOccurs="unbounded">
|
||||||
<xs:element name="realm" maxOccurs="unbounded" minOccurs="0" type="realm-type"/>
|
<xs:element name="realm" maxOccurs="unbounded" minOccurs="0" type="realm-type"/>
|
||||||
<xs:element name="secure-deployment" maxOccurs="unbounded" minOccurs="0" type="secure-deployment-type"/>
|
<xs:element name="secure-deployment" maxOccurs="unbounded" minOccurs="0" type="secure-deployment-type"/>
|
||||||
|
<xs:element name="secure-server" maxOccurs="unbounded" minOccurs="0" type="secure-deployment-type"/>
|
||||||
</xs:choice>
|
</xs:choice>
|
||||||
</xs:complexType>
|
</xs:complexType>
|
||||||
|
|
||||||
|
@ -84,7 +85,7 @@
|
||||||
<xs:element name="allow-any-hostname" type="xs:boolean" minOccurs="0" maxOccurs="1" />
|
<xs:element name="allow-any-hostname" type="xs:boolean" minOccurs="0" maxOccurs="1" />
|
||||||
<xs:element name="use-resource-role-mappings" type="xs:boolean" minOccurs="0" maxOccurs="1" />
|
<xs:element name="use-resource-role-mappings" type="xs:boolean" minOccurs="0" maxOccurs="1" />
|
||||||
<xs:element name="cors-max-age" type="xs:integer" minOccurs="0" maxOccurs="1"/>
|
<xs:element name="cors-max-age" type="xs:integer" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="auth-server-url" type="xs:string" minOccurs="1" maxOccurs="1"/>
|
<xs:element name="auth-server-url" type="xs:string" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="realm" type="xs:string" minOccurs="1" maxOccurs="1"/>
|
<xs:element name="realm" type="xs:string" minOccurs="1" maxOccurs="1"/>
|
||||||
<xs:element name="disable-trust-manager" type="xs:boolean" minOccurs="0" maxOccurs="1" />
|
<xs:element name="disable-trust-manager" type="xs:boolean" minOccurs="0" maxOccurs="1" />
|
||||||
<xs:element name="cors-allowed-methods" type="xs:string" minOccurs="0" maxOccurs="1"/>
|
<xs:element name="cors-allowed-methods" type="xs:string" minOccurs="0" maxOccurs="1"/>
|
||||||
|
@ -99,9 +100,9 @@
|
||||||
<xs:element name="connection-pool-size" type="xs:integer" minOccurs="0" maxOccurs="1"/>
|
<xs:element name="connection-pool-size" type="xs:integer" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="expose-token" type="xs:boolean" minOccurs="0" maxOccurs="1"/>
|
<xs:element name="expose-token" type="xs:boolean" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="ssl-required" type="xs:string" minOccurs="0" maxOccurs="1" />
|
<xs:element name="ssl-required" type="xs:string" minOccurs="0" maxOccurs="1" />
|
||||||
<xs:element name="realm-public-key" type="xs:string" minOccurs="1" maxOccurs="1"/>
|
<xs:element name="realm-public-key" type="xs:string" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="credential" type="credential-type" minOccurs="1" maxOccurs="1"/>
|
<xs:element name="credential" type="credential-type" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="redirect-rewrite-rule" type="redirect-rewrite-rule-type" minOccurs="1" maxOccurs="1"/>
|
<xs:element name="redirect-rewrite-rule" type="redirect-rewrite-rule-type" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="auth-server-url-for-backend-requests" type="xs:string" minOccurs="0" maxOccurs="1"/>
|
<xs:element name="auth-server-url-for-backend-requests" type="xs:string" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="always-refresh-token" type="xs:boolean" minOccurs="0" maxOccurs="1"/>
|
<xs:element name="always-refresh-token" type="xs:boolean" minOccurs="0" maxOccurs="1"/>
|
||||||
<xs:element name="register-node-at-startup" type="xs:boolean" minOccurs="0" maxOccurs="1"/>
|
<xs:element name="register-node-at-startup" type="xs:boolean" minOccurs="0" maxOccurs="1"/>
|
||||||
|
|
|
@ -73,7 +73,7 @@ public class SubsystemParsingTestCase extends AbstractSubsystemBaseTest {
|
||||||
ModelNode deployment = new ModelNode();
|
ModelNode deployment = new ModelNode();
|
||||||
deployment.get("realm").set("demo");
|
deployment.get("realm").set("demo");
|
||||||
deployment.get("resource").set("customer-portal");
|
deployment.get("resource").set("customer-portal");
|
||||||
service.addSecureDeployment(deploymentOp, deployment);
|
service.addSecureDeployment(deploymentOp, deployment, false);
|
||||||
|
|
||||||
addCredential(addr, service, "secret", "secret1");
|
addCredential(addr, service, "secret", "secret1");
|
||||||
addCredential(addr, service, "jwt.client-keystore-file", "/tmp/foo.jks");
|
addCredential(addr, service, "jwt.client-keystore-file", "/tmp/foo.jks");
|
||||||
|
|
|
@ -40,6 +40,10 @@
|
||||||
<token-store>session</token-store>
|
<token-store>session</token-store>
|
||||||
<principal-attribute>sub</principal-attribute>
|
<principal-attribute>sub</principal-attribute>
|
||||||
</realm>
|
</realm>
|
||||||
|
<realm name="jboss-infra">
|
||||||
|
<realm-public-key>MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAqKoq+a9MgXepmsPJDmo45qswuChW9pWjanX68oIBuI4hGvhQxFHryCow230A+sr7tFdMQMt8f1l/ysmV/fYAuW29WaoY4kI4Ou1yYPuwywKSsxT6PooTs83hKyZ1h4LZMj5DkLGDDDyVRHob2WmPaYg9RGVRw3iGGsD/p+Yb+L/gnBYQnZZ7lYqmN7h36p5CkzzlgXQA1Ha8sQxL+rJNH8+sZm0vBrKsoII3Of7TqHGsm1RwFV3XCuGJ7S61AbjJMXL5DQgJl9Z5scvxGAyoRLKC294UgMnQdzyBTMPw2GybxkRKmiK2KjQKmcopmrJp/Bt6fBR6ZkGSs9qUlxGHgwIDAQAB</realm-public-key>
|
||||||
|
<auth-server-url>http://localhost:8180/auth</auth-server-url>
|
||||||
|
</realm>
|
||||||
<secure-deployment name="web-console">
|
<secure-deployment name="web-console">
|
||||||
<realm>master</realm>
|
<realm>master</realm>
|
||||||
<resource>web-console</resource>
|
<resource>web-console</resource>
|
||||||
|
@ -69,4 +73,16 @@
|
||||||
</credential>
|
</credential>
|
||||||
<redirect-rewrite-rule name="^/wsmaster/api/(.*)$">/api/$1/</redirect-rewrite-rule>
|
<redirect-rewrite-rule name="^/wsmaster/api/(.*)$">/api/$1/</redirect-rewrite-rule>
|
||||||
</secure-deployment>
|
</secure-deployment>
|
||||||
|
<secure-deployment name="wildfly-management">
|
||||||
|
<realm>jboss-infra</realm>
|
||||||
|
<resource>wildfly-management</resource>
|
||||||
|
<bearer-only>true</bearer-only>
|
||||||
|
<ssl-required>EXTERNAL</ssl-required>
|
||||||
|
<principal-attribute>preferred_username</principal-attribute>
|
||||||
|
</secure-deployment>
|
||||||
|
<secure-server name="wildfly-console">
|
||||||
|
<realm>jboss-infra</realm>
|
||||||
|
<resource>wildfly-console</resource>
|
||||||
|
<public-client>true</public-client>
|
||||||
|
</secure-server>
|
||||||
</subsystem>
|
</subsystem>
|
|
@ -38,5 +38,9 @@
|
||||||
<module name="org.jboss.vfs"/>
|
<module name="org.jboss.vfs"/>
|
||||||
<module name="org.jboss.as.web-common"/>
|
<module name="org.jboss.as.web-common"/>
|
||||||
<module name="org.jboss.metadata"/>
|
<module name="org.jboss.metadata"/>
|
||||||
|
<module name="org.keycloak.keycloak-adapter-core"/>
|
||||||
|
<module name="org.keycloak.keycloak-wildfly-elytron-oidc-adapter"/>
|
||||||
|
<module name="org.wildfly.security.elytron"/>
|
||||||
|
<module name="io.undertow.core"/>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
</module>
|
</module>
|
||||||
|
|
|
@ -45,9 +45,9 @@
|
||||||
<selenium.version>2.53.0</selenium.version>
|
<selenium.version>2.53.0</selenium.version>
|
||||||
<arquillian-drone.version>2.0.1.Final</arquillian-drone.version>
|
<arquillian-drone.version>2.0.1.Final</arquillian-drone.version>
|
||||||
<arquillian-graphene.version>2.1.0.Alpha3</arquillian-graphene.version>
|
<arquillian-graphene.version>2.1.0.Alpha3</arquillian-graphene.version>
|
||||||
<arquillian-wildfly-container.version>2.1.0.Beta1</arquillian-wildfly-container.version>
|
<arquillian-wildfly-container.version>2.1.0.Alpha2</arquillian-wildfly-container.version>
|
||||||
<arquillian-wls-container.version>1.0.1.Final</arquillian-wls-container.version>
|
<arquillian-wls-container.version>1.0.1.Final</arquillian-wls-container.version>
|
||||||
<arquillian-infinispan-container.version>1.2.0.Beta2</arquillian-infinispan-container.version>
|
<arquillian-infinispan-container.version>1.2.0.Alpha2</arquillian-infinispan-container.version>
|
||||||
<version.shrinkwrap.resolvers>2.2.2</version.shrinkwrap.resolvers>
|
<version.shrinkwrap.resolvers>2.2.2</version.shrinkwrap.resolvers>
|
||||||
<undertow-embedded.version>1.0.0.Alpha2</undertow-embedded.version>
|
<undertow-embedded.version>1.0.0.Alpha2</undertow-embedded.version>
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue