/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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.apache.openejb; import org.apache.openejb.loader.SystemInstance; import org.apache.openejb.util.LogCategory; import org.apache.openejb.util.Logger; import org.apache.xbean.finder.ResourceFinder; import java.io.IOException; import java.net.URL; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.List; import static java.util.Arrays.asList; /** * The Extensions API mimics the equivalent CDI Extension/@Observes API * <p/> * Via the Extensions/@Obverves API it is possible to listen for any number of * internal events fired by OpenEJB/TomEE during the life of the server and * deployment of applications. * <p/> * Extensions are any java class that have one or more @Observes methods using the following format: * <p/> * - public void <any-name>(@Observes <any-type> event) * <p/> * Extensions can be registered in any number of ways: * <p/> * 1. Via a META-INF/org.apache.openejb.extension text file containing the name of exactly one * 2. Via a <Service> tag in the server's openejb.xml or tomee.xml configuration file * 3. Via a <Service> tag in a META-INF/resources.xml in the application * 4. Directly calling SystemInstance#addObserver(Object) * * @see org.apache.openejb.observer.ObserverManager * @see org.apache.openejb.observer.Observes * @see org.apache.openejb.loader.SystemInstance#fireEvent(Object) * @see org.apache.openejb.loader.SystemInstance#addObserver(Object) */ public final class Extensions { private static final Logger LOGGER = Logger.getInstance(LogCategory.OPENEJB, Extensions.class); private Extensions() { // no-op } public static Collection<Class<?>> findExtensions(final Finder finder) { try { return finder.findAvailableClasses("org.apache.openejb.extension"); } catch (final IOException e) { LOGGER.error("Extension scanning of 'META-INF/org.apache.openejb.extension' files failed", e); return Collections.emptySet(); } } public static void installExtensions(final Finder finder) { try { final List<Class<?>> classes = finder.findAvailableClasses("org.apache.openejb.extension"); addExtensions(classes); } catch (final IOException e) { LOGGER.error("Extension scanning of 'META-INF/org.apache.openejb.extension' files failed", e); } } public static void addExtensions(final ClassLoader loader, final Collection<String> classes) { for (final String clazz : classes) { try { final Object object = loader.loadClass(clazz).getConstructor().newInstance(); SystemInstance.get().addObserver(object); } catch (final Throwable t) { LOGGER.error("Extension construction failed" + clazz, t); } } } public static void addExtensions(final Collection<Class<?>> classes) { for (final Class<?> clazz : classes) { try { final Object object = clazz.getConstructor().newInstance(); SystemInstance.get().addObserver(object); } catch (final Throwable t) { LOGGER.error("Extension construction failed" + clazz.getName(), t); } } } public static class Finder extends ResourceFinder { public Finder(final String path, final boolean useLoader, final URL... urls) { super(path, !useLoader && urls == null ? new URL[0] : urls); } // ensure we support multiple class by file public List<String> findAvailableStrings(final String uri) throws IOException { List<String> strings = super.findAvailableStrings(uri); if (!strings.isEmpty()) { final List<String> copy = new ArrayList<>(strings); strings.clear(); for (final String s : copy) { strings.addAll(asList(s.replace("\r", "").split("\n"))); } } return strings; } } }