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

ro.isdc.wro.util.WroUtil Maven / Gradle / Ivy

There is a newer version: 2.1.1
Show newest version
/*
 * Copyright (c) 2008. All rights reserved.
 */
package ro.isdc.wro.util;

import static org.apache.commons.lang3.Validate.notNull;

import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.Reader;
import java.io.Writer;
import java.util.Enumeration;
import java.util.Locale;
import java.util.Properties;
import java.util.TimeZone;
import java.util.UUID;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import javax.servlet.http.HttpServletRequest;

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.FilenameUtils;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.Validate;
import org.apache.commons.lang3.time.FastDateFormat;

import ro.isdc.wro.WroRuntimeException;
import ro.isdc.wro.model.WroModel;
import ro.isdc.wro.model.factory.WroModelFactory;
import ro.isdc.wro.model.resource.Resource;
import ro.isdc.wro.model.resource.locator.ServletContextUriLocator;
import ro.isdc.wro.model.resource.processor.ResourcePostProcessor;
import ro.isdc.wro.model.resource.processor.ResourcePreProcessor;


/**
 * Utility class.
 *
 * @author Alex Objelean
 * @created Created on Nov 13, 2008
 */
public final class WroUtil {
  private static final String SEPARATOR_WINDOWS = "\\";
  /**
   * Empty line pattern.
   */
  public static final Pattern EMTPY_LINE_PATTERN = Pattern.compile(loadRegexpWithKey("emptyLine"), Pattern.MULTILINE);
  /**
   * Thread safe date format used to transform milliseconds into date as string to put in response header. The localy is
   * set explicitly to US to conform to specification.
   */
  private static final FastDateFormat DATE_FORMAT = FastDateFormat.getInstance("E, dd MMM yyyy HH:mm:ss z",
      TimeZone.getTimeZone("GMT"), Locale.US);
  /**
   * Patterns used to search for mangled Accept-Encoding header.
   */
  private static final Pattern PATTERN_ACCEPT_ENCODING = Pattern.compile(loadRegexpWithKey("requestHeader.acceptEncoding"));
  private static final Pattern PATTERN_GZIP = Pattern.compile(loadRegexpWithKey("requestHeader.gzip"));

  private static final AtomicInteger threadFactoryNumber = new AtomicInteger(1);
  public static final InputStream EMPTY_STREAM = new ByteArrayInputStream("".getBytes());

  /**
   * @return {@link ThreadFactory} with daemon threads.
   */
  public static ThreadFactory createDaemonThreadFactory(final String name) {
    return new ThreadFactory() {
      private final String prefix = "wro4j-" + name + "-" + threadFactoryNumber.getAndIncrement() + "-thread-";
      private final AtomicInteger threadNumber = new AtomicInteger(1);

      public Thread newThread(final Runnable runnable) {
        final Thread thread = new Thread(runnable, prefix + threadNumber.getAndIncrement());
        thread.setDaemon(true);
        return thread;
      }
    };
  }

  /**
   * Transforms milliseconds into date format for response header of this form: Sat, 10 Apr 2010 17:31:31 GMT.
   *
   * @param milliseconds
   *          to transform
   * @return string representation of the date.
   */
  public static String toDateAsString(final long milliseconds) {
    return DATE_FORMAT.format(milliseconds);
  }

  /**
   * Retrieve pathInfo from a given location.
   *
   * @param request
   * @param location
   *          where to search contextPath.
   * @return pathInfo value.
   */
  public static String getPathInfoFromLocation(final HttpServletRequest request, final String location) {
    if (StringUtils.isEmpty(location)) {
      throw new IllegalArgumentException("Location cannot be empty string!");
    }
    final String contextPath = request.getContextPath();
    if (contextPath != null) {
      if (startsWithIgnoreCase(location, contextPath)) {
        return location.substring(contextPath.length());
      } else {
        return location;
      }
    }
    final String noSlash = location.substring(1);
    final int nextSlash = noSlash.indexOf('/');
    if (nextSlash == -1) {
      return "";
    }
    final String pathInfo = noSlash.substring(nextSlash);
    return pathInfo;
  }

  /**
   * 

* Case insensitive check if a String starts with a specified prefix. *

*

* nulls are handled without exceptions. Two null references are considered to be equal. The * comparison is case insensitive. *

* *
   * StringUtils.startsWithIgnoreCase(null, null)      = true
   * StringUtils.startsWithIgnoreCase(null, "abcdef")  = false
   * StringUtils.startsWithIgnoreCase("abc", null)     = false
   * StringUtils.startsWithIgnoreCase("abc", "abcdef") = true
   * StringUtils.startsWithIgnoreCase("abc", "ABCDEF") = true
   * 
* * @see java.lang.String#startsWith(String) * @param str * the String to check, may be null * @param prefix * the prefix to find, may be null * @return true if the String starts with the prefix, case insensitive, or both null * @since 2.4 */ public static boolean startsWithIgnoreCase(final String str, final String prefix) { return startsWith(str, prefix, true); } /** * Creates a folder like implementation for a class. Ex: com.mycompany.MyClass -> com/mycompany/ * * @param clazz * used as a base location for determining the package path. * @return a string representation of the path where the class resides. */ public static String toPackageAsFolder(final Class clazz) { Validate.notNull(clazz, "Class cannot be null!"); return clazz.getPackage().getName().replace('.', '/'); } /** *

* Check if a String starts with a specified prefix (optionally case insensitive). *

* * @see java.lang.String#startsWith(String) * @param str * the String to check, may be null * @param prefix * the prefix to find, may be null * @param ignoreCase * inidicates whether the compare should ignore case (case insensitive) or not. * @return true if the String starts with the prefix or both null */ private static boolean startsWith(final String str, final String prefix, final boolean ignoreCase) { if (str == null || prefix == null) { return (str == null && prefix == null); } if (prefix.length() > str.length()) { return false; } return str.regionMatches(ignoreCase, 0, prefix, 0, prefix.length()); } /** * Retrieve servletPath from a given location. * * @param location * where to search the servletPath. * @return ServletPath string value. */ public static String getServletPathFromLocation(final HttpServletRequest request, final String location) { return location.replace(getPathInfoFromLocation(request, location), ""); } /** * Analyze headers of the request and searches for mangled (by proxy) for "Accept-Encoding" header and its mangled * variations and gzip header value and its mangled variations. * * @return true if this request support gzip encoding. */ @SuppressWarnings("unchecked") public static boolean isGzipSupported(final HttpServletRequest request) { if (request != null) { final Enumeration headerNames = request.getHeaderNames(); if (headerNames != null) { while (headerNames.hasMoreElements()) { final String headerName = headerNames.nextElement(); final Matcher m = PATTERN_ACCEPT_ENCODING.matcher(headerName); if (m.find()) { final String headerValue = request.getHeader(headerName); final Matcher mValue = PATTERN_GZIP.matcher(headerValue); return mValue.find(); } } } } return false; } /** * Transforms a java multi-line string into javascript multi-line string. This technique was found at {@link http * ://stackoverflow.com/questions/805107/multiline-strings-in-javascript/} * * @param data * a string containing new lines. * @return a string which being evaluated on the client-side will be treated as a correct multi-line string. */ public static String toJSMultiLineString(final String data) { final StringBuffer result = new StringBuffer("["); if (data != null) { final String[] lines = data.split("\n"); if (lines.length == 0) { result.append("\"\""); } for (int i = 0; i < lines.length; i++) { final String line = lines[i]; result.append("\""); result.append(line.replace(SEPARATOR_WINDOWS, "\\\\").replace("\"", "\\\"").replaceAll("\\r|\\n", "")); // this is used to force a single line to have at least one new line (otherwise cssLint fails). if (lines.length == 1) { result.append("\\n"); } result.append("\""); if (i < lines.length - 1) { result.append(","); } } } result.append("].join(\"\\n\")"); return result.toString(); } /** * Utility used to verify that requestURI matches provided path */ public static boolean matchesUrl(final HttpServletRequest request, final String path) { final Pattern pattern = Pattern.compile(".*" + path + "[/]?", Pattern.CASE_INSENSITIVE); if (request.getRequestURI() != null) { final Matcher m = pattern.matcher(request.getRequestURI()); return m.matches(); } return false; } /** * A factory method for creating a {@link ResourceProcessor} based on provided {@link ResourcePreProcessor}. * * @param preProcessor * {@link ResourcePreProcessor} to use as a {@link ResourceProcessor}. * @return instance of {@link ResourceProcessor}. */ public static ResourcePostProcessor newResourceProcessor(final Resource resource, final ResourcePreProcessor preProcessor) { return new ResourcePostProcessor() { public void process(final Reader reader, final Writer writer) throws IOException { preProcessor.process(resource, reader, writer); } }; } /** * A simple way to create a {@link WroModelFactory}. * * @param model * {@link WroModel} instance to be returned by the factory. */ public static WroModelFactory factoryFor(final WroModel model) { return new WroModelFactory() { public WroModel create() { return model; } public void destroy() { } }; } public static ObjectFactory simpleObjectFactory(final T object) { return new ObjectFactory() { public T create() { return object; } }; } /** * Load the regular expression stored in in regexp.properties resource file. * * @param key * the key of the regexp to load. * @return regular expression value. */ public static String loadRegexpWithKey(final String key) { try { final InputStream stream = WroUtil.class.getResourceAsStream("regexp.properties"); final Properties props = new RegexpProperties().load(stream); return props.getProperty(key); } catch (final IOException e) { throw new WroRuntimeException("Could not load pattern with key: " + key + " from property file", e); } } /** * @return the implementation version of wro4j. */ public static String getImplementationVersion() { return WroUtil.class.getPackage().getImplementationVersion(); } /** * Copy and close the reader and writer streams. * * @param reader * The source stream. * @param writer * The destination stream. * @throws IOException * If content cannot be copy. */ public static void safeCopy(final Reader reader, final Writer writer) throws IOException { try { IOUtils.copy(reader, writer); } finally { IOUtils.closeQuietly(reader); IOUtils.closeQuietly(writer); } } /** * @return a generated {@link File} with unique name located in temp folder. */ public static File createTempFile() { return createTempFile("temp"); } /** * @return a folder with unique name.. */ public static File createTempDirectory() { final String fileName = String.format("wro4j-%s", UUID.randomUUID().toString()); final File file = new File(FileUtils.getTempDirectory(), fileName); file.mkdir(); return file; } /** * Creates a temp file which has a certain extension. * * @param extension * of the created temp file. */ public static File createTempFile(final String extension) { try { final String fileName = String.format("wro4j-%s.%s", UUID.randomUUID().toString(), extension); final File file = new File(createTempDirectory(), fileName); file.createNewFile(); return file; } catch (final IOException e) { throw WroRuntimeException.wrap(e); } } /** * Cleans the image url by trimming result and removing \' or \" characters if such exists. * * @param imageUrl * to clean. * @return cleaned image URL. */ public static final String cleanImageUrl(final String imageUrl) { notNull(imageUrl); return imageUrl.replace('\'', ' ').replace('\"', ' ').trim(); } /** * Removes the query string from the provided path (everything followed by '?' including the question mark). */ public static final String removeQueryString(final String path) { return path.replaceFirst("\\?.*", ""); } /** * @return current working directory */ public static final File getWorkingDirectory() { return new File(System.getProperty("user.dir")); } /** * Similar to {@link FilenameUtils#getFullPath(String)}, but fixes the problem with Windows platform for situations * when the path starts with "/" (servlet context relative resources) which are resolved to "\" on windows. * * @param path * to compute filePath from. * @return full path from the provided path. */ public static final String getFullPath(final String path) { final String fullPath = FilenameUtils.getFullPath(path); return replaceWithServletContextSeparatorIfNedded(fullPath); } /** * Similar to {@link FilenameUtils#normalize(String)}, but fixes the problem with Windows platform for situations * when the path starts with "/" (servlet context relative resources) which are resolved to "\" on windows. * * @param path * to compute filePath from. * @return full path from the provided path. */ public static final String normalize(final String path) { final String normalized = FilenameUtils.normalize(path); return replaceWithServletContextSeparatorIfNedded(normalized); } private static String replaceWithServletContextSeparatorIfNedded(String normalized) { if (normalized.startsWith(SEPARATOR_WINDOWS) || normalized.contains(SEPARATOR_WINDOWS)) { normalized = normalized.replace(SEPARATOR_WINDOWS, ServletContextUriLocator.PREFIX); } return normalized; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy