/************************************************************************* * Copyright 2009-2014 Eucalyptus Systems, Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; version 3 of the License. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see http://www.gnu.org/licenses/. * * Please contact Eucalyptus Systems, Inc., 6755 Hollister Ave., Goleta * CA 93117, USA or visit http://www.eucalyptus.com/licenses/ if you need * additional information or have any questions. ************************************************************************/ package com.eucalyptus.auth; import java.lang.reflect.Modifier; import java.util.List; import javax.annotation.Nonnull; import javax.annotation.Nullable; import org.apache.log4j.Logger; import com.eucalyptus.bootstrap.ServiceJarDiscovery; import com.eucalyptus.util.Classes; import com.eucalyptus.util.TypeMappers; import com.google.common.base.Optional; import com.google.common.collect.Lists; import javaslang.Tuple; import javaslang.Tuple2; /** * */ public class PolicyResourceContext implements AutoCloseable { private static final Logger logger = Logger.getLogger( PolicyResourceContext.class ); private static final List<PolicyResourceInterceptor> resourceInterceptors = Lists.newCopyOnWriteArrayList(); private static final ThreadLocal<Tuple2<String,String>> resourceIdentityLocal = new ThreadLocal<>( ); private static final PolicyResourceIdentity resourceIdentity = new PolicyResourceIdentity( ) { @Override public void setIdentity( final String resourceType, final String resourceId ) { resourceIdentityLocal.set( Tuple.of( resourceType, resourceId) ); } }; private PolicyResourceContext( ) { } public static PolicyResourceContext of( final Object resource, final String action ) { notifyResourceInterceptors( resource instanceof PolicyResourceInfo ? (PolicyResourceInfo) resource : TypeMappers.transform( resource, PolicyResourceInfo.class ), action ); return new PolicyResourceContext( ); } /** * Context without a resource, e.g. create action */ public static PolicyResourceContext of( final String resourceAccountNumber, final Class<?> resourceClass, final String action ) { notifyResourceInterceptors( resourceInfo( resourceAccountNumber, null, resourceClass ), action ); return new PolicyResourceContext( ); } public static <T> PolicyResourceInfo<T> resourceInfo( @Nullable final String accountNumber, @Nonnull final T resourceObject ) { return resourceInfo( accountNumber, resourceObject, (Class<? extends T>)resourceObject.getClass( ) ); } public static <T> PolicyResourceInfo<T> resourceInfo( @Nullable final String accountNumber, @Nullable final T resourceObject, @Nonnull final Class<? extends T> resourceClass ) { return new PolicyResourceInfo<T>( ) { @Nullable @Override public String getResourceAccountNumber( ) { return accountNumber; } @Nonnull @Override public Class<? extends T> getResourceClass( ) { return resourceClass; } @Nullable @Override public T getResourceObject( ) { return resourceObject; } }; } @Override public void close( ) { notifyResourceInterceptors( null, null ); resourceIdentity.setIdentity( null, null ); } public static String getResourceType( ){ final Tuple2<String,String> resourceTuple = resourceIdentityLocal.get( ); return resourceTuple == null ? null : resourceTuple._1; } public static String getResourceId( ){ final Tuple2<String,String> resourceTuple = resourceIdentityLocal.get( ); return resourceTuple == null ? null : resourceTuple._2; } private static void notifyResourceInterceptors( final PolicyResourceInfo<?> policyResourceInfo, final String action ) { for ( final PolicyResourceContext.PolicyResourceInterceptor interceptor : resourceInterceptors ) { interceptor.onResource( policyResourceInfo, action ); interceptor.pushResource( resourceIdentity ); } } public interface PolicyResourceInfo<T> { @Nullable String getResourceAccountNumber( ); @Nonnull Class<? extends T> getResourceClass( ); @Nullable T getResourceObject( ); } public interface PolicyResourceIdentity { void setIdentity( String resourceType, String resourceId ); } public interface PolicyResourceInterceptor { void onResource( @Nullable PolicyResourceInfo<?> resource, @Nullable String action ); default void pushResource( PolicyResourceIdentity resourceIdentity ) { } } public static class AccountNumberPolicyResourceInterceptor implements PolicyResourceInterceptor { private static final ThreadLocal<String> accountNumberThreadLocal = new ThreadLocal<>( ); @Override public void onResource( final PolicyResourceInfo<?> resource, final String action ) { accountNumberThreadLocal.set( resource == null ? null : resource.getResourceAccountNumber( ) ); } public static Optional<String> getCurrentResourceAccountNumber( ) { return Optional.fromNullable( accountNumberThreadLocal.get( ) ); } } @SuppressWarnings( "UnusedDeclaration" ) public static class PolicyResourceInterceptorDiscovery extends ServiceJarDiscovery { @SuppressWarnings( "unchecked" ) @Override public boolean processClass( final Class candidate ) throws Exception { if ( PolicyResourceContext.PolicyResourceInterceptor.class.isAssignableFrom( candidate ) && !Modifier.isAbstract( candidate.getModifiers() ) ) { logger.info( "Registered PolicyResourceInterceptor: " + candidate ); resourceInterceptors.add( (PolicyResourceContext.PolicyResourceInterceptor) Classes.newInstance( candidate ) ); return true; } return false; } @Override public Double getPriority( ) { return 0.3d; } } }