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

com.thaiopensource.util.Service Maven / Gradle / Ivy

The newest version!
package com.thaiopensource.util;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.io.UnsupportedEncodingException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;

public final class Service {
  private final Class serviceClass;
  private final Enumeration configFiles;
  private Iterator classNames = null;
  private final List providers = new ArrayList();
  private Loader loader;

  private class ProviderIterator implements Iterator {
    private int nextIndex = 0;

    public boolean hasNext() {
      return nextIndex < providers.size() || moreProviders();
    }

    public T next() {
      try {
	return providers.get(nextIndex++);
      }
      catch (IndexOutOfBoundsException e) {
	throw new NoSuchElementException();
      }
    }

    public void remove() {
      throw new UnsupportedOperationException();
    }
  }

  private static class Singleton implements Enumeration {
    private T obj;
    private Singleton(T obj) {
      this.obj = obj;
    }

    public boolean hasMoreElements() {
      return obj != null;
    }

    public T nextElement() {
      if (obj == null)
	throw new NoSuchElementException();
      T tem = obj;
      obj = null;
      return tem;
    }
  }

  // JDK 1.1
  private static class Loader {
    Enumeration getResources(String resName) {
      ClassLoader cl = Loader.class.getClassLoader();
      URL url;
      if (cl == null)
	url = ClassLoader.getSystemResource(resName);
      else
	url = cl.getResource(resName);
      return new Singleton(url);
    }

    Class loadClass(String name) throws ClassNotFoundException {
      return Class.forName(name);
    }
  }

  // JDK 1.2+
  private static class Loader2 extends Loader {
    private ClassLoader cl;

    Loader2() {
      cl = Loader2.class.getClassLoader();
      // If the thread context class loader has the class loader
      // of this class as an ancestor, use the thread context class
      // loader.  Otherwise, the thread context class loader
      // probably hasn't been set up properly, so don't use it.
      ClassLoader clt = Thread.currentThread().getContextClassLoader();
      for (ClassLoader tem = clt; tem != null; tem = tem.getParent())
	if (tem == cl) {
	  cl = clt;
	  break;
	}
    }

    Enumeration getResources(String resName) {
      try {
        Enumeration resources = cl.getResources(resName);
        if (resources.hasMoreElements())
	  return resources;
        // Some application servers apparently do not implement findResources
        // in their class loaders, so fall back to getResource.
        return new Singleton(cl.getResource(resName));
      }
      catch (IOException e) {
	return new Singleton(null);
      }
    }

    Class loadClass(String name) throws ClassNotFoundException {
      return Class.forName(name, true, cl);
    }
  }

  static public  Service newInstance(Class cls) {
    return new Service(cls);
  }

  private Service(Class cls) {
    try {
      loader = new Loader2();
    }
    catch (NoSuchMethodError e) {
      loader = new Loader();
    }
    serviceClass = cls;
    String resName = "META-INF/services/" + serviceClass.getName();
    configFiles = loader.getResources(resName);
  }

  public Iterator getProviders() {
    return new ProviderIterator();
  }

  synchronized private boolean moreProviders() {
    for (;;) {
      while (classNames == null) {
	if (!configFiles.hasMoreElements())
	  return false;
	classNames = parseConfigFile(configFiles.nextElement());
      }
      while (classNames.hasNext()) {
	String className = classNames.next();
	try {
	  Class cls = loader.loadClass(className);
	  Object obj = cls.newInstance();
	  if (serviceClass.isInstance(obj)) {
	    providers.add(serviceClass.cast(obj));
	    return true;
	  }
	}
	catch (ClassNotFoundException e) { }
	catch (InstantiationException e) { }
	catch (IllegalAccessException e) { }
	catch (LinkageError e) { }
      }
      classNames = null;
    }
  }

  private static final int START = 0;
  private static final int IN_NAME = 1;
  private static final int IN_COMMENT = 2;

  private static Iterator parseConfigFile(URL url) {
    try {
      InputStream in = url.openStream();
      Reader r;
      try {
	r = new InputStreamReader(in, "UTF-8");
      }
      catch (UnsupportedEncodingException e) {
	r = new InputStreamReader(in, "UTF8");
      }
      r = new BufferedReader(r);
      List tokens = new ArrayList();
      StringBuilder tokenBuf = new StringBuilder();
      int state = START;
      for (;;) {
	int n = r.read();
	if (n < 0)
	  break;
	char c = (char)n;
	switch (c) {
	case '\r':
	case '\n':
	  state = START;
	  break;
	case ' ':
	case '\t':
	  break;
	case '#':
	  state = IN_COMMENT;
	  break;
	default:
	  if (state != IN_COMMENT) {
	    state = IN_NAME;
	    tokenBuf.append(c);
	  }
	  break;
	}
	if (tokenBuf.length() != 0 && state != IN_NAME) {
	  tokens.add(tokenBuf.toString());
	  tokenBuf.setLength(0);
	}
      }
      if (tokenBuf.length() != 0)
	tokens.add(tokenBuf.toString());
      return tokens.iterator();
    }
    catch (IOException e) {
      return null;
    }
  }


  public static void main(String[] args) throws ClassNotFoundException {
    Service svc = Service.newInstance(Class.forName(args[0]));
    for (Iterator iter = svc.getProviders(); iter.hasNext();)
      System.out.println(iter.next().getClass().getName());
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy