001/* 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * https://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017package org.apache.commons.configuration2.reloading; 018 019import org.apache.commons.configuration2.event.Event; 020import org.apache.commons.configuration2.event.EventListener; 021import org.apache.commons.configuration2.event.EventListenerList; 022import org.apache.commons.configuration2.event.EventSource; 023import org.apache.commons.configuration2.event.EventType; 024 025/** 026 * <p> 027 * A class for adding support for reload operations in a generic way. 028 * </p> 029 * <p> 030 * A {@code ReloadingController} monitors a specific source and triggers reloading events if necessary. So it does not 031 * perform reloading itself, but only sends out notifications when it thinks that this should be done. This allows for a 032 * very generic setup in which different components involved in reloading are loosely coupled via events. 033 * </p> 034 * <p> 035 * A typical usage scenario is as follows: 036 * </p> 037 * <ul> 038 * <li>A {@code ReloadingController} instance is created and initialized with a {@link ReloadingDetector} object.</li> 039 * <li>A number of {@link EventListener} objects for reloading events can be registered at the controller.</li> 040 * <li>Now the controller's {@code checkForReloading()} method is called whenever a check is to be performed. This could 041 * be done for instance by a timer in regular intervals or by any other means appropriate for a specific 042 * application.</li> 043 * <li>When a check reveals that a reload operation is necessary all registered event listeners are notified.</li> 044 * <li>Typically one of the listeners is responsible to perform the actual reload operation. (How this is done is not in 045 * the scope of the controller object.) After this has been done, the controller's {@code resetReloadingState()} method 046 * must be called. It tells the controller that the last notification has been processed and that new checks are 047 * possible again. It is important that this method is called. Otherwise, {@code checkForReloading()} will not do any 048 * new checks or send out event notifications any more.</li> 049 * </ul> 050 * <p> 051 * This class can be accessed from multiple threads concurrently. It shields the associated {@link ReloadingDetector} 052 * object for concurrent access, so that a concrete detector implementation does not have to be thread-safe. 053 * </p> 054 * 055 * @since 2.0 056 */ 057public class ReloadingController implements EventSource { 058 059 /** Stores a reference to the reloading detector. */ 060 private final ReloadingDetector detector; 061 062 /** The helper object which manages the registered event listeners. */ 063 private final EventListenerList listeners; 064 065 /** A flag whether this controller is in reloading state. */ 066 private boolean reloadingState; 067 068 /** 069 * Creates a new instance of {@code ReloadingController} and associates it with the given {@code ReloadingDetector} 070 * object. 071 * 072 * @param detect the {@code ReloadingDetector} (must not be <strong>null</strong>) 073 * @throws IllegalArgumentException if the detector is undefined 074 */ 075 public ReloadingController(final ReloadingDetector detect) { 076 if (detect == null) { 077 throw new IllegalArgumentException("ReloadingDetector must not be null!"); 078 } 079 080 detector = detect; 081 listeners = new EventListenerList(); 082 } 083 084 /** 085 * {@inheritDoc} This class generates events of type {@code ReloadingEvent}. 086 */ 087 @Override 088 public <T extends Event> void addEventListener(final EventType<T> eventType, final EventListener<? super T> listener) { 089 listeners.addEventListener(eventType, listener); 090 } 091 092 /** 093 * Performs a check whether a reload operation is necessary. This method has to be called to trigger the generation of 094 * reloading events. It delegates to the associated {@link ReloadingDetector} and sends out notifications if necessary. 095 * The argument can be an arbitrary data object; it will be part of the event notification sent out when a reload 096 * operation should be performed. The return value indicates whether a change was detected and an event was sent. Once a 097 * need for a reload is detected, this controller is in <em>reloading state</em>. Until this state is reset (by calling 098 * {@link #resetReloadingState()}), no further reloading checks are performed by this method, and no events are fired; 099 * it then returns always <strong>true</strong>. 100 * 101 * @param data additional data for an event notification 102 * @return a flag whether a reload operation is necessary 103 */ 104 public boolean checkForReloading(final Object data) { 105 boolean sendEvent = false; 106 synchronized (this) { 107 if (isInReloadingState()) { 108 return true; 109 } 110 if (getDetector().isReloadingRequired()) { 111 sendEvent = true; 112 reloadingState = true; 113 } 114 } 115 116 if (sendEvent) { 117 listeners.fire(new ReloadingEvent(this, data)); 118 return true; 119 } 120 return false; 121 } 122 123 /** 124 * Gets the {@code ReloadingDetector} used by this controller. 125 * 126 * @return the {@code ReloadingDetector} 127 */ 128 public ReloadingDetector getDetector() { 129 return detector; 130 } 131 132 /** 133 * Tests whether this controller is in <em>reloading state</em>. A return value of <strong>true</strong> means that a previous 134 * invocation of {@code checkForReloading()} has detected the necessity for a reload operation, but 135 * {@code resetReloadingState()} has not been called yet. In this state no further reloading checks are possible. 136 * 137 * @return a flag whether this controller is in reloading state 138 */ 139 public synchronized boolean isInReloadingState() { 140 return reloadingState; 141 } 142 143 @Override 144 public <T extends Event> boolean removeEventListener(final EventType<T> eventType, final EventListener<? super T> listener) { 145 return listeners.removeEventListener(eventType, listener); 146 } 147 148 /** 149 * Resets the reloading state. This tells the controller that reloading has been performed and new checks are possible 150 * again. If this controller is not in reloading state, this method has no effect. 151 */ 152 public synchronized void resetReloadingState() { 153 if (isInReloadingState()) { 154 getDetector().reloadingPerformed(); 155 reloadingState = false; 156 } 157 } 158}