org.smallmind.quorum.juggler.Juggler Maven / Gradle / Ivy
/*
* Copyright (c) 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020 David Berkman
*
* This file is part of the SmallMind Code Project.
*
* The SmallMind Code Project is free software, you can redistribute
* it and/or modify it under either, at your discretion...
*
* 1) The terms of GNU Affero General Public License as published by the
* Free Software Foundation, either version 3 of the License, or (at
* your option) any later version.
*
* ...or...
*
* 2) The terms of the Apache License, Version 2.0.
*
* The SmallMind Code Project 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 or Apache License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* and the Apache License along with the SmallMind Code Project. If not, see
* or .
*
* Additional permission under the GNU Affero GPL version 3 section 7
* ------------------------------------------------------------------
* If you modify this Program, or any covered work, by linking or
* combining it with other code, such other code is not for that reason
* alone subject to any of the requirements of the GNU Affero GPL
* version 3.
*/
package org.smallmind.quorum.juggler;
import java.lang.reflect.Array;
import java.lang.reflect.Method;
import java.security.SecureRandom;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.ConcurrentSkipListMap;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import org.smallmind.scribe.pen.LoggerManager;
public class Juggler implements BlackList {
private enum State {DECONSTRUCTED, INITIALIZED, STARTED, STOPPED}
private final SecureRandom random = new SecureRandom();
private final JugglingPinFactory jugglingPinFactory;
private final P[] providers;
private final Class
providerClass;
private final Class resourceClass;
private final int recoveryCheckSeconds;
private ProviderRecoveryWorker recoveryWorker = null;
private ArrayList> sourcePins;
private ArrayList> targetPins;
private ConcurrentSkipListMap> blacklistMap;
private State state = State.DECONSTRUCTED;
public Juggler (Class providerClass, Class resourceClass, int recoveryCheckSeconds, JugglingPinFactory jugglingPinFactory, P provider, int size) {
this(providerClass, resourceClass, recoveryCheckSeconds, jugglingPinFactory, generateArray(provider, providerClass, size));
}
public Juggler (Class
providerClass, Class resourceClass, int recoveryCheckSeconds, JugglingPinFactory jugglingPinFactory, P... providers) {
this.providerClass = providerClass;
this.resourceClass = resourceClass;
this.recoveryCheckSeconds = recoveryCheckSeconds;
this.jugglingPinFactory = jugglingPinFactory;
this.providers = providers;
}
private static
P[] generateArray (P provider, Class
providerClass, int size) {
P[] array = (P[])Array.newInstance(providerClass, size);
Arrays.fill(array, provider);
return array;
}
public synchronized void initialize ()
throws JugglerResourceCreationException {
if (state.equals(State.DECONSTRUCTED)) {
sourcePins = new ArrayList<>(providers.length);
targetPins = new ArrayList<>(providers.length);
blacklistMap = new ConcurrentSkipListMap<>();
for (P provider : providers) {
targetPins.add(jugglingPinFactory.createJugglingPin(provider, resourceClass));
}
while (!targetPins.isEmpty()) {
sourcePins.add(targetPins.remove(random.nextInt(targetPins.size())));
}
state = State.INITIALIZED;
}
}
public synchronized void startup () {
startup(null);
}
public synchronized void startup (Method method, Object... args) {
if (state.equals(State.INITIALIZED)) {
Thread recoveryThread;
Iterator> sourcePinIter = sourcePins.iterator();
while (sourcePinIter.hasNext()) {
JugglingPin pin = sourcePinIter.next();
try {
pin.start(method, args);
} catch (JugglerResourceException jugglerResourceException) {
try {
LoggerManager.getLogger(Juggler.class).error(jugglerResourceException);
} finally {
sourcePinIter.remove();
blacklistMap.put(System.currentTimeMillis(), new BlacklistEntry<>(pin, jugglerResourceException));
}
}
}
if (recoveryCheckSeconds > 0) {
recoveryThread = new Thread(recoveryWorker = new ProviderRecoveryWorker(recoveryCheckSeconds));
recoveryThread.setDaemon(true);
recoveryThread.start();
}
state = State.STARTED;
}
}
public synchronized R pickResource ()
throws NoAvailableJugglerResourceException {
if (!(state.equals(State.INITIALIZED) || state.equals(State.STARTED))) {
throw new IllegalStateException("Juggler must be in the initialized or started state");
}
while (!(sourcePins.isEmpty() && targetPins.isEmpty())) {
R resource;
JugglingPin pin;
if (sourcePins.isEmpty()) {
ArrayList> tempPins = sourcePins;
sourcePins = targetPins;
targetPins = tempPins;
}
pin = sourcePins.remove(random.nextInt(sourcePins.size()));
try {
resource = pin.obtain();
targetPins.add(pin);
return resource;
} catch (Exception exception) {
try {
LoggerManager.getLogger(Juggler.class).error(exception);
} finally {
blacklistMap.put(System.currentTimeMillis(), new BlacklistEntry<>(pin, exception));
}
}
}
throw generateTerminatingException();
}
private NoAvailableJugglerResourceException generateTerminatingException () {
NoAvailableJugglerResourceException noAvailableJugglerResourceException = null;
boolean first = true;
for (BlacklistEntry blacklistEntry : blacklistMap.descendingMap().values()) {
if (first) {
noAvailableJugglerResourceException = new NoAvailableJugglerResourceException(blacklistEntry.getThrowable(), "All available resources(%s) have been black listed", providerClass.getSimpleName());
} else {
noAvailableJugglerResourceException.addSuppressed(blacklistEntry.getThrowable());
}
first = false;
}
return noAvailableJugglerResourceException;
}
@Override
public synchronized void addToBlackList (BlacklistEntry blacklistEntry) {
if (sourcePins.remove(blacklistEntry.getJugglingPin())) {
blacklistMap.put(System.currentTimeMillis(), blacklistEntry);
LoggerManager.getLogger(Juggler.class).info("Added resource(%s) to black list", blacklistEntry.getJugglingPin().describe());
} else if (targetPins.remove(blacklistEntry.getJugglingPin())) {
blacklistMap.put(System.currentTimeMillis(), blacklistEntry);
LoggerManager.getLogger(Juggler.class).info("Added resource(%s) to black list", blacklistEntry.getJugglingPin().describe());
}
}
public synchronized void shutdown () {
shutdown(null);
}
public synchronized void shutdown (Method method, Object... args) {
if (state.equals(State.STARTED)) {
if (recoveryWorker != null) {
try {
recoveryWorker.abort();
} catch (InterruptedException interruptedException) {
LoggerManager.getLogger(Juggler.class).error(interruptedException);
}
}
for (JugglingPin pin : sourcePins) {
try {
pin.stop(method, args);
} catch (Exception exception) {
LoggerManager.getLogger(Juggler.class).error(exception);
}
}
while (!targetPins.isEmpty()) {
JugglingPin pin = targetPins.remove(0);
try {
pin.stop(method, args);
} catch (Exception exception) {
LoggerManager.getLogger(Juggler.class).error(exception);
} finally {
sourcePins.add(pin);
}
}
state = State.STOPPED;
}
}
public synchronized void deconstruct () {
deconstruct(null);
}
public synchronized void deconstruct (Method method, Object... args) {
if (state.equals(State.STOPPED)) {
for (JugglingPin pin : sourcePins) {
try {
pin.close(method, args);
} catch (Exception exception) {
LoggerManager.getLogger(Juggler.class).error(exception);
}
}
state = State.DECONSTRUCTED;
}
}
private class ProviderRecoveryWorker implements Runnable {
private final CountDownLatch terminationLatch;
private final CountDownLatch exitLatch;
private final long recoveryCheckMillis;
public ProviderRecoveryWorker (int recoveryCheckSeconds) {
terminationLatch = new CountDownLatch(1);
exitLatch = new CountDownLatch(1);
recoveryCheckMillis = recoveryCheckSeconds * 1000;
}
public void abort ()
throws InterruptedException {
terminationLatch.countDown();
exitLatch.await();
}
@Override
public void run () {
try {
while (!terminationLatch.await(3, TimeUnit.SECONDS)) {
Map.Entry> firstEntry;
while (((firstEntry = blacklistMap.firstEntry()) != null) && ((firstEntry.getKey() + recoveryCheckMillis) <= System.currentTimeMillis())) {
if (firstEntry.getValue().getJugglingPin().recover()) {
synchronized (Juggler.this) {
JugglingPin recoveredPin;
if ((recoveredPin = blacklistMap.remove(firstEntry.getKey()).getJugglingPin()) != null) {
targetPins.add(recoveredPin);
LoggerManager.getLogger(Juggler.class).info("Recovered resource(%s) from black list", recoveredPin.describe());
} else {
LoggerManager.getLogger(ProviderRecoveryWorker.class).fatal("We've lost a resource(%s), which should never occur - please notify a system administrator", providerClass.getSimpleName());
}
}
}
}
}
} catch (InterruptedException interruptedException) {
LoggerManager.getLogger(ProviderRecoveryWorker.class).error(interruptedException);
}
exitLatch.countDown();
}
}
}