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

com.oracle.tools.deferred.Supervised Maven / Gradle / Ivy

/*
 * File: Supervised.java
 *
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * The contents of this file are subject to the terms and conditions of 
 * the Common Development and Distribution License 1.0 (the "License").
 *
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the License by consulting the LICENSE.txt file
 * distributed with this file, or by consulting https://oss.oracle.com/licenses/CDDL
 *
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing the software, include this License Header Notice in each
 * file and include the License file LICENSE.txt.
 *
 * MODIFICATIONS:
 * If applicable, add the following below the License Header, with the fields
 * enclosed by brackets [] replaced by your own identifying information:
 * "Portions Copyright [year] [name of copyright owner]"
 */

package com.oracle.tools.deferred;

import java.util.concurrent.TimeUnit;

/**
 * A {@link Supervised} is a {@link Deferred} adapter that has
 * knowledge of the last known availability of an object, the purpose of which
 * is to ensure that the underlying object is protected from repeated and
 * perhaps continuously unrelenting requests by an application to access it,
 * when it simply may be unavailable for a long period of time.
 * 

* A program will use a {@link Supervised} object to avoid the situation when * multiple {@link Thread}s attempt to call {@link #get()} or a single * {@link Thread} calls {@link #get()} in a tight loop. It protects the adapted * {@link Deferred#get()} method from being called repeatedly, allowing the underlying * resource a chance to "recover" when required. *

* Copyright (c) 2012. All Rights Reserved. Oracle Corporation.
* Oracle is a registered trademark of Oracle Corporation and/or its affiliates. * * @author Brian Oliver */ public class Supervised implements Deferred { /** * The {@link Deferred} being adapted and protected by the * {@link Supervised}. */ private Deferred deferred; /** * The instant in time (since the epoc) when the last attempt to acquire * the object failed. */ private volatile long instantOfLastFailure; /** * The duration that must pass before attempting to acquire a object * (since the last failed attempt) */ private long retryDelayDuration; /** * The {@link TimeUnit} for the {@link #retryDelayDuration}. */ private TimeUnit retryDelayTimeUnit; /** * Constructor for a {@link Supervised}. * * @param deferred the {@link Deferred} to supervise */ public Supervised(Deferred deferred) { this.deferred = deferred; this.instantOfLastFailure = -1; this.retryDelayDuration = 250L; this.retryDelayTimeUnit = TimeUnit.MILLISECONDS; } /** * Obtains the adapted {@link Deferred}. * * @return the adapted {@link Deferred} */ public Deferred getDeferred() { return deferred; } /** * Determines a request to {@link #get()} the underlying object will fail * immediately. * * @return true the object is accessible, false if * a {@link #get()} request will fail immediately */ public boolean isAccessible() { return instantOfLastFailure < 0 || System.currentTimeMillis() > instantOfLastFailure + retryDelayTimeUnit.toMillis(retryDelayDuration); } /** * Applications should call this method when they determine, through some * external means, that the underlying object they have acquired from the * {@link Supervised} is no longer available. *

* This ensures that a {@link Supervised} can protect against * repeated requests for the object from parts of an application. */ public void resourceNoLongerAvailable() { instantOfLastFailure = System.currentTimeMillis(); } @Override public T get() throws TemporarilyUnavailableException, PermanentlyUnavailableException { if (isAccessible()) { // we synchronize here to protect the underlying object, // that may not be able to cope with a lot of concurrent access synchronized (this) { // the current thread may have be held up for a while waiting, // so we check again if the object is still accessible if (isAccessible()) { try { // attempt to get the object return deferred.get(); } catch (PermanentlyUnavailableException e) { // when a object is unavailable, we must re-throw it resourceNoLongerAvailable(); throw e; } catch (RuntimeException e) { // for runtime exceptions we re-throw them resourceNoLongerAvailable(); throw e; } } else { return null; } } } else { return null; } } @Override public Class getDeferredClass() { return deferred.getDeferredClass(); } @Override public String toString() { return String.format("Supervised<%s>{%s}", getDeferredClass(), getDeferred()); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy