Package com.developmentsprint.spring.breaker.support

Source Code of com.developmentsprint.spring.breaker.support.CompositeCircuitManager

/**
* Copyright 2014 Development Sprint, LLC.
*
* 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 com.developmentsprint.spring.breaker.support;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Map;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;

import com.developmentsprint.spring.breaker.CircuitManager;

/**
* Composite {@link CircuitManager} implementation that iterates over a given collection of delegate {@link CircuitManager} instances.
*
* <p>
* Allows {@link NoOpCircuitManager} to be automatically added to the end of the list for handling circuit breaker declarations without a backing manager.
* Otherwise, any custom {@link CircuitManager} may play that role of the last delegate as well, lazily creating circuit breakers.
*
* <p>
* Note: Regular CircuitManagers that this composite manager delegates to need to return {@code null} from {@link #circuitManagers} if they are unaware of the
* specified circuit manager name, allowing for iteration to the next delegate in line. However, most {@link CircuitManager} implementations fall back to lazy
* creation of named circuit manager once requested; check out the specific configuration details for a 'static' mode with fixed circuit manager names, if
* available.
*
* @author Todd Orr
* @since 1.0
* @see #setFallbackToNoOpCircuitManager
*/
public class CompositeCircuitManager implements CircuitManager, InitializingBean, ApplicationContextAware {

    private final List<CircuitManager> circuitManagers = new ArrayList<CircuitManager>();

    private boolean fallbackToNoOpCircuitManager = false;

    private ApplicationContext applicationContext;

    /**
     * Construct an empty CompositeCircuitManager, with delegate CircuitManagers to be added via the {@link #setCircuitManagers "CircuitManagers"} property.
     */
    public CompositeCircuitManager() {
    }

    /**
     * Construct a CompositeCircuitManager from the given delegate CircuitManagers.
     *
     * @param CircuitManagers
     *            the CircuitManagers to delegate to
     */
    public CompositeCircuitManager(CircuitManager... CircuitManagers) {
        setCircuitManagers(Arrays.asList(CircuitManagers));
    }

    /**
     * Specify the CircuitManagers to delegate to.
     */
    public void setCircuitManagers(Collection<CircuitManager> CircuitManagers) {
        this.circuitManagers.clear(); // just here to preserve compatibility with previous behavior
        this.circuitManagers.addAll(CircuitManagers);
    }

    /**
     * Indicate whether a {@link NoOpCircuitManager} should be added at the end of the delegate list. In this case, any {@code getCache} requests not handled by
     * the configured CircuitManagers will be automatically handled by the {@link NoOpCircuitManager} (and hence never return {@code null}).
     */
    public void setFallbackToNoOpCircuitManager(boolean fallbackToNoOpCache) {
        this.fallbackToNoOpCircuitManager = fallbackToNoOpCache;
    }

    public void afterPropertiesSet() {
        if (this.fallbackToNoOpCircuitManager) {
            this.circuitManagers.add(new NoOpCircuitManager());
        }
    }

    @Override
    public Object execute(Invoker invoker) {
        String declaredManager = invoker.getCircuitBreakerAttribute().getCircuitManager();

        CircuitManager applicableCircuitManager = null;
        for (CircuitManager manager : circuitManagers) {
            if (manager != null) {
                for (Map.Entry<String, ? extends CircuitManager> entry : applicationContext.getBeansOfType(manager.getClass()).entrySet()) {
                    if (entry.getKey().equals(declaredManager)) {
                        applicableCircuitManager = entry.getValue();
                    }
                }
            }
        }

        if (applicableCircuitManager == null && fallbackToNoOpCircuitManager) {
            applicableCircuitManager = new NoOpCircuitManager();
        }

        return applicableCircuitManager.execute(invoker);
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }

}
TOP

Related Classes of com.developmentsprint.spring.breaker.support.CompositeCircuitManager

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.