Package org.apache.myfaces.tobago.internal.context

Source Code of org.apache.myfaces.tobago.internal.context.ResourceLocator

/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements.  See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership.  The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License.  You may obtain a copy of the License at
*
*   http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied.  See the License for the
* specific language governing permissions and limitations
* under the License.
*/

package org.apache.myfaces.tobago.internal.context;

import org.apache.myfaces.tobago.context.ThemeImpl;
import org.apache.myfaces.tobago.internal.config.ThemeBuilder;
import org.apache.myfaces.tobago.internal.config.TobagoConfigFragment;
import org.apache.myfaces.tobago.internal.config.TobagoConfigParser;
import org.apache.myfaces.tobago.internal.util.IoUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.LineNumberReader;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.Enumeration;
import java.util.Properties;
import java.util.Set;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;

/**
* <p>
* This class helps to locate all resources of the ResourceManager.
* It will be called in the initialization phase.
* </p>
* <p>
* Basically it looks at the following places:
* <ul>
* <li>Directly in the root of the webapp.</li>
* <li>The root of all JARs which containing a <code>/META-INF/tobago-config.xml</code></li>
* <li>The directory <code>/META-INF/resources</code> of all JARs, if they contains such directory.</li>
* </ul>
* </p>
*
* @since 1.0.7
*/
class ResourceLocator {

  private static final Logger LOG = LoggerFactory.getLogger(ResourceLocator.class);

  private static final String META_INF_TOBAGO_CONFIG_XML = "META-INF/tobago-config.xml";
  private static final String META_INF_RESOURCE_INDEX = "META-INF/tobago-resource-index.txt";
  private static final String META_INF_RESOURCES = "META-INF/resources";

  private ServletContext servletContext;
  private ResourceManagerImpl resourceManager;
  private ThemeBuilder themeBuilder;

  public ResourceLocator(
      final ServletContext servletContext, final ResourceManagerImpl resourceManager, final ThemeBuilder themeBuilder) {
    this.servletContext = servletContext;
    this.resourceManager = resourceManager;
    this.themeBuilder = themeBuilder;
  }

  public void locate()
      throws ServletException {
    // TBD should the resource dir used from tobago-config.xml?
    locateResourcesInWar(servletContext, resourceManager, "/");
    locateResourcesFromClasspath(resourceManager);
    locateResourcesServlet30Alike(resourceManager);
  }

  private void locateResourcesInWar(
      final ServletContext servletContext, final ResourceManagerImpl resources, String path)
      throws ServletException {

    if (path.startsWith("/WEB-INF/")) {
      return; // ignore
    }
    // fix for jetty6
    if (path.endsWith("/") && path.length() > 1) {
      path = path.substring(0, path.length() - 1);
    }
    final Set<String> resourcePaths = servletContext.getResourcePaths(path);
    if (resourcePaths == null || resourcePaths.isEmpty()) {
      if (LOG.isDebugEnabled()) {
        LOG.debug("Skipping empty resource path: path='{}'", path);
      }
      return;
    }
    for (final String childPath : resourcePaths) {
      if (childPath.endsWith("/")) {
        // ignore, because weblogic puts the path directory itself in the Set
        if (!childPath.equals(path)) {
          if (LOG.isDebugEnabled()) {
            LOG.debug("childPath dir {}", childPath);
          }
          locateResourcesInWar(servletContext, resources, childPath);
        }
      } else {
        //Log.debug("add resc " + childPath);
        if (childPath.endsWith(".properties")) {
          final InputStream inputStream = servletContext.getResourceAsStream(childPath);
          try {
            addProperties(inputStream, resources, childPath, false, 0);
          } finally {
            IoUtils.closeQuietly(inputStream);
          }
        } else if (childPath.endsWith(".properties.xml")) {
          final InputStream inputStream = servletContext.getResourceAsStream(childPath);
          try {
            addProperties(inputStream, resources, childPath, true, 0);
          } catch (final RuntimeException e) {
            LOG.error("childPath = \"" + childPath + "\" ", e);
            throw e;
          } finally {
            IoUtils.closeQuietly(inputStream);
          }
        } else {
          resources.add(childPath);
        }
      }
    }
  }

  private void locateResourcesFromClasspath(final ResourceManagerImpl resources)
      throws ServletException {

    try {
      if (LOG.isInfoEnabled()) {
        LOG.info("Searching for and '" + META_INF_TOBAGO_CONFIG_XML + "'");
      }
      final ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
      final Enumeration<URL> urls = classLoader.getResources(META_INF_TOBAGO_CONFIG_XML);

      while (urls.hasMoreElements()) {
        final URL tobagoConfigUrl = urls.nextElement();
        final TobagoConfigFragment tobagoConfig = new TobagoConfigParser().parse(tobagoConfigUrl);
        for (final ThemeImpl theme : tobagoConfig.getThemeDefinitions()) {
          detectThemeVersion(tobagoConfigUrl, theme);
          themeBuilder.addTheme(theme);
          final String prefix = ensureSlash(theme.getResourcePath());
          final String protocol = tobagoConfigUrl.getProtocol();
          // tomcat uses jar // weblogic uses zip // IBM WebSphere uses wsjar
          if (!"jar".equals(protocol) && !"zip".equals(protocol) && !"wsjar".equals(protocol)) {
            LOG.warn("Unknown protocol '" + tobagoConfigUrl + "'");
          }
          addResources(resources, tobagoConfigUrl, prefix, 0);
        }
      }
    } catch (final Exception e) {
      if (e instanceof ServletException) {
        throw (ServletException) e;
      } else {
        throw new ServletException(e);
      }
    }
  }

  /**
   * Searches the /WEB-INF/lib directory for *.jar files which contains /META-INF/resources directory
   * to hold resources and add them to the ResourceManager.
   *
   * @param resources Resource Manager which collects all the resources.
   * @throws ServletException An error while accessing the resource.
   */
  private void locateResourcesServlet30Alike(final ResourceManagerImpl resources) throws ServletException {

    try {
      if (LOG.isInfoEnabled()) {
        LOG.info("Searching for '" + META_INF_RESOURCES + "'");
      }
      final ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
      final Enumeration<URL> urls = classLoader.getResources(META_INF_RESOURCES);

      while (urls.hasMoreElements()) {
        final URL resourcesUrl = urls.nextElement();

        LOG.info("resourcesUrl='" + resourcesUrl + "'");
        if (!resourcesUrl.toString().matches(".*/WEB-INF/lib/.*\\.jar\\!.*")) {
          LOG.info("skip ...");
          continue;
          // only resources from jar files in the /WEB-INF/lib should be considered (like in Servlet 3.0 spec.)
        }
        LOG.info("going on ...");

        final String protocol = resourcesUrl.getProtocol();
        // tomcat uses jar
        // weblogic uses zip
        // IBM WebSphere uses wsjar
        if (!"jar".equals(protocol) && !"zip".equals(protocol) && !"wsjar".equals(protocol)) {
          LOG.warn("Unknown protocol '" + resourcesUrl + "'");
        }
        addResourcesFromZip(
            resources,
            resourcesUrl.getFile(),
            resourcesUrl.getProtocol(),
            "/" + META_INF_RESOURCES,
            META_INF_RESOURCES.length() + 1);
      }
    } catch (final IOException e) {
      final String msg = "while loading ";
      LOG.error(msg, e);
      throw new ServletException(msg, e);
    }
  }

  private void addResources(
      final ResourceManagerImpl resources, final URL themeUrl, final String prefix, final int skipPrefix)
      throws IOException, ServletException {

    final String fileName = themeUrl.getFile();
    LOG.info("fileName='" + fileName + "'");
    final String resourceIndex
        = fileName.substring(0, fileName.lastIndexOf(META_INF_TOBAGO_CONFIG_XML)) + META_INF_RESOURCE_INDEX;
    LOG.info("resourceIndex='" + resourceIndex + "'");




    final URL resource = findMatchingResourceIndexUrl(themeUrl);


    LOG.info("resource='" + resource + "'");
    if (resource != null) {
      addResourcesFromIndexFile(resources, resource.openStream(), skipPrefix);
    } else {
      addResourcesFromZip(resources, fileName, themeUrl.getProtocol(), prefix, skipPrefix);
    }
  }

