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

com.tvd12.reflections.util.ConfigurationBuilder Maven / Gradle / Ivy

The newest version!
package com.tvd12.reflections.util;

import java.net.URL;
import java.util.Collection;
import java.util.List;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadFactory;
import java.util.function.Predicate;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;

import com.tvd12.reflections.Configuration;
import com.tvd12.reflections.Reflections;
import com.tvd12.reflections.ReflectionsException;
import com.tvd12.reflections.adapters.JavaReflectionAdapter;
import com.tvd12.reflections.adapters.JavassistAdapter;
import com.tvd12.reflections.adapters.MetadataAdapter;
import com.tvd12.reflections.concurrent.ThreadFactoryBuilder;
import com.tvd12.reflections.scanners.Scanner;
import com.tvd12.reflections.scanners.SubTypesScanner;
import com.tvd12.reflections.scanners.TypeAnnotationsScanner;
import com.tvd12.reflections.serializers.Serializer;
import com.tvd12.reflections.serializers.XmlSerializer;

/**
 * a fluent builder for {@link com.tvd12.reflections.Configuration}, to be used for constructing a {@link com.tvd12.reflections.Reflections} instance
 * 

usage: *

 *      new Reflections(
 *          new ConfigurationBuilder()
 *              .filterInputsBy(new FilterBuilder().include("your project's common package prefix here..."))
 *              .setUrls(ClasspathHelper.forClassLoader())
 *              .setScanners(new SubTypesScanner(), new TypeAnnotationsScanner().filterResultsBy(myClassAnnotationsFilter)));
 * 
*
{@link #executorService} is used optionally used for parallel scanning. if value is null then scanning is done in a simple for loop *

defaults: accept all for {@link #inputsFilter}, * {@link #executorService} is null, * {@link #serializer} is {@link com.tvd12.reflections.serializers.XmlSerializer} */ /*lazy*/ @SuppressWarnings("rawtypes") public class ConfigurationBuilder implements Configuration { @Nonnull private Set scanners; @Nonnull private Set urls; protected MetadataAdapter metadataAdapter; @Nullable private Predicate inputsFilter; /*lazy*/ private Serializer serializer; @Nullable private ExecutorService executorService; @Nullable private ClassLoader[] classLoaders; private boolean expandSuperTypes = true; public ConfigurationBuilder() { scanners = Sets.newHashSet(new TypeAnnotationsScanner(), new SubTypesScanner()); urls = Sets.newHashSet(); } /** constructs a {@link ConfigurationBuilder} using the given parameters, in a non statically typed way. * that is, each element in {@code params} is guessed by it's type and populated into the configuration. *

    *
  • {@link String} - add urls using {@link ClasspathHelper#forPackage(String, ClassLoader...)} ()}
  • *
  • {@link Class} - add urls using {@link ClasspathHelper#forClass(Class, ClassLoader...)}
  • *
  • {@link ClassLoader} - use these classloaders in order to find urls in ClasspathHelper.forPackage(), ClasspathHelper.forClass() and for resolving types
  • *
  • {@link Scanner} - use given scanner, overriding the default scanners
  • *
  • {@link URL} - add the given url for scanning
  • *
  • {@code Object[]} - flatten and use each element as above
  • *
* * an input {@link FilterBuilder} will be set according to given packages. *

use any parameter type in any order. this constructor uses instanceof on each param and instantiate a {@link ConfigurationBuilder} appropriately. * */ @SuppressWarnings("unchecked") public static ConfigurationBuilder build(final @Nullable Object... params) { ConfigurationBuilder builder = new ConfigurationBuilder(); //flatten List parameters = Lists.newArrayList(); if (params != null) { for (Object param : params) { if (param != null) { if (param.getClass().isArray()) { for (Object p : (Object[]) param) if (p != null) parameters.add(p); } else if (param instanceof Iterable) { for (Object p : (Iterable) param) if (p != null) parameters.add(p); } else parameters.add(param); } } } List loaders = Lists.newArrayList(); for (Object param : parameters) if (param instanceof ClassLoader) loaders.add((ClassLoader) param); ClassLoader[] classLoaders = loaders.isEmpty() ? null : loaders.toArray(new ClassLoader[loaders.size()]); FilterBuilder filter = new FilterBuilder(); List scanners = Lists.newArrayList(); for (Object param : parameters) { if (param instanceof String) { builder.addUrls(ClasspathHelper.forPackage((String) param, classLoaders)); filter.includePackage((String) param); } else if (param instanceof Class) { if (Scanner.class.isAssignableFrom((Class) param)) { try { builder.addScanners(((Scanner) ((Class) param).newInstance())); } catch (Exception e) { /*fallback*/ } } builder.addUrls(ClasspathHelper.forClass((Class) param, classLoaders)); filter.includePackage(((Class) param)); } else if (param instanceof Scanner) { scanners.add((Scanner) param); } else if (param instanceof URL) { builder.addUrls((URL) param); } else if (param instanceof ClassLoader) { /* already taken care */ } else if (param instanceof Predicate) { filter.add((Predicate) param); } else if (param instanceof ExecutorService) { builder.setExecutorService((ExecutorService) param); } else if (Reflections.log != null) { throw new ReflectionsException("could not use param " + param); } } if (builder.getUrls().isEmpty()) { if (classLoaders != null) { builder.addUrls(ClasspathHelper.forClassLoader(classLoaders)); //default urls getResources("") } else { builder.addUrls(ClasspathHelper.forClassLoader()); //default urls getResources("") } } builder.filterInputsBy(filter); if (!scanners.isEmpty()) { builder.setScanners(scanners.toArray(new Scanner[scanners.size()])); } if (!loaders.isEmpty()) { builder.addClassLoaders(loaders); } return builder; } public ConfigurationBuilder forPackages(String... packages) { for (String pkg : packages) { addUrls(ClasspathHelper.forPackage(pkg)); } return this; } @Nonnull public Set getScanners() { return scanners; } /** set the scanners instances for scanning different metadata */ public ConfigurationBuilder setScanners(@Nonnull final Scanner... scanners) { this.scanners.clear(); return addScanners(scanners); } /** set the scanners instances for scanning different metadata */ public ConfigurationBuilder addScanners(final Scanner... scanners) { this.scanners.addAll(Sets.newHashSet(scanners)); return this; } @Nonnull public Set getUrls() { return urls; } /** set the urls to be scanned *

use {@link com.tvd12.reflections.util.ClasspathHelper} convenient methods to get the relevant urls * */ public ConfigurationBuilder setUrls(@Nonnull final Collection urls) { this.urls = Sets.newHashSet(urls); return this; } /** set the urls to be scanned *

use {@link com.tvd12.reflections.util.ClasspathHelper} convenient methods to get the relevant urls * */ public ConfigurationBuilder setUrls(final URL... urls) { this.urls = Sets.newHashSet(urls); return this; } /** add urls to be scanned *

use {@link com.tvd12.reflections.util.ClasspathHelper} convenient methods to get the relevant urls * */ public ConfigurationBuilder addUrls(final Collection urls) { this.urls.addAll(urls); return this; } /** add urls to be scanned *

use {@link com.tvd12.reflections.util.ClasspathHelper} convenient methods to get the relevant urls * */ public ConfigurationBuilder addUrls(final URL... urls) { this.urls.addAll(Sets.newHashSet(urls)); return this; } /** returns the metadata adapter. * if javassist library exists in the classpath, this method returns {@link JavassistAdapter} otherwise defaults to {@link JavaReflectionAdapter}. *

the {@link JavassistAdapter} is preferred in terms of performance and class loading. */ public MetadataAdapter getMetadataAdapter() { if (metadataAdapter != null) return metadataAdapter; else { try { return (metadataAdapter = new JavassistAdapter()); } catch (Throwable e) { if (Reflections.log != null) Reflections.log.warn("could not create JavassistAdapter, using JavaReflectionAdapter", e); return (metadataAdapter = new JavaReflectionAdapter()); } } } /** sets the metadata adapter used to fetch metadata from classes */ public ConfigurationBuilder setMetadataAdapter(final MetadataAdapter metadataAdapter) { this.metadataAdapter = metadataAdapter; return this; } @Nullable public Predicate getInputsFilter() { return inputsFilter; } /** sets the input filter for all resources to be scanned. *

supply a {@link com.google.common.base.Predicate} or use the {@link FilterBuilder}*/ public void setInputsFilter(@Nullable Predicate inputsFilter) { this.inputsFilter = inputsFilter; } /** sets the input filter for all resources to be scanned. *

supply a {@link com.google.common.base.Predicate} or use the {@link FilterBuilder}*/ public ConfigurationBuilder filterInputsBy(Predicate inputsFilter) { this.inputsFilter = inputsFilter; return this; } @Nullable public ExecutorService getExecutorService() { return executorService; } /** sets the executor service used for scanning. */ public ConfigurationBuilder setExecutorService(@Nullable ExecutorService executorService) { this.executorService = executorService; return this; } /** sets the executor service used for scanning to ThreadPoolExecutor with core size as {@link java.lang.Runtime#availableProcessors()} *

default is ThreadPoolExecutor with a single core */ public ConfigurationBuilder useParallelExecutor() { return useParallelExecutor(Runtime.getRuntime().availableProcessors()); } /** sets the executor service used for scanning to ThreadPoolExecutor with core size as the given availableProcessors parameter. * the executor service spawns daemon threads by default. *

default is ThreadPoolExecutor with a single core */ public ConfigurationBuilder useParallelExecutor(final int availableProcessors) { ThreadFactory factory = new ThreadFactoryBuilder().setDaemon(true).setNameFormat("com.tvd12.reflections-scanner-%d").build(); setExecutorService(Executors.newFixedThreadPool(availableProcessors, factory)); return this; } public Serializer getSerializer() { return serializer != null ? serializer : (serializer = new XmlSerializer()); //lazily defaults to XmlSerializer } /** sets the serializer used when issuing {@link com.tvd12.reflections.Reflections#save} */ public ConfigurationBuilder setSerializer(Serializer serializer) { this.serializer = serializer; return this; } /** get class loader, might be used for scanning or resolving methods/fields */ @Nullable public ClassLoader[] getClassLoaders() { return classLoaders; } @Override public boolean shouldExpandSuperTypes() { return expandSuperTypes; } /** * if set to true, Reflections will expand super types after scanning. *

see {@link com.tvd12.reflections.Reflections#expandSuperTypes()} */ public ConfigurationBuilder setExpandSuperTypes(boolean expandSuperTypes) { this.expandSuperTypes = expandSuperTypes; return this; } /** set class loader, might be used for resolving methods/fields */ public void setClassLoaders(@Nullable ClassLoader[] classLoaders) { this.classLoaders = classLoaders; } /** add class loader, might be used for resolving methods/fields */ public ConfigurationBuilder addClassLoader(ClassLoader classLoader) { return addClassLoaders(classLoader); } /** add class loader, might be used for resolving methods/fields */ public ConfigurationBuilder addClassLoaders(ClassLoader... classLoaders) { this.classLoaders = this.classLoaders == null ? classLoaders : ObjectArrays.concat(this.classLoaders, classLoaders, ClassLoader.class); return this; } /** add class loader, might be used for resolving methods/fields */ public ConfigurationBuilder addClassLoaders(Collection classLoaders) { return addClassLoaders(classLoaders.toArray(new ClassLoader[classLoaders.size()])); } }