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

org.atmosphere.cpr.DefaultBroadcaster Maven / Gradle / Ivy

There is a newer version: 3.0.10
Show newest version
/*
 * Copyright 2015 Async-IO.org
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
 * use this file except in compliance with the License. You may obtain a copy of
 * the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
 * License for the specific language governing permissions and limitations under
 * the License.
 */

package org.atmosphere.cpr;

import org.atmosphere.cache.BroadcastMessage;
import org.atmosphere.cache.CacheMessage;
import org.atmosphere.cpr.BroadcastFilter.BroadcastAction;
import org.atmosphere.lifecycle.LifecycleHandler;
import org.atmosphere.pool.PoolableBroadcasterFactory;
import org.atmosphere.util.Utils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.net.URI;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;
import java.util.UUID;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.Callable;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.Future;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.RejectedExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;

import static org.atmosphere.cpr.ApplicationConfig.BACKWARD_COMPATIBLE_WEBSOCKET_BEHAVIOR;
import static org.atmosphere.cpr.ApplicationConfig.BROADCASTER_CACHE_STRATEGY;
import static org.atmosphere.cpr.ApplicationConfig.BROADCASTER_SHAREABLE_LISTENERS;
import static org.atmosphere.cpr.ApplicationConfig.BROADCASTER_WAIT_TIME;
import static org.atmosphere.cpr.ApplicationConfig.CACHE_MESSAGE_ON_IO_FLUSH_EXCEPTION;
import static org.atmosphere.cpr.ApplicationConfig.MAX_INACTIVE;
import static org.atmosphere.cpr.ApplicationConfig.OUT_OF_ORDER_BROADCAST;
import static org.atmosphere.cpr.ApplicationConfig.SUSPENDED_ATMOSPHERE_RESOURCE_UUID;
import static org.atmosphere.cpr.ApplicationConfig.WRITE_TIMEOUT;
import static org.atmosphere.cpr.BroadcasterLifeCyclePolicy.ATMOSPHERE_RESOURCE_POLICY.NEVER;
import static org.atmosphere.cpr.FrameworkConfig.INJECTED_ATMOSPHERE_RESOURCE;

/**
 * The default {@link Broadcaster} implementation.
 * 

* Broadcast messages to suspended responses using the caller's Thread. * This basic {@link Broadcaster} use an {@link java.util.concurrent.ExecutorService} * to broadcast messages, hence the broadcast operation is asynchronous. Make sure * you block on {@link #broadcast(Object)}.get()} if you need synchronous operations. * * @author Jeanfrancois Arcand */ public class DefaultBroadcaster implements Broadcaster { public static final int POLLING_DEFAULT = 100; public static final String CACHED = DefaultBroadcaster.class.getName() + ".messagesCached"; private static final Logger logger = LoggerFactory.getLogger(DefaultBroadcaster.class); private static final String DESTROYED = "This Broadcaster has been destroyed and cannot be used {} by invoking {}"; private static final List EMPTY_LISTENERS = new ArrayList(); protected final ConcurrentLinkedQueue resources = new ConcurrentLinkedQueue(); protected BroadcasterConfig bc; protected final BlockingQueue messages = new LinkedBlockingQueue(); protected Collection broadcasterListeners; protected final AtomicBoolean started = new AtomicBoolean(false); protected final AtomicBoolean initialized = new AtomicBoolean(false); protected final AtomicBoolean destroyed = new AtomicBoolean(false); protected SCOPE scope = SCOPE.APPLICATION; protected String name = DefaultBroadcaster.class.getSimpleName(); protected final ConcurrentLinkedQueue delayedBroadcast = new ConcurrentLinkedQueue(); protected final ConcurrentLinkedQueue broadcastOnResume = new ConcurrentLinkedQueue(); protected final ConcurrentLinkedQueue lifeCycleListeners = new ConcurrentLinkedQueue(); protected final ConcurrentHashMap writeQueues = new ConcurrentHashMap(); protected final WriteQueue uniqueWriteQueue = new WriteQueue("-1"); protected final AtomicInteger dispatchThread = new AtomicInteger(); protected Future[] notifierFuture; protected Future[] asyncWriteFuture; private POLICY policy = POLICY.FIFO; private final AtomicLong maxSuspendResource = new AtomicLong(-1); private final AtomicBoolean requestScoped = new AtomicBoolean(false); private final AtomicBoolean recentActivity = new AtomicBoolean(false); private BroadcasterLifeCyclePolicy lifeCyclePolicy = new BroadcasterLifeCyclePolicy.Builder() .policy(NEVER).build(); protected URI uri; protected AtmosphereConfig config; private final Object[] awaitBarrier = new Object[0]; private final AtomicBoolean outOfOrderBroadcastSupported = new AtomicBoolean(false); protected int writeTimeoutInSecond = -1; protected int waitTime = POLLING_DEFAULT; private boolean backwardCompatible = false; private LifecycleHandler lifecycleHandler; private Future currentLifecycleTask; private boolean cacheOnIOFlushException = true; protected boolean sharedListeners = false; protected boolean candidateForPoolable; protected final String usingTokenIdForAttribute = UUID.randomUUID().toString(); public DefaultBroadcaster() { } public Broadcaster initialize(String name, URI uri, AtmosphereConfig config) { this.name = name; this.uri = uri; this.config = config; bc = createBroadcasterConfig(config); String s = config.getInitParameter(BROADCASTER_CACHE_STRATEGY); if (s != null) { logger.warn("{} is no longer supported. Use BroadcastInterceptor instead. By default the original message will be cached.", BROADCASTER_CACHE_STRATEGY); } s = config.getInitParameter(OUT_OF_ORDER_BROADCAST); if (s != null) { outOfOrderBroadcastSupported.set(Boolean.valueOf(s)); } s = config.getInitParameter(BROADCASTER_WAIT_TIME); if (s != null) { waitTime = Integer.valueOf(s); } s = config.getInitParameter(WRITE_TIMEOUT); if (s != null) { writeTimeoutInSecond = Integer.valueOf(s); } if (outOfOrderBroadcastSupported.get()) { logger.trace("{} supports Out Of Order Broadcast: {}", name, outOfOrderBroadcastSupported.get()); } initialized.set(true); backwardCompatible = Boolean.parseBoolean(config.getInitParameter(BACKWARD_COMPATIBLE_WEBSOCKET_BEHAVIOR)); cacheOnIOFlushException = config.getInitParameter(CACHE_MESSAGE_ON_IO_FLUSH_EXCEPTION, true); sharedListeners = config.getInitParameter(BROADCASTER_SHAREABLE_LISTENERS, false); if (sharedListeners) { broadcasterListeners = config.getBroadcasterFactory().broadcasterListeners(); } else { broadcasterListeners = new ConcurrentLinkedQueue(); } candidateForPoolable = PoolableBroadcasterFactory.class.isAssignableFrom(config.getBroadcasterFactory().getClass()); return this; } public Broadcaster initialize(String name, AtmosphereConfig config) { return initialize(name, URI.create("http://localhost"), config); } /** * Create {@link BroadcasterConfig}. * * @param config the {@link AtmosphereConfig} * @return an instance of {@link BroadcasterConfig} */ protected BroadcasterConfig createBroadcasterConfig(AtmosphereConfig config) { return new BroadcasterConfig(config.framework().broadcasterFilters, config, getID()).init(); } @Override public synchronized void destroy() { try { logger.trace("Broadcaster {} will be pooled: {}", getID(), candidateForPoolable); if (!candidateForPoolable) { if (notifyOnPreDestroy()) return; if (destroyed.getAndSet(true)) return; logger.trace("Broadcaster {} is being destroyed and cannot be re-used. Policy was {}", getID(), policy); logger.trace("Broadcaster {} is being destroyed and cannot be re-used. Resources are {}", getID(), resources); started.set(false); releaseExternalResources(); killReactiveThreads(); if (bc != null) { bc.destroy(); } lifeCycleListeners.clear(); delayedBroadcast.clear(); if (!sharedListeners) { broadcasterListeners.clear(); } } resources.clear(); broadcastOnResume.clear(); messages.clear(); writeQueues.clear(); if (config.getBroadcasterFactory() != null) { config.getBroadcasterFactory().remove(this, this.getID()); } } catch (Throwable t) { logger.error("Unexpected exception during Broadcaster destroy {}", getID(), t); } } @Override public Collection getAtmosphereResources() { return Collections.unmodifiableCollection(resources); } @Override public void setScope(SCOPE scope) { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "setScope"); return; } this.scope = scope; if (scope != SCOPE.REQUEST) { return; } logger.debug("Changing broadcaster scope for {}. This broadcaster will be destroyed.", getID()); synchronized (resources) { try { // Next, we need to create a new broadcaster per resource. for (AtmosphereResource resource : resources) { Broadcaster b = config.getBroadcasterFactory() .get(getClass(), getClass().getSimpleName() + "/" + config.uuidProvider().generateUuid()); /** * REQUEST_SCOPE means one BroadcasterCache per Broadcaster, */ if (DefaultBroadcaster.class.isAssignableFrom(this.getClass())) { BroadcasterCache cache = config.framework().newClassInstance(BroadcasterCache.class, bc.getBroadcasterCache().getClass()); cache.configure(config); b.getBroadcasterConfig().setBroadcasterCache(cache); } resource.setBroadcaster(b); b.setScope(SCOPE.REQUEST); if (resource.getAtmosphereResourceEvent().isSuspended()) { b.addAtmosphereResource(resource); } logger.debug("Resource {} not using broadcaster {}", resource, b.getID()); } // Do not destroy because this is a new Broadcaster if (resources.isEmpty()) { return; } destroy(); } catch (Exception e) { logger.error("Failed to set request scope for current resources", e); } } } @Override public SCOPE getScope() { return scope; } @Override public synchronized void setID(String id) { if (id == null) { id = getClass().getSimpleName() + "/" + config.uuidProvider().generateUuid(); } if (config.getBroadcasterFactory() == null) return; // we are shutdown or destroyed, but someone still reference Broadcaster b = config.getBroadcasterFactory().lookup(this.getClass(), id); if (b != null && b.getScope() == SCOPE.REQUEST) { throw new IllegalStateException("Broadcaster ID already assigned to SCOPE.REQUEST. Cannot change the id"); } else if (b != null) { return; } config.getBroadcasterFactory().remove(this, name); this.name = id; config.getBroadcasterFactory().add(this, name); bc.broadcasterID(name); } /** * Rename this Broadcaster without invoking it's associated {@link org.atmosphere.cpr.BroadcasterFactory}. This * method must be carefully used as it could easily create memory leak as the Broadcaster won't be removed * from its {@link org.atmosphere.cpr.BroadcasterFactory}. * * @param id the new name * @return this; */ public Broadcaster rename(String id) { this.name = id; return this; } @Override public String getID() { return name; } @Override public void resumeAll() { synchronized (resources) { for (AtmosphereResource r : resources) { try { r.resume(); } catch (Throwable t) { logger.trace("resumeAll", t); } finally { removeAtmosphereResource(r); } } } } @Override public void releaseExternalResources() { } @Override public void setBroadcasterLifeCyclePolicy(final BroadcasterLifeCyclePolicy lifeCyclePolicy) { this.lifeCyclePolicy = lifeCyclePolicy; if (lifecycleHandler != null) lifecycleHandler.on(this); } @Override public BroadcasterLifeCyclePolicy getBroadcasterLifeCyclePolicy() { return lifeCyclePolicy; } @Override public void addBroadcasterLifeCyclePolicyListener(BroadcasterLifeCyclePolicyListener b) { lifeCycleListeners.add(b); } @Override public void removeBroadcasterLifeCyclePolicyListener(BroadcasterLifeCyclePolicyListener b) { lifeCycleListeners.remove(b); } @Override public boolean isDestroyed() { return destroyed.get(); } @Override public Future awaitAndBroadcast(Object t, long time, TimeUnit timeUnit) { if (resources.isEmpty()) { synchronized (awaitBarrier) { try { logger.trace("Awaiting for AtmosphereResource for {} {}", time, timeUnit); awaitBarrier.wait(translateTimeUnit(time, timeUnit)); } catch (Throwable e) { logger.warn("awaitAndBroadcast", e); return null; } } } return broadcast(t); } @Override public Broadcaster addBroadcasterListener(BroadcasterListener b) { if (!sharedListeners && !broadcasterListeners.contains(b)) { broadcasterListeners.add(b); } return this; } @Override public Broadcaster removeBroadcasterListener(BroadcasterListener b) { if (!sharedListeners) broadcasterListeners.remove(b); return this; } protected Runnable getBroadcastHandler() { return new Runnable() { public void run() { while (!isDestroyed()) { Deliver msg = null; try { msg = messages.poll(waitTime, TimeUnit.MILLISECONDS); if (msg == null) { dispatchThread.decrementAndGet(); return; } } catch (InterruptedException ex) { logger.trace("{} got interrupted for Broadcaster {}", Thread.currentThread().getName(), getID()); logger.trace("", ex); return; } finally { if (outOfOrderBroadcastSupported.get()) { bc.getExecutorService().submit(this); } } try { logger.trace("{} is about to broadcast {}", getID(), msg); push(msg); } catch (Throwable ex) { if (!started.get() || destroyed.get()) { logger.trace("Failed to submit broadcast handler runnable on shutdown for Broadcaster {}", getID(), ex); return; } else { logger.warn("This message {} will be lost", msg); logger.warn("Failed to submit broadcast handler runnable to for Broadcaster" + getID(), ex); } } finally { if (outOfOrderBroadcastSupported.get()) { return; } } } } }; } protected Runnable getAsyncWriteHandler(final WriteQueue writeQueue) { return new Runnable() { public void run() { while (!isDestroyed()) { AsyncWriteToken token = null; try { token = writeQueue.queue.poll(waitTime, TimeUnit.MILLISECONDS); if (token == null && !outOfOrderBroadcastSupported.get()) { synchronized (writeQueue) { if (writeQueue.queue.size() == 0) { writeQueue.monitored.set(false); writeQueues.remove(writeQueue.uuid); return; } } } else if (token == null) { return; } } catch (InterruptedException ex) { logger.trace("{} got interrupted for Broadcaster {}", Thread.currentThread().getName(), getID()); logger.trace("", ex); return; } finally { if (!bc.getAsyncWriteService().isShutdown() && outOfOrderBroadcastSupported.get()) { bc.getAsyncWriteService().submit(this); } } // Shield us from https://github.com/Atmosphere/atmosphere/issues/1187 if (token != null) { synchronized (token.resource) { try { logger.trace("About to write to {}", token.resource); executeAsyncWrite(token); } catch (Throwable ex) { if (!started.get() || destroyed.get()) { logger.trace("Failed to execute a write operation. Broadcaster is destroyed or not yet started for Broadcaster {}", getID(), ex); return; } else { try { if (token != null) { logger.warn("This message {} will be lost for AtmosphereResource {}, adding it to the BroadcasterCache", token.originalMessage, token.resource != null ? token.resource.uuid() : "null"); cacheLostMessage(token.resource, token, true); } } finally { if (token != null) { removeAtmosphereResource(token.resource, false); } logger.warn("Failed to execute a write operation for Broadcaster " + getID(), ex); } } } finally { if (!bc.getAsyncWriteService().isShutdown() && outOfOrderBroadcastSupported.get()) { return; } } } } } } }; } protected void start() { if (!initialized.get()) { logger.warn("Broadcaster {} not initialized", getID()); } if (!started.getAndSet(true)) { bc.getBroadcasterCache().start(); // Only start if we know a child haven't started them. if (notifierFuture == null && asyncWriteFuture == null) { spawnReactor(); } } } protected void spawnReactor() { killReactiveThreads(); int threads = outOfOrderBroadcastSupported.get() ? reactiveThreadsCount() : 1; notifierFuture = new Future[threads]; if (outOfOrderBroadcastSupported.get()) { asyncWriteFuture = new Future[threads]; for (int i = 0; i < threads; i++) { notifierFuture[i] = bc.getExecutorService().submit(getBroadcastHandler()); asyncWriteFuture[i] = bc.getAsyncWriteService().submit(getAsyncWriteHandler(uniqueWriteQueue)); } } else { notifierFuture[0] = bc.getExecutorService().submit(getBroadcastHandler()); } dispatchThread.set(threads); } protected void killReactiveThreads() { if (notifierFuture != null) { for (Future f : notifierFuture) { if (f != null) f.cancel(false); } } if (asyncWriteFuture != null) { for (Future f : asyncWriteFuture) { if (f != null) f.cancel(false); } } } /** * Return the default number of reactive threads that will be waiting for work when a broadcast operation * is executed. * * @return the default number of reactive threads */ protected int reactiveThreadsCount() { return Runtime.getRuntime().availableProcessors() * 2; } protected void push(Deliver deliver) { if (destroyed.get()) { return; } deliverPush(deliver, true); } protected void deliverPush(Deliver deliver, boolean rec) { recentActivity.set(true); String prevMessage = deliver.message.toString(); if (rec && !delayedBroadcast.isEmpty()) { Iterator i = delayedBroadcast.iterator(); StringBuilder b = new StringBuilder(); while (i.hasNext()) { Deliver e = i.next(); e.future.cancel(true); try { // Append so we do a single flush if (e.message instanceof String && deliver.message instanceof String) { b.append(e.message); } else { deliverPush(e, false); } } finally { i.remove(); } } if (b.length() > 0) { deliver.message = b.append(deliver.message).toString(); } } Object finalMsg = callable(deliver.message); if (finalMsg == null) { logger.error("Callable exception. Please catch all exceptions from your callable. Message {} will be lost and all AtmosphereResource " + "associated with this Broadcaster resumed.", deliver.message); entryDone(deliver.future); switch (deliver.type) { case ALL: synchronized (resources) { for (AtmosphereResource r : resources) { if (Utils.resumableTransport(r.transport())) try { r.resume(); } catch (Throwable t) { logger.trace("resumeAll", t); } } } break; case RESOURCE: deliver.resource.resume(); break; case SET: for (AtmosphereResource r : deliver.resources) { r.resume(); } break; } return; } notifyOnMessage(deliver); Object prevM = deliver.originalMessage; deliver.originalMessage = (deliver.originalMessage != deliver.message ? callable(deliver.originalMessage) : finalMsg); if (deliver.originalMessage == null) { logger.trace("Broadcasted message was null {}", prevM); entryDone(deliver.future); return; } deliver.message = finalMsg; // We cache first, and if the broadcast succeed, we will remove it. AtmosphereResource cache = deliver.type != Deliver.TYPE.RESOURCE ? null : deliver.resource; deliver.cache = bc.getBroadcasterCache().addToCache(getID(), cache != null ? cache.uuid() : BroadcasterCache.NULL, new BroadcastMessage(deliver.originalMessage)); if (resources.isEmpty()) { logger.trace("No resource available for {} and message {}", getID(), finalMsg); entryDone(deliver.future); return; } try { if (logger.isTraceEnabled()) { for (AtmosphereResource r : resources) { logger.trace("AtmosphereResource {} available for {}", r.uuid(), deliver.message); } } boolean hasFilters = bc.hasPerRequestFilters(); Object beforeProcessingMessage = deliver.message; switch (deliver.type) { case ALL: AtomicInteger count = new AtomicInteger(resources.size()); for (AtmosphereResource r : resources) { deliver.message = beforeProcessingMessage; boolean deliverMessage = perRequestFilter(r, deliver); if (endBroadcast(deliver, deliverMessage)) continue; if (deliver.writeLocally) { queueWriteIO(r, hasFilters ? new Deliver(r, deliver) : deliver, count); } } break; case RESOURCE: boolean deliverMessage = perRequestFilter(deliver.resource, deliver); if (endBroadcast(deliver, deliverMessage)) return; if (deliver.writeLocally) { queueWriteIO(deliver.resource, deliver, new AtomicInteger(1)); } break; case SET: count = new AtomicInteger(deliver.resources.size()); for (AtmosphereResource r : deliver.resources) { deliver.message = beforeProcessingMessage; deliverMessage = perRequestFilter(r, deliver); if (endBroadcast(deliver, deliverMessage)) continue; if (deliver.writeLocally) { queueWriteIO(r, hasFilters ? new Deliver(r, deliver) : deliver, count); } } break; } deliver.message = prevMessage; } catch (InterruptedException ex) { logger.debug(ex.getMessage(), ex); } } protected boolean endBroadcast(Deliver deliver, boolean deliverMessage) { if (!deliverMessage || deliver.message == null) { logger.debug("Skipping broadcast delivery {} for resource {} ", deliver.message, deliver.resource != null ? deliver.resource.uuid() : "null"); bc.getBroadcasterCache().clearCache(getID(), deliver.resource != null ? deliver.resource.uuid() : BroadcasterCache.NULL, deliver.cache); entryDone(deliver.future); return true; } return false; } protected void queueWriteIO(AtmosphereResource r, Deliver deliver, AtomicInteger count) throws InterruptedException { if (deliver.async) { // The onStateChange/onRequest may change the isResumed value, hence we need to make sure only one thread flip // the switch to garantee the Entry will be cached in the order it was broadcasted. // Without synchronizing we may end up with a out of order BroadcasterCache queue. if (!bc.getBroadcasterCache().getClass().equals(BroadcasterCache.DEFAULT.getClass().getName())) { if (r.isResumed() || r.isCancelled()) { logger.trace("AtmosphereResource {} has been resumed or cancelled, unable to Broadcast message {}", r.uuid(), deliver.message); /** * https://github.com/Atmosphere/atmosphere/issues/1886 * Before caching the message, double check if the client has reconnected, and if true, send the * cached message. */ AtmosphereResource r2 = config.resourcesFactory().find(r.uuid()); logger.trace("Found an AtmosphereResource {} in state {}", r2, r.isSuspended()); if (r2 != null && r2.isSuspended() && r.hashCode() != r2.hashCode()) { // Prevent other Broadcast to happens removeAtmosphereResource(r2); checkCachedAndPush(r2, r2.getAtmosphereResourceEvent()); } return; } } AsyncWriteToken w = new AsyncWriteToken(r, deliver.message, deliver.future, deliver.originalMessage, deliver.cache, count); if (!outOfOrderBroadcastSupported.get()) { WriteQueue writeQueue = writeQueues.get(r.uuid()); if (writeQueue == null) { writeQueue = new WriteQueue(r.uuid()); writeQueues.put(r.uuid(), writeQueue); } writeQueue.queue.put(w); synchronized (writeQueue) { if (!writeQueue.monitored.getAndSet(true)) { logger.trace("Broadcaster {} is about to queueWriteIO for AtmosphereResource {}", name, r.uuid()); bc.getAsyncWriteService().submit(getAsyncWriteHandler(writeQueue)); } } } else { uniqueWriteQueue.queue.offer(w); } } else { executeBlockingWrite(r, deliver, count); } } protected void executeBlockingWrite(AtmosphereResource r, Deliver deliver, AtomicInteger count) throws InterruptedException { // We deliver using the calling thread. synchronized (r) { executeAsyncWrite(new AsyncWriteToken(r, deliver.message, deliver.future, deliver.originalMessage, deliver.cache, count)); } } public final static class WriteQueue { final BlockingQueue queue = new LinkedBlockingQueue(); final AtomicBoolean monitored = new AtomicBoolean(); final String uuid; private WriteQueue(String uuid) { this.uuid = uuid; } public List asString() { List l = new ArrayList(); for (AsyncWriteToken w : queue) { l.add(w.toString()); } return l; } } protected boolean perRequestFilter(AtmosphereResource r, Deliver msg) { // A broadcaster#broadcast(msg,Set) may contains null value. if (r == null) { logger.trace("Null AtmosphereResource passed inside a Set"); return false; } if (bc.hasPerRequestFilters()) { BroadcastAction a = bc.filter(r, msg.message, msg.originalMessage); if (a.action() == BroadcastAction.ACTION.ABORT) { return false; } msg.message = a.message(); } return true; } private Object callable(Object msg) { if (Callable.class.isAssignableFrom(msg.getClass())) { try { return Callable.class.cast(msg).call(); } catch (Exception e) { logger.warn("Callable exception", e); return null; } } return msg; } protected void executeAsyncWrite(final AsyncWriteToken token) { boolean notifyListeners = true; boolean lostCandidate = false; if (token.resource == null) throw new NullPointerException(); final AtmosphereResourceEventImpl event = (AtmosphereResourceEventImpl) token.resource.getAtmosphereResourceEvent(); final AtmosphereResourceImpl r = AtmosphereResourceImpl.class.cast(token.resource); final boolean willBeResumed = Utils.resumableTransport(r.transport()); List listeners = willBeResumed ? new ArrayList() : EMPTY_LISTENERS; final AtmosphereRequest request = r.getRequest(false); try { event.setMessage(token.msg); // Make sure we cache the message in case the AtmosphereResource has been cancelled, resumed or the client disconnected. if (!isAtmosphereResourceValid(r)) { logger.trace("AtmosphereResource {} state is invalid for Broadcaster {}. Message will be cached", r.uuid(), name); removeAtmosphereResource(r, false); return; } bc.getBroadcasterCache().clearCache(getID(), r != null ? r.uuid() : BroadcasterCache.NULL, token.cache); try { request.setAttribute(getID(), token.future); request.setAttribute(MAX_INACTIVE, System.currentTimeMillis()); request.setAttribute(usingTokenIdForAttribute, token); if (willBeResumed && !r.atmosphereResourceEventListener().isEmpty()) { listeners.addAll(r.atmosphereResourceEventListener()); } prepareInvokeOnStateChange(r, event); } catch (Throwable t) { logger.debug("Invalid AtmosphereResource state {}. The connection has been remotely" + " closed and message {} will be added to the configured BroadcasterCache for later retrieval", r.uuid(), event.getMessage()); logger.trace("If you are using Tomcat 7.0.22 and lower, you're most probably hitting http://is.gd/NqicFT"); logger.trace("ApplicationConfig.CACHE_MESSAGE_ON_IO_FLUSH_EXCEPTION {}", cacheOnIOFlushException, t); lostCandidate = cacheOnIOFlushException ? cacheOnIOFlushException : cacheMessageOnIOException(t); // The Request/Response associated with the AtmosphereResource has already been written and commited removeAtmosphereResource(r, false); r.removeFromAllBroadcasters(); event.setCancelled(true); event.setThrowable(t); r.setIsInScope(false); return; } try { request.setAttribute(FrameworkConfig.MESSAGE_WRITTEN, "true"); } catch (NullPointerException ex) { // GlassFish and Tomcat may have recycled the request at that moment so the operation will fail. // In that case we don't cache the message as it has been successfully written or cached later // in the code. logger.trace("NPE after the message has been written for {}", r.uuid()); } } finally { if (notifyListeners) { // Long Polling listener will be cleared when the resume() is called. if (willBeResumed) { event.setMessage(token.msg); for (AtmosphereResourceEventListener e : listeners) { e.onBroadcast(event); } // Listener wil be called later } else if (!event.isResumedOnTimeout()) { r.notifyListeners(); } } if (token.lastBroadcasted()) { notifyBroadcastListener(); } if (token.future != null) token.future.done(); if (lostCandidate) { cacheLostMessage(r, token, true); } try { request.removeAttribute(getID()); request.removeAttribute(usingTokenIdForAttribute); } catch (NullPointerException ex) { logger.trace("NPE after the message has been written for {}", r.uuid()); } token.destroy(); } } protected boolean cacheMessageOnIOException(Throwable cause) { for (StackTraceElement element : cause.getStackTrace()) { if (element.getMethodName().equals("flush") || element.getMethodName().equals("flushBuffer")) { return false; } } return true; } protected boolean checkCachedAndPush(final AtmosphereResource r, final AtmosphereResourceEvent e) { boolean cache = retrieveTrackedBroadcast(r, e); if (!cache) return false; if (!((List) e.getMessage()).isEmpty()) { logger.debug("Sending cached message {} to {}", e.getMessage(), r.uuid()); List cacheMessages = (List) e.getMessage(); BroadcasterFuture f = new BroadcasterFuture(e.getMessage(), 1); LinkedList filteredMessage = new LinkedList(); LinkedList filteredMessageClone = null; Deliver deliver; Object newMessage; for (Object o : cacheMessages) { newMessage = filter(o); if (newMessage == null) { continue; } deliver = new Deliver(newMessage, r, f, o); // Can be aborted by a Filter if (!perRequestFilter(r, deliver)) { continue; } if (deliver.message != null) { filteredMessage.addLast(deliver.message); } } if (filteredMessage.size() == 0) { return false; } e.setMessage(filteredMessage); final boolean willBeResumed = Utils.resumableTransport(r.transport()); if (willBeResumed) { filteredMessageClone = (LinkedList) filteredMessage.clone(); } List listeners = willBeResumed ? new ArrayList() : EMPTY_LISTENERS; AtmosphereResourceImpl rImpl = AtmosphereResourceImpl.class.cast(r); if (willBeResumed && !rImpl.atmosphereResourceEventListener().isEmpty()) { listeners.addAll(rImpl.atmosphereResourceEventListener()); } // Must make sure execute only one thread synchronized (rImpl) { try { rImpl.getRequest().setAttribute(CACHED, "true"); prepareInvokeOnStateChange(r, e); } catch (Throwable t) { // An exception occurred logger.error("Unable to write cached message {} for {}", e.getMessage(), r.uuid()); logger.error("", t); for (Object o : cacheMessages) { bc.getBroadcasterCache().addToCache(getID(), r != null ? r.uuid() : BroadcasterCache.NULL, new BroadcastMessage(o)); } return true; } // If long-polling or JSONP is used we need to set the messages for the event again, because onResume() have cleared them if (willBeResumed) { e.setMessage(filteredMessageClone); } for (AtmosphereResourceEventListener l : willBeResumed ? listeners : rImpl.atmosphereResourceEventListener()) { l.onBroadcast(e); } switch (r.transport()) { case UNDEFINED: case JSONP: case AJAX: case LONG_POLLING: return true; case SSE: break; default: try { r.getResponse().flushBuffer(); } catch (IOException ioe) { logger.trace("", ioe); AtmosphereResourceImpl.class.cast(r)._destroy(); } break; } } } return false; } protected boolean retrieveTrackedBroadcast(final AtmosphereResource r, final AtmosphereResourceEvent e) { logger.trace("Checking cached message for {}", r.uuid()); List missedMsg = bc.getBroadcasterCache().retrieveFromCache(getID(), r.uuid()); if (missedMsg != null && !missedMsg.isEmpty()) { e.setMessage(missedMsg); return true; } return false; } protected void invokeOnStateChange(final AtmosphereResource r, final AtmosphereResourceEvent e) { try { logger.trace("{} is broadcasting to {}", name, r.uuid()); r.getAtmosphereHandler().onStateChange(e); } catch (Throwable t) { if (!InterruptedException.class.isAssignableFrom(t.getClass())) { onException(t, r); } } } protected void prepareInvokeOnStateChange(final AtmosphereResource r, final AtmosphereResourceEvent e) { if (writeTimeoutInSecond != -1) { logger.trace("Registering Write timeout {} for {}", writeTimeoutInSecond, r.uuid()); WriteOperation w = new WriteOperation(r, e, Thread.currentThread()); bc.getScheduledExecutorService().schedule(w, writeTimeoutInSecond, TimeUnit.MILLISECONDS); try { w.call(); } catch (Exception ex) { logger.warn("", ex); } } else { invokeOnStateChange(r, e); } } final class WriteOperation implements Callable { private final AtmosphereResource r; private final AtmosphereResourceEvent e; private AtomicBoolean completed = new AtomicBoolean(); private AtomicBoolean executed = new AtomicBoolean(); private final Thread ioThread; private WriteOperation(AtmosphereResource r, AtmosphereResourceEvent e, Thread ioThread) { this.r = r; this.e = e; this.ioThread = ioThread; } @Override public Object call() throws Exception { if (!completed.getAndSet(true)) { invokeOnStateChange(r, e); logger.trace("Cancelling Write timeout {} for {}", writeTimeoutInSecond, r.uuid()); executed.set(true); } else if (!executed.get()) { // https://github.com/Atmosphere/atmosphere/issues/902 try { ioThread.interrupt(); } catch (Throwable t) { // Swallow, this is already enough embarrassing logger.trace("I/O failure, unable to interrupt the thread", t); } logger.trace("Honoring Write timeout {} for {}", writeTimeoutInSecond, r.uuid()); onException(new IOException("Unable to write after " + writeTimeoutInSecond), r); AtmosphereResourceImpl.class.cast(r).cancel(); } return null; } public void interrupt() { } } public void onException(Throwable t, final AtmosphereResource ar) { onException(t, ar, true); } public void onException(Throwable t, final AtmosphereResource ar, boolean notifyAndCache) { final AtmosphereResourceImpl r = AtmosphereResourceImpl.class.cast(ar); logger.trace("I/O Exception (or related) during execution of the write operation for " + "AtmosphereResource {} and Broadcaster {}. Message will be cached {}", new String[]{ar.uuid(), getID(), String.valueOf(notifyAndCache)}); logger.trace("{}", t); // Remove to prevent other broadcast to re-use it. removeAtmosphereResource(r); if (notifyAndCache) { final AtmosphereResourceEventImpl event = r.getAtmosphereResourceEvent(); event.setThrowable(t); r.notifyListeners(event); r.removeEventListeners(); } if (notifyAndCache) { cacheLostMessage(r, (AsyncWriteToken) r.getRequest(false).getAttribute(usingTokenIdForAttribute), notifyAndCache); } /** * Make sure we resume the connection on every IOException. */ if (bc != null && bc.getAsyncWriteService() != null) { bc.getAsyncWriteService().execute(new Runnable() { @Override public void run() { try { logger.trace("Forcing connection close {}", ar.uuid()); r.resume(); r.close(); } catch (Throwable t) { logger.trace("Was unable to resume a corrupted AtmosphereResource {}", r); logger.trace("Cause", t); } } }); } else { r.resume(); } } /** * Cache the message because an unexpected exception occurred. * * @param r {@link AtmosphereResource} */ public void cacheLostMessage(AtmosphereResource r, boolean force) { AtmosphereRequest request = AtmosphereResourceImpl.class.cast(r).getRequest(false); try { cacheLostMessage(r, (AsyncWriteToken) request.getAttribute(usingTokenIdForAttribute), force); } finally { request.removeAttribute(usingTokenIdForAttribute); } } /** * Cache the message because an unexpected exception occurred. * * @param r {@link AtmosphereResource} */ public void cacheLostMessage(AtmosphereResource r, AsyncWriteToken token) { cacheLostMessage(r, token, false); } /** * Cache the message because an unexpected exception occurred. * * @param r {@link AtmosphereResource} */ public void cacheLostMessage(AtmosphereResource r, AsyncWriteToken token, boolean force) { if (!force) { return; } try { if (token != null && token.originalMessage != null) { bc.getBroadcasterCache().addToCache(getID(), r != null ? r.uuid() : BroadcasterCache.NULL, new BroadcastMessage(String.valueOf(token.future.hashCode()), token.originalMessage)); logger.trace("Lost message cached {}", token.originalMessage); } } catch (Throwable t2) { logger.error("Unable to cache message {} for AtmosphereResource {}", token.originalMessage, r != null ? r.uuid() : ""); logger.error("Unable to cache message", t2); } } @Override public void setSuspendPolicy(long maxSuspendResource, POLICY policy) { this.maxSuspendResource.set(maxSuspendResource); this.policy = policy; } @Override public Future broadcast(Object msg) { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "broadcast(T msg)"); return futureDone(msg); } start(); Object newMsg = filter(msg); if (newMsg == null) { logger.debug("Broadcast Interrupted {}", msg); return futureDone(msg); } int callee = resources.size() == 0 ? 1 : resources.size(); BroadcasterFuture f = new BroadcasterFuture(newMsg, callee); dispatchMessages(new Deliver(newMsg, f, msg)); return f; } protected BroadcasterFuture futureDone(Object msg) { notifyBroadcastListener(); return (new BroadcasterFuture(msg)).done(); } protected void dispatchMessages(Deliver e) { messages.offer(e); if (dispatchThread.get() == 0) { dispatchThread.incrementAndGet(); getBroadcasterConfig().getExecutorService().submit(getBroadcastHandler()); } } /** * Invoke the {@link BroadcastFilter} * * @param msg * @return */ protected Object filter(Object msg) { BroadcastAction a = bc.filter(msg); if (a.action() == BroadcastAction.ACTION.ABORT || msg == null) return null; else return a.message(); } @Override public Future broadcast(Object msg, AtmosphereResource r) { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "broadcast(T msg, AtmosphereResource r"); return futureDone(msg); } start(); Object newMsg = filter(msg); if (newMsg == null) return futureDone(msg); BroadcasterFuture f = new BroadcasterFuture(newMsg, 1); dispatchMessages(new Deliver(newMsg, r, f, msg)); return f; } @Override public Future broadcastOnResume(Object msg) { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "broadcastOnResume(T msg)"); return futureDone(msg); } start(); Object newMsg = filter(msg); if (newMsg == null) return futureDone(msg); BroadcasterFuture f = new BroadcasterFuture(newMsg, resources.size()); broadcastOnResume.offer(new Deliver(newMsg, f, msg)); return f; } protected void broadcastOnResume(AtmosphereResource r) { for (Deliver e : broadcastOnResume) { e.async = false; push(new Deliver(r, e)); } if (resources.isEmpty()) { broadcastOnResume.clear(); } } @Override public Future broadcast(Object msg, Set subset) { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "broadcast(T msg, Set subset)"); return futureDone(msg); } start(); Object newMsg = filter(msg); if (newMsg == null) return futureDone(msg); BroadcasterFuture f = new BroadcasterFuture(null, newMsg, subset.size()); dispatchMessages(new Deliver(newMsg, subset, f, msg)); return f; } @Override public Broadcaster addAtmosphereResource(AtmosphereResource r) { try { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "addAtmosphereResource(AtmosphereResource r"); return this; } start(); if (scope == SCOPE.REQUEST && requestScoped.getAndSet(true)) { throw new IllegalStateException("Broadcaster " + this + " cannot be used as its scope is set to REQUEST"); } // To avoid excessive synchronization, we allow resources.size() to get larger that maxSuspendResource if (maxSuspendResource.get() > 0 && resources.size() >= maxSuspendResource.get()) { // Resume the first in. if (policy == POLICY.FIFO) { // TODO handle null return from poll() AtmosphereResource resource = resources.poll(); try { logger.warn("Too many resource. Forcing resume of {} ", resource.uuid()); resource.resume(); } catch (Throwable t) { logger.warn("failed to resume resource {} ", resource, t); } } else if (policy == POLICY.REJECT) { throw new RejectedExecutionException(String.format("Maximum suspended AtmosphereResources %s", maxSuspendResource)); } } if (!r.isSuspended()) { logger.warn("AtmosphereResource {} is not suspended. If cached messages exists, this may cause unexpected situation. Suspend first", r.uuid()); } if (!backwardCompatible && resources.contains(r)) { boolean duplicate = r.transport() != AtmosphereResource.TRANSPORT.WEBSOCKET || AtmosphereResourceImpl.class.cast(r).getRequest(false).getAttribute(INJECTED_ATMOSPHERE_RESOURCE) != null; if (duplicate) { AtmosphereResourceImpl dup = (AtmosphereResourceImpl) config.resourcesFactory().find(r.uuid()); if (dup != null && dup != r) { logger.warn("Duplicate resource {}. Could be caused by a dead connection not detected by your server. Replacing the old one with the fresh one", r.uuid()); AtmosphereResourceImpl.class.cast(dup).dirtyClose(); } else { logger.debug("Duplicate resource {}", r.uuid()); return this; } } else { logger.debug("Duplicate resource {}", r.uuid()); return this; } } // Only synchronize if we have a valid BroadcasterCache if (!bc.getBroadcasterCache().getClass().equals(BroadcasterCache.DEFAULT.getClass().getName())) { // In case we are adding messages to the cache, we need to make sure the operation is done before. synchronized (resources) { cacheAndSuspend(r); } } else { cacheAndSuspend(r); } } finally { // OK reset if (resources.size() > 0) { synchronized (awaitBarrier) { awaitBarrier.notifyAll(); } } } return this; } /** * Look in the cache to see if there are messages available, and take the appropriate actions. * * @param r AtmosphereResource */ protected void cacheAndSuspend(AtmosphereResource r) { // In case the connection is closed, for whatever reason if (!isAtmosphereResourceValid(r)) { logger.debug("Unable to add AtmosphereResource {}", r.uuid()); return; } boolean wasResumed = checkCachedAndPush(r, r.getAtmosphereResourceEvent()); if (!wasResumed && isAtmosphereResourceValid(r)) { logger.trace("Associating AtmosphereResource {} with Broadcaster {}", r.uuid(), getID()); String parentUUID = r.transport().equals(AtmosphereResource.TRANSPORT.WEBSOCKET) ? (String) AtmosphereResourceImpl.class.cast(r).getRequest(false).getAttribute(SUSPENDED_ATMOSPHERE_RESOURCE_UUID) : null; if (!backwardCompatible && parentUUID != null) { AtmosphereResource p = config.resourcesFactory().find(parentUUID); if (p != null && !resources.contains(p)) { notifyAndAdd(p); } else if (p == null) { notifyAndAdd(r); } else { logger.trace("AtmosphereResource {} was already mapped to {}", r.uuid(), parentUUID); } } else { notifyAndAdd(r); } } else if (!wasResumed) { logger.debug("Unable to add AtmosphereResource {} to {}", r.uuid(), name); } } protected void notifyAndAdd(AtmosphereResource r) { resources.add(r); r.addBroadcaster(this); notifyOnAddAtmosphereResourceListener(r); } private boolean isAtmosphereResourceValid(AtmosphereResource r) { return !r.isResumed() && !r.isCancelled() && AtmosphereResourceImpl.class.cast(r).isInScope(); } protected void entryDone(final BroadcasterFuture f) { notifyBroadcastListener(); if (f != null) f.done(); } protected void notifyBroadcastListener() { for (BroadcasterListener b : broadcasterListeners) { try { b.onComplete(this); } catch (Exception ex) { logger.warn("", ex); } } } protected void notifyOnAddAtmosphereResourceListener(AtmosphereResource r) { for (BroadcasterListener b : broadcasterListeners) { try { b.onAddAtmosphereResource(this, r); } catch (Exception ex) { logger.warn("", ex); } } } protected void notifyOnRemoveAtmosphereResourceListener(AtmosphereResource r) { for (BroadcasterListener b : broadcasterListeners) { try { b.onRemoveAtmosphereResource(this, r); } catch (Exception ex) { logger.warn("", ex); } } } protected void notifyOnMessage(Deliver deliver) { for (BroadcasterListener b : broadcasterListeners) { try { b.onMessage(this, deliver); } catch (Exception ex) { logger.warn("", ex); } } } @Override public Broadcaster removeAtmosphereResource(AtmosphereResource r) { return removeAtmosphereResource(r, true); } protected Broadcaster removeAtmosphereResource(AtmosphereResource r, boolean executeDone) { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "removeAtmosphereResource(AtmosphereResource r)"); return this; } boolean removed = resources.remove(r); if (removed) { if (r.isSuspended()) { logger.trace("Excluded from {} : {}", getID(), r.uuid()); bc.getBroadcasterCache().excludeFromCache(getID(), r); } notifyOnRemoveAtmosphereResourceListener(r); } else { logger.trace("Unable to remove {} from {}", r.uuid(), getID()); } r.removeBroadcaster(this); if (!removed) return this; logger.trace("Removing AtmosphereResource {} for Broadcaster {}", r.uuid(), name); writeQueues.remove(r.uuid()); // Here we need to make sure we aren't in the process of broadcasting and unlock the Future. if (executeDone) { AtmosphereResourceImpl aImpl = AtmosphereResourceImpl.class.cast(r); BroadcasterFuture f = (BroadcasterFuture) aImpl.getRequest(false).getAttribute(getID()); if (f != null && !f.isDone() && !f.isCancelled()) { aImpl.getRequest(false).removeAttribute(getID()); entryDone(f); } } return this; } @Override public void setBroadcasterConfig(BroadcasterConfig bc) { this.bc = bc; } @Override public BroadcasterConfig getBroadcasterConfig() { return bc; } @Override public Future delayBroadcast(Object o) { return delayBroadcast(o, 0, null); } @Override public Future delayBroadcast(final Object o, long delay, TimeUnit t) { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "delayBroadcast(final T o, long delay, TimeUnit t)"); return null; } start(); final Object msg = filter(o); if (msg == null) return null; final BroadcasterFuture future = new BroadcasterFuture(msg); final Deliver e = new Deliver(msg, future, o); Future f; if (delay > 0) { f = bc.getScheduledExecutorService().schedule(new Callable() { public Object call() throws Exception { delayedBroadcast.remove(e); if (Callable.class.isAssignableFrom(o.getClass())) { try { Object r = Callable.class.cast(o).call(); final Object msg = filter(r); if (msg != null) { Deliver deliver = new Deliver(msg, future, r); push(deliver); } return msg; } catch (Exception e1) { logger.error("", e); } } final Object msg = filter(o); final Deliver e = new Deliver(msg, future, o); push(e); return msg; } }, delay, t); e.future = new BroadcasterFuture(f, msg); } delayedBroadcast.offer(e); return future; } @Override public Future scheduleFixedBroadcast(final Object o, long period, TimeUnit t) { return scheduleFixedBroadcast(o, 0, period, t); } @Override public Future scheduleFixedBroadcast(final Object o, long waitFor, long period, TimeUnit t) { if (destroyed.get()) { logger.debug(DESTROYED, getID(), "scheduleFixedBroadcast(final Object o, long waitFor, long period, TimeUnit t)"); return null; } start(); if (period == 0 || t == null) { return null; } final Object msg = filter(o); if (msg == null) return null; final BroadcasterFuture f = new BroadcasterFuture(msg); return (Future) bc.getScheduledExecutorService().scheduleWithFixedDelay(new Runnable() { public void run() { if (Callable.class.isAssignableFrom(o.getClass())) { try { Object r = Callable.class.cast(o).call(); final Object msg = filter(r); if (msg != null) { Deliver deliver = new Deliver(msg, f, r); push(deliver); } return; } catch (Exception e) { logger.error("", e); } } final Object msg = filter(o); final Deliver e = new Deliver(msg, f, o); push(e); } }, waitFor, period, t); } @Override public String toString() { return new StringBuilder().append("\n\tName: ").append(name) .append("\n\tAtmosphereResource: ").append(resources.size()) .append("\n\tBroadcasterCache ").append(bc.getBroadcasterCache()) .toString(); } protected final static class AsyncWriteToken { AtmosphereResource resource; Object msg; BroadcasterFuture future; Object originalMessage; CacheMessage cache; AtomicInteger count; public AsyncWriteToken(AtmosphereResource resource, Object msg, BroadcasterFuture future, Object originalMessage, AtomicInteger count) { this.resource = resource; this.msg = msg; this.future = future; this.originalMessage = originalMessage; this.count = count; } public AsyncWriteToken(AtmosphereResource resource, Object msg, BroadcasterFuture future, Object originalMessage, CacheMessage cache, AtomicInteger count) { this.resource = resource; this.msg = msg; this.future = future; this.originalMessage = originalMessage; this.cache = cache; this.count = count; } public void destroy() { this.resource = null; this.msg = null; this.future = null; this.originalMessage = null; } public boolean lastBroadcasted() { return count.decrementAndGet() == 0; } @Override public String toString() { return "AsyncWriteToken{" + "resource=" + resource + ", msg=" + msg + ", future=" + future + '}'; } } private long translateTimeUnit(long period, TimeUnit tu) { if (period == -1) return period; switch (tu) { case SECONDS: return TimeUnit.MILLISECONDS.convert(period, TimeUnit.SECONDS); case MINUTES: return TimeUnit.MILLISECONDS.convert(period, TimeUnit.MINUTES); case HOURS: return TimeUnit.MILLISECONDS.convert(period, TimeUnit.HOURS); case DAYS: return TimeUnit.MILLISECONDS.convert(period, TimeUnit.DAYS); case MILLISECONDS: return period; case MICROSECONDS: return TimeUnit.MILLISECONDS.convert(period, TimeUnit.MICROSECONDS); case NANOSECONDS: return TimeUnit.MILLISECONDS.convert(period, TimeUnit.NANOSECONDS); } return period; } boolean notifyOnPreDestroy() { for (BroadcasterListener b : broadcasterListeners) { try { b.onPreDestroy(this); } catch (RuntimeException ex) { if (BroadcasterListener.BroadcastListenerException.class.isAssignableFrom(ex.getClass())) { logger.trace("onPreDestroy", ex); return true; } logger.warn("onPreDestroy", ex); } } return false; } public Collection broadcasterListeners() { return broadcasterListeners; } public BroadcasterLifeCyclePolicy lifeCyclePolicy() { return lifeCyclePolicy; } public ConcurrentLinkedQueue lifeCycleListeners() { return lifeCycleListeners; } public BlockingQueue messages() { return messages; } public ConcurrentHashMap writeQueues() { return writeQueues; } public POLICY policy() { return policy; } public boolean outOfOrderBroadcastSupported() { return outOfOrderBroadcastSupported.get(); } public AtomicBoolean recentActivity() { return recentActivity; } public LifecycleHandler lifecycleHandler() { return lifecycleHandler; } public DefaultBroadcaster lifecycleHandler(LifecycleHandler lifecycleHandler) { this.lifecycleHandler = lifecycleHandler; return this; } public Future currentLifecycleTask() { return currentLifecycleTask; } public DefaultBroadcaster currentLifecycleTask(Future currentLifecycleTask) { this.currentLifecycleTask = currentLifecycleTask; return this; } }