All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.cxf.rs.security.saml.authorization.ClaimsAuthorizingInterceptor Maven / Gradle / Ivy

There is a newer version: 3.0.0-milestone2
Show newest version
/**
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you 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.apache.cxf.rs.security.saml.authorization;

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.logging.Logger;

import org.apache.cxf.common.logging.LogUtils;
import org.apache.cxf.common.util.ClassHelper;
import org.apache.cxf.interceptor.Fault;
import org.apache.cxf.interceptor.security.AccessDeniedException;
import org.apache.cxf.message.Message;
import org.apache.cxf.phase.AbstractPhaseInterceptor;
import org.apache.cxf.phase.Phase;
import org.apache.cxf.security.SecurityContext;
import org.apache.cxf.security.claims.authorization.Claim;
import org.apache.cxf.security.claims.authorization.ClaimMode;
import org.apache.cxf.security.claims.authorization.Claims;
import org.apache.cxf.service.Service;
import org.apache.cxf.service.invoker.MethodDispatcher;
import org.apache.cxf.service.model.BindingOperationInfo;


public class ClaimsAuthorizingInterceptor extends AbstractPhaseInterceptor {

    private static final Logger LOG = LogUtils.getL7dLogger(ClaimsAuthorizingInterceptor.class);
    
    private static final Set SKIP_METHODS;
    static {
        SKIP_METHODS = new HashSet();
        SKIP_METHODS.addAll(Arrays.asList(
            new String[] {"wait", "notify", "notifyAll", 
                          "equals", "toString", "hashCode"}));
    }
    
    private Map> claims = new HashMap>();
    private Map nameAliases = Collections.emptyMap();
    private Map formatAliases = Collections.emptyMap();
    
    public ClaimsAuthorizingInterceptor() {
        super(Phase.PRE_INVOKE);
    }
    
    public void handleMessage(Message message) throws Fault {
        SecurityContext sc = message.get(SecurityContext.class);
        if (!(sc instanceof SAMLSecurityContext)) {
            throw new AccessDeniedException("Security Context is unavailable or unrecognized");
        }
        
        Method method = getTargetMethod(message);
        
        if (authorize((SAMLSecurityContext)sc, method)) {
            return;
        }
        
        throw new AccessDeniedException("Unauthorized");
    }
    
    public void setClaims(Map> claimsMap) {
        claims.putAll(claimsMap);
    }
    
    protected Method getTargetMethod(Message m) {
        BindingOperationInfo bop = m.getExchange().get(BindingOperationInfo.class);
        if (bop != null) {
            MethodDispatcher md = (MethodDispatcher) 
                m.getExchange().get(Service.class).get(MethodDispatcher.class.getName());
            return md.getMethod(bop);
        } 
        Method method = (Method)m.get("org.apache.cxf.resource.method");
        if (method != null) {
            return method;
        }
        throw new AccessDeniedException("Method is not available : Unauthorized");
    }

    protected boolean authorize(SAMLSecurityContext sc, Method method) {
        List list = claims.get(method.getName());
        org.apache.cxf.rs.security.saml.assertion.Claims actualClaims = sc.getClaims();
        
        for (ClaimBean claimBean : list) {
            org.apache.cxf.rs.security.saml.assertion.Claim claim =  claimBean.getClaim();
            org.apache.cxf.rs.security.saml.assertion.Claim matchingClaim = 
                actualClaims.findClaimByFormatAndName(claim.getNameFormat(), claim.getName());
            if (matchingClaim == null) {
                if (claimBean.getClaimMode() == ClaimMode.STRICT) {
                    return false;
                } else {
                    continue;
                }
            }
            List claimValues = claim.getValues();
            List matchingClaimValues = matchingClaim.getValues();
            if (claimBean.isMatchAll() 
                && !matchingClaimValues.containsAll(claimValues)) {    
                return false;
            } else {
                boolean matched = false;
                for (String value : matchingClaimValues) {
                    if (claimValues.contains(value)) {
                        matched = true;    
                        break;
                    }
                }
                if (!matched) {
                    return false;
                }
            }
        }
        return true;
    }
    
    public void setSecuredObject(Object object) {
        Class cls = ClassHelper.getRealClass(object);
        findClaims(cls);
        if (claims.isEmpty()) {
            LOG.warning("The claims list is empty, the service object is not protected");
        }
    }

    protected void findClaims(Class cls) {
        if (cls == null || cls == Object.class) {
            return;
        }
        List clsClaims = 
            getClaims(cls.getAnnotation(Claims.class), cls.getAnnotation(Claim.class));
        for (Method m : cls.getMethods()) {
            if (SKIP_METHODS.contains(m.getName())) {
                continue;
            }
            List methodClaims = 
                getClaims(m.getAnnotation(Claims.class), m.getAnnotation(Claim.class));
            
            List allClaims = new ArrayList(methodClaims);
            for (ClaimBean bean : clsClaims) {
                if (isClaimOverridden(bean, methodClaims)) {
                    continue;
                }
                allClaims.add(bean);
            }
            
            claims.put(m.getName(), allClaims);
        }
        if (!claims.isEmpty()) {
            return;
        }
        
        findClaims(cls.getSuperclass());
        
        if (!claims.isEmpty()) {
            return;
        }
        
        for (Class interfaceCls : cls.getInterfaces()) {
            findClaims(interfaceCls);
        }
    }
    
    private static boolean isClaimOverridden(ClaimBean bean, List mClaims) {
        for (ClaimBean methodBean : mClaims) {    
            if (bean.getClaim().getName().equals(methodBean.getClaim().getName())
                && bean.getClaim().getNameFormat().equals(methodBean.getClaim().getNameFormat())) {
                return true;
            }
        }
        return false;
    }
    
    private List getClaims(
            Claims claimsAnn, Claim claimAnn) {
        List claimsList = new ArrayList();
        
        List annClaims = new ArrayList();
        if (claimsAnn != null) {
            annClaims.addAll(Arrays.asList(claimsAnn.value()));
        } else if (claimAnn != null) {
            annClaims.add(claimAnn);
        }
        for (Claim ann : annClaims) {
            org.apache.cxf.rs.security.saml.assertion.Claim claim = 
                new org.apache.cxf.rs.security.saml.assertion.Claim();
            
            String claimName = ann.name();
            if (nameAliases.containsKey(claimName)) {
                claimName = nameAliases.get(claimName);
            }
            String claimFormat = ann.format();
            if (formatAliases.containsKey(claimFormat)) {
                claimFormat = formatAliases.get(claimFormat);
            }
            
            claim.setName(claimName);
            claim.setNameFormat(claimFormat);
            claim.setValues(Arrays.asList(ann.value()));
            
            claimsList.add(new ClaimBean(claim, ann.mode(), ann.matchAll()));
        }
        return claimsList;
    }

    public void setNameAliases(Map nameAliases) {
        this.nameAliases = nameAliases;
    }

    public void setFormatAliases(Map formatAliases) {
        this.formatAliases = formatAliases;
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy