AbstractCircuitBreaker.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3.concurrent;

  18. import java.beans.PropertyChangeListener;
  19. import java.beans.PropertyChangeSupport;
  20. import java.util.concurrent.atomic.AtomicReference;

  21. /**
  22.  * Base class for circuit breakers.
  23.  *
  24.  * @param <T> the type of the value monitored by this circuit breaker
  25.  * @since 3.5
  26.  */
  27. public abstract class AbstractCircuitBreaker<T> implements CircuitBreaker<T> {

  28.     /**
  29.      * An internal enumeration representing the different states of a circuit
  30.      * breaker. This class also contains some logic for performing state
  31.      * transitions. This is done to avoid complex if-conditions in the code of
  32.      * {@link CircuitBreaker}.
  33.      */
  34.     protected enum State {

  35.         /** The closed state. */
  36.         CLOSED {
  37.             /**
  38.              * {@inheritDoc}
  39.              */
  40.             @Override
  41.             public State oppositeState() {
  42.                 return OPEN;
  43.             }
  44.         },

  45.         /** The open state. */
  46.         OPEN {
  47.             /**
  48.              * {@inheritDoc}
  49.              */
  50.             @Override
  51.             public State oppositeState() {
  52.                 return CLOSED;
  53.             }
  54.         };

  55.         /**
  56.          * Returns the opposite state to the represented state. This is useful
  57.          * for flipping the current state.
  58.          *
  59.          * @return the opposite state
  60.          */
  61.         public abstract State oppositeState();
  62.     }

  63.     /**
  64.      * The name of the <em>open</em> property as it is passed to registered
  65.      * change listeners.
  66.      */
  67.     public static final String PROPERTY_NAME = "open";

  68.     /**
  69.      * Converts the given state value to a boolean <em>open</em> property.
  70.      *
  71.      * @param state the state to be converted
  72.      * @return the boolean open flag
  73.      */
  74.     protected static boolean isOpen(final State state) {
  75.         return state == State.OPEN;
  76.     }

  77.     /** The current state of this circuit breaker. */
  78.     protected final AtomicReference<State> state = new AtomicReference<>(State.CLOSED);

  79.     /** An object for managing change listeners registered at this instance. */
  80.     private final PropertyChangeSupport changeSupport;

  81.     /**
  82.      * Creates an {@link AbstractCircuitBreaker}. It also creates an internal {@link PropertyChangeSupport}.
  83.      */
  84.     public AbstractCircuitBreaker() {
  85.         changeSupport = new PropertyChangeSupport(this);
  86.     }

  87.     /**
  88.      * Adds a change listener to this circuit breaker. This listener is notified whenever
  89.      * the state of this circuit breaker changes. If the listener is
  90.      * <strong>null</strong>, it is silently ignored.
  91.      *
  92.      * @param listener the listener to be added
  93.      */
  94.     public void addChangeListener(final PropertyChangeListener listener) {
  95.         changeSupport.addPropertyChangeListener(listener);
  96.     }

  97.     /**
  98.      * Changes the internal state of this circuit breaker. If there is actually a change
  99.      * of the state value, all registered change listeners are notified.
  100.      *
  101.      * @param newState the new state to be set
  102.      */
  103.     protected void changeState(final State newState) {
  104.         if (state.compareAndSet(newState.oppositeState(), newState)) {
  105.             changeSupport.firePropertyChange(PROPERTY_NAME, !isOpen(newState), isOpen(newState));
  106.         }
  107.     }

  108.     /**
  109.      * {@inheritDoc}
  110.      */
  111.     @Override
  112.     public abstract boolean checkState();

  113.     /**
  114.      * {@inheritDoc}
  115.      */
  116.     @Override
  117.     public void close() {
  118.         changeState(State.CLOSED);
  119.     }

  120.     /**
  121.      * {@inheritDoc}
  122.      */
  123.     @Override
  124.     public abstract boolean incrementAndCheckState(T increment);

  125.     /**
  126.      * {@inheritDoc}
  127.      */
  128.     @Override
  129.     public boolean isClosed() {
  130.         return !isOpen();
  131.     }

  132.     /**
  133.      * {@inheritDoc}
  134.      */
  135.     @Override
  136.     public boolean isOpen() {
  137.         return isOpen(state.get());
  138.     }

  139.     /**
  140.      * {@inheritDoc}
  141.      */
  142.     @Override
  143.     public void open() {
  144.         changeState(State.OPEN);
  145.     }

  146.     /**
  147.      * Removes the specified change listener from this circuit breaker.
  148.      *
  149.      * @param listener the listener to be removed
  150.      */
  151.     public void removeChangeListener(final PropertyChangeListener listener) {
  152.         changeSupport.removePropertyChangeListener(listener);
  153.     }

  154. }