  /**
   * Find a matching tobago-resource-index.txt to the given tobago-config.xml
   * We look here all possible URL from ClassLoader, because an AppServer may protect direct access...
   */
  private URL findMatchingResourceIndexUrl(final URL themeUrl) throws IOException {
    final String themeProtocol = themeUrl.getProtocol();
    final String themeDir
        = themeUrl.getFile().substring(0, themeUrl.getFile().length() - META_INF_TOBAGO_CONFIG_XML.length());
    final ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
    final Enumeration<URL> urls = classLoader.getResources(META_INF_RESOURCE_INDEX);
    URL url = null;
    while (urls.hasMoreElements()) {
      url = urls.nextElement();
      if (url.getProtocol().equals(themeProtocol)
          && url.getFile().startsWith(themeDir)) {
        break;
      }
      url = null;
    }
    return url;
  }

  private void addResourcesFromIndexFile(
      final ResourceManagerImpl resources, final InputStream indexStream, final int skipPrefix)
      throws IOException, ServletException {
    final LineNumberReader reader = new LineNumberReader(new InputStreamReader(indexStream));
    String name;
    while (null != (name = reader.readLine())) {
      addResource(resources, name, skipPrefix);
    }
  }

  private void addResourcesFromZip(
      final ResourceManagerImpl resources, String fileName, final String protocol,
      final String prefix, final int skipPrefix)
      throws ServletException, IOException {

    final int exclamationPoint = fileName.indexOf("!");
    if (exclamationPoint != -1) {
      fileName = fileName.substring(0, exclamationPoint);
    }
    if (LOG.isInfoEnabled()) {
      LOG.info("Adding resources from fileName='" + fileName + "' prefix='" + prefix + "' skip=" + skipPrefix + "");
    }

    // JBoss 6 introduced vfs protocol
    if (protocol.equals("vfs")) {
      LOG.warn("Protocol '" + protocol + "' is not supported. If resource is needed by the application, you'll"
          + "need to put a index file tobago-resource-index.txt in the JAR. File='" + fileName + "'");
      return;
    }

    URL jarFile;
    try {
      // JBoss 5.0.0 introduced vfszip protocol
      if (protocol.equals("vfszip")) {
        fileName = new File(fileName).getParentFile().getParentFile().getPath();
        if (File.separatorChar == '\\' && fileName.contains("\\")) {
          fileName = fileName.replace('\\', '/');
          if (LOG.isInfoEnabled()) {
            LOG.info("Fixed slashes for virtual filesystem protocol on windows system: " + fileName);
          }
        }
      }
      jarFile = new URL(fileName);
    } catch (final MalformedURLException e) {
      // workaround for weblogic on windows
      jarFile = new URL("file:" + fileName);
    }
    InputStream stream = null;
    ZipInputStream zipStream = null;
    try {
      stream = jarFile.openStream();
      zipStream = new ZipInputStream(stream);
      for (ZipEntry nextEntry = zipStream.getNextEntry(); nextEntry != null; nextEntry = zipStream.getNextEntry()) {
        if (nextEntry.isDirectory()) {
          continue;
        }
        final String name = "/" + nextEntry.getName();
        if (name.startsWith(prefix)) {
          addResource(resources, name, skipPrefix);
        }
      }
    } finally {
      IoUtils.closeQuietly(stream);
      IoUtils.closeQuietly(zipStream);
    }
  }

  private void addResource(final ResourceManagerImpl resources, final String name, final int skipPrefix)
      throws ServletException {

    if (name.endsWith(".class")) {
      // ignore the class files
    } else if (name.endsWith(".properties")) {
      if (LOG.isInfoEnabled()) {
        LOG.info("Adding properties from: '" + name.substring(1) + "'");
      }
      final InputStream inputStream
          = Thread.currentThread().getContextClassLoader().getResourceAsStream(name.substring(1));
      try {
        addProperties(inputStream, resources, name, false, skipPrefix);
      } finally {
        IoUtils.closeQuietly(inputStream);
      }
    } else if (name.endsWith(".properties.xml")) {
      if (LOG.isInfoEnabled()) {
        LOG.info("Adding properties from: '" + name.substring(1) + "'");
      }
      final InputStream inputStream
          = Thread.currentThread().getContextClassLoader().getResourceAsStream(name.substring(1));
      try {
        addProperties(inputStream, resources, name, true, skipPrefix);
      } finally {
        IoUtils.closeQuietly(inputStream);
      }
    } else {
      resources.add(name.substring(skipPrefix));
    }
  }

  private String ensureSlash(String resourcePath) {
    if (!resourcePath.startsWith("/")) {
      resourcePath = '/' + resourcePath;
    }
    if (!resourcePath.endsWith("/")) {
      resourcePath = resourcePath + '/';
    }
    return resourcePath;
  }

  private void addProperties(
      final InputStream stream, final ResourceManagerImpl resources, final String childPath, final boolean xml,
      final int skipPrefix)
      throws ServletException {

    final String directory = childPath.substring(skipPrefix, childPath.lastIndexOf('/'));
    final String filename = childPath.substring(childPath.lastIndexOf('/') + 1);

    int end = filename.lastIndexOf('.');
    if (xml) {
      end = filename.lastIndexOf('.', end - 1);
    }
    final String locale = filename.substring(0, end);

    final Properties temp = new Properties();
    try {
      if (xml) {
        temp.loadFromXML(stream);
        if (LOG.isDebugEnabled()) {
          LOG.debug(childPath);
          LOG.debug("xml properties: {}", temp.size());
        }
      } else {
        temp.load(stream);
        if (LOG.isDebugEnabled()) {
          LOG.debug(childPath);
          LOG.debug("    properties: {}", temp.size());
        }
      }
    } catch (final IOException e) {
      final String msg = "while loading " + childPath;
      LOG.error(msg, e);
      throw new ServletException(msg, e);
    } finally {
      IoUtils.closeQuietly(stream);
    }

    final Enumeration e = temp.propertyNames();
    while (e.hasMoreElements()) {
      final String key = (String) e.nextElement();
      resources.add(directory + '/' + locale + '/' + key, temp.getProperty(key));
      if (LOG.isDebugEnabled()) {
        LOG.debug(directory + '/' + locale + '/' + key + "=" + temp.getProperty(key));
      }
    }
  }

  private void detectThemeVersion(final URL tobagoConfigUrl, final ThemeImpl theme) throws IOException {
    if (theme.isVersioned()) {
      final String themeUrlStr = tobagoConfigUrl.toString();
      final int index = themeUrlStr.indexOf(META_INF_TOBAGO_CONFIG_XML);
      final String metaInf = themeUrlStr.substring(0, index) + "META-INF/MANIFEST.MF";
      final Properties properties = new Properties();
      final URL url = new URL(metaInf);
      InputStream inputStream = null;
      String version = null;
      try {
        inputStream = url.openStream();
        properties.load(inputStream);
        version = properties.getProperty("Implementation-Version");
      } catch (final FileNotFoundException e) {
        // may happen (e. g. in tests)
        LOG.error("No Manifest-File found.");
      } finally {
        IoUtils.closeQuietly(inputStream);
      }
      if (version != null) {
        theme.setVersion(version);
      } else {
        theme.setVersioned(false);
        LOG.error("No Implementation-Version found in Manifest-File for theme: '" + theme.getName()
            + "'. Resetting the theme to unversioned. Please correct the Manifest-File.");
      }
    }
  }

}
TOP

Related Classes of org.apache.myfaces.tobago.internal.context.ResourceLocator

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.