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

org.jvnet.hk2.component.classmodel.InhabitantsParsingContextGenerator Maven / Gradle / Ivy

There is a newer version: 2.0.5
Show newest version
/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright (c) 2007-2011 Oracle and/or its affiliates. All rights reserved.
 *
 * The contents of this file are subject to the terms of either the GNU
 * General Public License Version 2 only ("GPL") or the Common Development
 * and Distribution License("CDDL") (collectively, the "License").  You
 * may not use this file except in compliance with the License.  You can
 * obtain a copy of the License at
 * https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
 * or packager/legal/LICENSE.txt.  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 at packager/legal/LICENSE.txt.
 *
 * GPL Classpath Exception:
 * Oracle designates this particular file as subject to the "Classpath"
 * exception as provided by Oracle in the GPL Version 2 section of the License
 * file that accompanied this code.
 *
 * 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]"
 *
 * Contributor(s):
 * If you wish your version of this file to be governed by only the CDDL or
 * only the GPL Version 2, indicate your decision by adding "[Contributor]
 * elects to include this software in this distribution under the [CDDL or GPL
 * Version 2] license."  If you don't indicate a single choice of license, a
 * recipient has the option to distribute your version of this file under
 * either the CDDL, the GPL Version 2 or to extend the choice of license to
 * its licensees as provided above.  However, if you add GPL Version 2 code
 * and therefore, elected the GPL Version 2 license, then the option applies
 * only if the new code is made subject to such option by the copyright
 * holder.
 */
package org.jvnet.hk2.component.classmodel;

import java.io.Closeable;
import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.logging.Level;
import java.util.logging.Logger;

import org.glassfish.hk2.classmodel.reflect.Parser;
import org.glassfish.hk2.classmodel.reflect.ParsingContext;
import org.glassfish.hk2.classmodel.reflect.util.ParsingConfig;
import org.glassfish.hk2.classmodel.reflect.util.ResourceLocator;
import org.jvnet.hk2.annotations.Contract;
import org.jvnet.hk2.annotations.Service;
import org.jvnet.hk2.component.Habitat;

import com.sun.hk2.component.InhabitantsScanner;

/**
 * Responsible for generating the collection of inhabitants, decoupling from
 * implementation detail for the caller.
 * 

* The caller is expected to continually build up the InhabitantsGenerator * context by calling add*(), followed by calling getModelInhabitants() to * obtain the progenitors of the inhabitants. *

* There are two ways to close this instance, either through {@link #getContext()} * or through calling {@link #close()} directly. * * @author Jerome Dochez * @author Jeff Trent * @since 3.1 */ public abstract class InhabitantsParsingContextGenerator implements Closeable { private static final Logger logger = Logger .getLogger(InhabitantsParsingContextGenerator.class.getName()); private final Parser parser; private final ParsingContext context; private final LinkedHashMap metaInfScanners = new LinkedHashMap(); /** * Factory for the {@link InhabitantsParsingContextGenerator} * * @param h habitat not currently used; reserved for future use * @return an empty context InhabitantsGenerator */ public static InhabitantsParsingContextGenerator create(Habitat h) { return new InhabitantsParsingContextGenerator(null, null) {}; } /** * Factory for the {@link InhabitantsParsingContextGenerator} * * @param h habitat not currently used; reserved for future use * @param es the executor to use for any async processing (e.g., parsing) * @param inhabitantsClassPath the fully qualified classpath in order to resolve class-model * * @return an empty context InhabitantsGenerator */ public static InhabitantsParsingContextGenerator create(Habitat h, ExecutorService es, ClassPath inhabitantsClassPath) { return new InhabitantsParsingContextGenerator(es, inhabitantsClassPath) {}; } protected InhabitantsParsingContextGenerator(ExecutorService es, final ClassPath inhabitantsClassPath) { // setup the parser ParsingContext.Builder builder = new ParsingContext.Builder(); final Set annotations = new HashSet(); annotations.add(Contract.class.getCanonicalName()); annotations.add(Service.class.getCanonicalName()); annotations.add("org.jvnet.hk2.config.Configured"); builder.config(new ParsingConfig() { final Set empty = Collections.emptySet(); public Set getAnnotationsOfInterest() { return empty; } public Set getTypesOfInterest() { return annotations; } @Override public boolean modelUnAnnotatedMembers() { return false; } }); // optionally provide an executor builder.executorService(es); // optionally provide an inhabitants locator if (null != inhabitantsClassPath) { builder.locator(new Locator(inhabitantsClassPath)); } context = builder.build(); parser = new Parser(context); } /** * Add the collection of files to the current InhabitantsGenerator context. * * @param files the files to parse. * @throws IOException */ public void parse(Collection files) throws IOException { for (File file : files) { parse(file); } } /** * Retrieves the parsing context that can be used for model generation elsewhere. * Note that this can be called at most once and then this instance is implicitly * closed. * * @return the parsing context given the code sources provided */ public ParsingContext getContext() { try { parser.awaitTermination(); } catch (InterruptedException e) { close(); throw new RuntimeException(e); } return context; } /** * @return the collection of {@link InhabitantsScanner}s being maintained */ public Collection getInhabitantsScanners() { return Collections.unmodifiableCollection(metaInfScanners.values()); } protected void addInhabitantsScanner(String name, InhabitantsScanner is) { synchronized (metaInfScanners) { if (!metaInfScanners.containsKey(name)) { metaInfScanners.put(name, is); } } } /** * Eventually we can perform optimizations here instead of a pass-thru to parseAlways. */ public void parse(final File f) throws IOException { parseAlways(parser, f); } protected void parseAlways(Parser parser, final File f) throws IOException { try { parser.parse(f, new Runnable() { public void run() { logger.log(Level.FINER, "Finished introspecting {0}", f.getName()); } }); } catch (IOException e) { logger.log(Level.FINE, "problem during parsing - closing prematurely", e); close(); } } @Override public void close() { if (null != parser) { parser.close(); } } private static class Locator implements ResourceLocator { private final ClassLoader resourceLoader; public Locator(ClassPath inhabitantsClassPath) { try { resourceLoader = new URLClassLoader(inhabitantsClassPath.getRawURLs(), null); if (logger.isLoggable(Level.FINE)) { logger.log(Level.FINER, "resourceLoader is {0}", Arrays.asList(inhabitantsClassPath.getRawURLs()).toString()); } } catch (IOException e) { throw new RuntimeException(e); } } @Override public URL getResource(String name) { if (name.startsWith("java/")) { logger.log(Level.FINE, "skipping {0}", name); return null; // wasteful to parse these } URL resource = resourceLoader.getResource(name); logger.log(Level.FINE, "resource {0} resolved to {1}", new Object[] {name, resource}); return resource; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy