Mercurial Hosting > luan
view src/org/eclipse/jetty/util/component/AggregateLifeCycle.java @ 1041:35e3c864d7a7
make View package local
author | Franklin Schmidt <fschmidt@gmail.com> |
---|---|
date | Fri, 04 Nov 2016 00:47:23 -0600 |
parents | 1d0c304e12b5 |
children |
line wrap: on
line source
// // ======================================================================== // Copyright (c) 1995-2014 Mort Bay Consulting Pty. Ltd. // ------------------------------------------------------------------------ // All rights reserved. This program and the accompanying materials // are made available under the terms of the Eclipse Public License v1.0 // and Apache License v2.0 which accompanies this distribution. // // The Eclipse Public License is available at // http://www.eclipse.org/legal/epl-v10.html // // The Apache License v2.0 is available at // http://www.opensource.org/licenses/apache2.0.php // // You may elect to redistribute this code under either of these licenses. // ======================================================================== // package org.eclipse.jetty.util.component; import java.io.IOException; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.Iterator; import java.util.List; import java.util.concurrent.CopyOnWriteArrayList; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * An AggregateLifeCycle is an {@link LifeCycle} implementation for a collection of contained beans. * <p> * Beans can be added the AggregateLifeCycle either as managed beans or as unmanaged beans. A managed bean is started, stopped and destroyed with the aggregate. * An unmanaged bean is associated with the aggregate for the purposes of {@link #dump()}, but it's lifecycle must be managed externally. * <p> * When a bean is added, if it is a {@link LifeCycle} and it is already started, then it is assumed to be an unmanaged bean. * Otherwise the methods {@link #addBean(Object, boolean)}, {@link #manage(Object)} and {@link #unmanage(Object)} can be used to * explicitly control the life cycle relationship. * <p> * If adding a bean that is shared between multiple {@link AggregateLifeCycle} instances, then it should be started before being added, so it is unmanaged, or * the API must be used to explicitly set it as unmanaged. * <p> */ public class AggregateLifeCycle extends AbstractLifeCycle implements Destroyable, Dumpable { private static final Logger LOG = LoggerFactory.getLogger(AggregateLifeCycle.class); private final List<Bean> _beans=new CopyOnWriteArrayList<Bean>(); private boolean _started=false; private class Bean { Bean(LifeCycle b,boolean managed) { _bean = b; _managed = managed; } final LifeCycle _bean; final boolean _managed; public String toString() { return "{"+_bean+","+_managed+"}"; } } /* ------------------------------------------------------------ */ /** * Start the managed lifecycle beans in the order they were added. * @see org.eclipse.jetty.util.component.AbstractLifeCycle#doStart() */ @Override protected void doStart() throws Exception { for (Bean b:_beans) { if (b._managed) { LifeCycle l = b._bean; if (!l.isRunning()) l.start(); } } // indicate that we are started, so that addBean will start other beans added. _started = true; super.doStart(); } /* ------------------------------------------------------------ */ /** * Stop the joined lifecycle beans in the reverse order they were added. * @see org.eclipse.jetty.util.component.AbstractLifeCycle#doStart() */ @Override protected void doStop() throws Exception { _started = false; super.doStop(); List<Bean> reverse = new ArrayList<Bean>(_beans); Collections.reverse(reverse); for (Bean b:reverse) { if (b._managed) { LifeCycle l = b._bean; if (l.isRunning()) l.stop(); } } } /* ------------------------------------------------------------ */ /** * Destroy the joined Destroyable beans in the reverse order they were added. * @see org.eclipse.jetty.util.component.Destroyable#destroy() */ public void destroy() { List<Bean> reverse = new ArrayList<Bean>(_beans); Collections.reverse(reverse); for (Bean b:reverse) { if (b._bean instanceof Destroyable && b._managed) { Destroyable d = (Destroyable)b._bean; d.destroy(); } } _beans.clear(); } /* ------------------------------------------------------------ */ /** Is the bean contained in the aggregate. * @param bean * @return True if the aggregate contains the bean */ private boolean contains(LifeCycle bean) { for (Bean b:_beans) if (b._bean==bean) return true; return false; } /* ------------------------------------------------------------ */ /** * Add an associated bean. * If the bean is a {@link LifeCycle}, then it will be managed if it is not * already started and umanaged if it is already started. The {@link #addBean(Object, boolean)} * method should be used if this is not correct, or the {@link #manage(Object)} and {@link #unmanage(Object)} * methods may be used after an add to change the status. * @param o the bean object to add * @return true if the bean was added or false if it has already been added. */ public boolean addBean(LifeCycle o) { // beans are joined unless they are started lifecycles return addBean(o,!o.isStarted()); } /* ------------------------------------------------------------ */ /** Add an associated lifecycle. * @param o The lifecycle to add * @param managed True if the LifeCycle is to be joined, otherwise it will be disjoint. * @return true if bean was added, false if already present. */ public boolean addBean(LifeCycle o, boolean managed) { if (contains(o)) return false; Bean b = new Bean(o,managed); _beans.add(b); // Start the bean if we are started if (managed && _started) { try { o.start(); } catch(Exception e) { throw new RuntimeException(e); } } return true; } /* ------------------------------------------------------------ */ /** Get dependent beans * @return List of beans. */ public Collection<Object> getBeans() { // return new ArrayList<Object>(_beans); ArrayList<Object> beans = new ArrayList<Object>(); for (Bean b:_beans) { beans.add(b._bean); } return beans; } /* ------------------------------------------------------------ */ /** * Remove an associated bean. */ public boolean removeBean (LifeCycle o) { Iterator<Bean> i = _beans.iterator(); while(i.hasNext()) { Bean b=i.next(); if (b._bean==o) { _beans.remove(b); return true; } } return false; } public String dump() { return dump(this); } public static String dump(Dumpable dumpable) { StringBuilder b = new StringBuilder(); try { dumpable.dump(b,""); } catch (IOException e) { LOG.warn("",e); } return b.toString(); } private void dump(Appendable out) throws IOException { dump(out,""); } private void dumpThis(Appendable out) throws IOException { out.append(String.valueOf(this)).append(" - ").append(getState()).append("\n"); } public static void dumpObject(Appendable out,Object o) throws IOException { try { if (o instanceof LifeCycle) out.append(String.valueOf(o)).append(" - ").append((AbstractLifeCycle.getState((LifeCycle)o))).append("\n"); else out.append(String.valueOf(o)).append("\n"); } catch(Throwable th) { out.append(" => ").append(th.toString()).append('\n'); } } public void dump(Appendable out,String indent) throws IOException { dumpThis(out); int size=_beans.size(); if (size==0) return; int i=0; for (Bean b : _beans) { i++; out.append(indent).append(" +- "); if (b._managed) { if (b._bean instanceof Dumpable) ((Dumpable)b._bean).dump(out,indent+(i==size?" ":" | ")); else dumpObject(out,b._bean); } else dumpObject(out,b._bean); } if (i!=size) out.append(indent).append(" |\n"); } public static void dump(Appendable out,String indent,Collection<?>... collections) throws IOException { if (collections.length==0) return; int size=0; for (Collection<?> c : collections) size+=c.size(); if (size==0) return; int i=0; for (Collection<?> c : collections) { for (Object o : c) { i++; out.append(indent).append(" +- "); if (o instanceof Dumpable) ((Dumpable)o).dump(out,indent+(i==size?" ":" | ")); else dumpObject(out,o); } if (i!=size) out.append(indent).append(" |\n"); } } }