All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy