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

org.jboss.forge.roaster.Roaster Maven / Gradle / Ivy

/*
 * Copyright 2012 Red Hat, Inc. and/or its affiliates.
 *
 * Licensed under the Eclipse Public License version 1.0, available at
 * http://www.eclipse.org/legal/epl-v10.html
 */

package org.jboss.forge.roaster;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
import java.util.ServiceLoader;

import org.jboss.forge.roaster.model.JavaType;
import org.jboss.forge.roaster.model.JavaUnit;
import org.jboss.forge.roaster.model.source.JavaSource;
import org.jboss.forge.roaster.spi.FormatterProvider;
import org.jboss.forge.roaster.spi.JavaParser;

/**
 * Responsible for parsing data into new {@link JavaType} instances.
 *
 * @author Lincoln Baxter, III
 */
public final class Roaster
{
   private Roaster()
   {
   }

   private static List parsers;
   private static List formatters;

   private static List getParsers()
   {
      synchronized (Roaster.class)
      {
         if (parsers == null || parsers.isEmpty())
         {
            parsers = new ArrayList<>();
            for (JavaParser p : ServiceLoader.load(JavaParser.class, Roaster.class.getClassLoader()))
            {
               parsers.add(p);
            }
         }
         if (parsers.isEmpty())
         {
            throw new IllegalStateException("No instances of [" + JavaParser.class.getName()
                     + "] were found on the classpath.");
         }
      }
      return parsers;
   }

   private static List getFormatters()
   {
      synchronized (Roaster.class)
      {
         if (formatters == null || formatters.isEmpty())
         {
            formatters = new ArrayList<>();
            for (FormatterProvider p : ServiceLoader.load(FormatterProvider.class, Roaster.class.getClassLoader()))
            {
               formatters.add(p);
            }
         }
         if (formatters.isEmpty())
         {
            throw new IllegalStateException("No instances of [" + FormatterProvider.class.getName()
                     + "] were found on the classpath.");
         }
      }
      return formatters;
   }

   /**
    * Create a new empty {@link JavaSource} instance.
    * 
    * @param type the type of the source
    * @param  the java type
    * @return a new {@link JavaType} instance of the given type
    * @throws IllegalStateException if no parser is available in the classPath
    * @throws ParserException if no parser is capable of parsing the provided data
    */
   public static > T create(final Class type)
   {
      for (JavaParser parser : getParsers())
      {
         final T result = parser.create(type);
         if (result != null)
         {
            return result;
         }
      }
      throw new ParserException("Cannot find JavaParserProvider capable of producing JavaSource of type "
               + type.getSimpleName(), new IllegalArgumentException(type.getName()));
   }

   /**
    * Open the given {@link File}, parsing its contents into a new {@link JavaType} instance.
    * 
    * @param file the file to read from
    * @return a the new {@link JavaType} instance
    * @throws IOException if the reading of the file fails
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static JavaType parse(final File file) throws IOException
   {
      return parse(JavaType.class, file);
   }

   /**
    * Parse the given {@link URL} data into a new {@link JavaType} instance.
    * 
    * @param url the url to read from
    * @return a the new {@link JavaType} instance
    * @throws IOException if the reading fails
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static JavaType parse(final URL url) throws IOException
   {
      return parse(JavaType.class, url);
   }

   /**
    * Read the given {@link InputStream} and parse the data into a new {@link JavaType} instance. The caller is
    * responsible to close the stream.
    * 
    * @param stream the stream to read from
    * @return a new {@link JavaType} instance
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static JavaType parse(final InputStream stream)
   {
      return parse(JavaType.class, stream);
   }

   /**
    * Parse the given character array into a new {@link JavaType} instance.
    * 
    * @param data the characters to parse
    * @return a the new {@link JavaType} instance
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static JavaType parse(final char[] data)
   {
      return parse(JavaType.class, data);
   }

   /**
    * Parse the given String data into a new {@link JavaType} instance.
    * 
    * @param data the data to parse
    * @return the new {@link JavaType} instance
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static JavaType parse(final String data)
   {
      return parse(JavaType.class, data);
   }

   /**
    * Read the given {@link URL} and parse its data into a new {@link JavaType} instance of the given type.
    * 
    * @param type the type of the source
    * @param url the url to read from
    * @param  the java type
    * @return a new {@link JavaType} instance of the given type
    * @throws IOException if a exception occurs while reading
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static > T parse(final Class type, final URL url) throws IOException
   {
      try (InputStream stream = url.openStream())
      {
         return parse(type, stream);
      }
   }

   /**
    * Read the given {@link File} and parse its data into a new {@link JavaType} instance of the given type.
    * 
    * @param type the type of the source
    * @param  the java type
    * @param file the file to read from
    * @return a new {@link JavaType} instance of the given type
    * @throws IOException if a exception occurs while reading
    * @throws FileNotFoundException if the file doesn't exists
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static > T parse(final Class type, final File file)
            throws FileNotFoundException, IOException
   {
      try (FileInputStream stream = new FileInputStream(file))
      {
         return parse(type, stream);
      }
   }

   /**
    * Read the given character array and parse its data into a new {@link JavaType} instance of the given type.
    * 
    * @param type the type of the source
    * @param data the characters to parse
    * @param  the java type
    * @return a new {@link JavaType} instance of the given type
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static > T parse(final Class type, final char[] data)
   {
      return parse(type, new String(data));
   }

   /**
    * Validates a code snippet and returns a {@link List} of {@link Problem}. Never returns {@code null}.
    * 
    * @param snippet any Java code
    * @return a list of problems (maybe empty)
    * @throws ParserException if no {@link JavaParser} implementation could be found
    */
   public static List validateSnippet(String snippet) throws ParserException
   {
      for (JavaParser parser : getParsers())
      {
         return parser.validateSnippet(snippet);
      }
      throw new ParserException("Cannot find JavaParser capable of validating the requested data");
   }

   /**
    * Read the given string and parse its data into a new {@link JavaType} instance of the given type.
    * 
    * @param type the type of the source
    * @param data the data to parse
    * @param  the java type
    * @return a new {@link JavaType} instance of the given type
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   @SuppressWarnings("unchecked")
   public static > T parse(final Class type, final String data)
   {
      for (JavaParser parser : getParsers())
      {
         final JavaUnit unit = parser.parseUnit(data);

         if (type.isInstance(unit.getGoverningType()))
         {
            return (T) unit.getGoverningType();
         }
         throw new ParserException("Source does not represent a [" + type.getSimpleName() + "], instead was ["
                  + unit.getGoverningType().getClass().getSimpleName() + "] - Cannot convert.");
      }
      throw new ParserException("Cannot find JavaParser capable of parsing the requested data"); 
   }

   /**
    * Read the given {@link InputStream} and parse its data into a new {@link JavaType} instance of the given type. The
    * caller is responsible for closing the stream.
    * 
    * @param stream the stream to read from
    * @param  the java type
    * @param type the type of the source
    * @return a new {@link JavaType} instance of the given type
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static > T parse(final Class type, final InputStream stream)
   {
      return parse(type, Streams.toString(stream));
   }

   /**
    * Read the given {@link String} and parse its data into a new {@link JavaUnit} instance of the given type.
    * 
    * @param data the data to parse
    * @return a new {@link JavaUnit} instance of the given type
    * @throws ParserException if no parser is capable of parsing the requested data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static JavaUnit parseUnit(final String data)
   {
      for (JavaParser parser : getParsers())
      {
         final JavaUnit unit = parser.parseUnit(data);
         if (unit != null)
            return unit;
      }
      throw new ParserException("Cannot find JavaParser capable of parsing the requested data"); 
   }

   /**
    * Read the given {@link InputStream} and parse its data into a new {@link JavaUnit} instance of the given type. The
    * caller is responsible for closing the stream.
    * 
    * @param data the stream to read from
    * @return a new {@link JavaUnit} instance of the given type
    * @throws ParserException if no parser is capable of parsing the provided data
    * @throws IllegalStateException if no parser is available in the classPath
    */
   public static JavaUnit parseUnit(final InputStream data)
   {
      return parseUnit(Streams.toString(data));
   }

   /**
    * Format the given {@link String} as a Java source file, using the built in code format style.
    * 
    * @param source a java source code
    * @return the formatted source code
    * @throws IllegalStateException if no formatter is available in the classPath
    */
   public static String format(String source)
   {
      String result = source;
      for (FormatterProvider formatter : getFormatters())
      {
         result = formatter.format(result);
      }
      return result;
   }

   /**
    * Format the given {@link String} as a Java source type, using the given code format {@link Properties}
    * 
    * @param properties the properties to use to format
    * @param source a java source code
    * @return the formatted source code
    * @throws IllegalStateException if no formatter is available in the classPath
    */
   public static String format(Properties properties, String source)
   {
      String result = source;
      for (FormatterProvider formatter : getFormatters())
      {
         result = formatter.format(properties, result);
      }
      return result;
   }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy