Package org.jvnet.hk2.component.classmodel

Source Code of org.jvnet.hk2.component.classmodel.InhabitantsParsingContextGenerator$Locator

/*
* 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.
* <p/>
* 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.
* <p/>
* 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<String, InhabitantsScanner> metaInfScanners =
            new LinkedHashMap<String, InhabitantsScanner>();

    /**
     * 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<String> annotations = new HashSet<String>();
        annotations.add(Contract.class.getCanonicalName());
        annotations.add(Service.class.getCanonicalName());
        annotations.add("org.jvnet.hk2.config.Configured");

        builder.config(new ParsingConfig() {
            final Set<String> empty = Collections.emptySet();

            public Set<String> getAnnotationsOfInterest() {
                return empty;
            }

            public Set<String> 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<File> files) throws IOException {
        for (File file : files) {
            parse(file);
        }
    }

    /**
     * Retrieves the parsing context that can be used for model generation elsewhere.
     * <b>Note that this can be called at most once and then this instance is implicitly
     * closed.</b>
     *
     * @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<InhabitantsScanner> 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;
      }
    }

}
TOP

Related Classes of org.jvnet.hk2.component.classmodel.InhabitantsParsingContextGenerator$Locator

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.