org.apache.myfaces.trinidadinternal.skin.StyleSheetEntry Maven / Gradle / Ivy
/*
* 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.trinidadinternal.skin;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.List;
import java.util.Map;
import javax.faces.context.FacesContext;
import org.apache.myfaces.trinidad.context.RenderingContext;
import org.apache.myfaces.trinidad.logging.TrinidadLogger;
import org.apache.myfaces.trinidad.share.io.InputStreamProvider;
import org.apache.myfaces.trinidad.share.io.NameResolver;
import org.apache.myfaces.trinidad.util.ClassLoaderUtils;
import org.apache.myfaces.trinidadinternal.share.io.CachingNameResolver;
import org.apache.myfaces.trinidadinternal.share.io.FileInputStreamProvider;
import org.apache.myfaces.trinidadinternal.share.io.URLInputStreamProvider;
import org.apache.myfaces.trinidadinternal.share.xml.JaxpXMLProvider;
import org.apache.myfaces.trinidadinternal.share.xml.ParseContextImpl;
import org.apache.myfaces.trinidadinternal.share.xml.XMLProvider;
import org.apache.myfaces.trinidadinternal.style.StyleContext;
import org.apache.myfaces.trinidadinternal.style.xml.StyleSheetDocumentUtils;
import org.apache.myfaces.trinidadinternal.style.xml.parse.StyleSheetDocument;
/**
* Package-private utility class used by Skin implementation
* to manage a single XSS or CSS skin stylesheet source file .
* This class calls the parsing code which parses either the XSS or CSS file (_createSkinStyleSheet),
* and it stores a StyleSheetDocument object, which is a parsed representation of a
* Trinidad style sheet document whether that is in the css or xss format or merged.
* This class could actually
* be pushed into an inner class in Skin, but at the moment
* it is separated out simply to reduce the amount of code in
* Skin.java.
*
* @version $Name: $ ($Revision: adfrt/faces/adf-faces-impl/src/main/java/oracle/adfinternal/view/faces/skin/StyleSheetEntry.java#0 $) $Date: 10-nov-2005.18:59:01 $
*/
class StyleSheetEntry
{
/**
* Creates a StyleSheetEntry for the specified context and styleSheetName.
* This method will log any errors/exceptions and return
* null if the style sheet source file could not be found/parsed.
*/
public static StyleSheetEntry createEntry(
StyleContext context,
String styleSheetName
)
{
// In order to create the StyleSheetEntry, we need to locate and
// parse the style sheet file. We use a NameResolver to use to
// find the style sheet.
NameResolver resolver = _getNameResolver(context, styleSheetName);
if (resolver == null)
return null;
// a private static inner class to store the document, icon, and skin properties
// =-=jmw @todo Should I just create a StyleSheetEntry directly,
// and make the constructor public? (probably)
StyleSheetEntry skinStyleSheet = _createSkinStyleSheet(resolver,
styleSheetName);
if (skinStyleSheet == null)
return null;
// We either a special subclass of StyleSheetEntry that will recalculate
// the StyleSheetEntry if the skin is dirty or if the web.xml's
// CHECK_FILE_MODIFICATION flag is set and there are file modifications.
boolean checkStylesModified = context.checkStylesModified();
return new CheckModifiedEntry(styleSheetName,
skinStyleSheet.getDocument(),
resolver,
checkStylesModified);
}
// Creates a StyleSheetEntry which never checks for
// modifications.
// =-=jmw there is a hotspot bug I filed with JDeveloper
// 4102252 when this is private I get an IllegalAccessException
// in CheckModifiedEntry. changing it to package private
StyleSheetEntry(
String styleSheetName,
StyleSheetDocument document
)
{
_name = styleSheetName;
_document = document;
}
StyleSheetEntry(String styleSheetName)
{
this(styleSheetName, null);
}
// Use full constructor
private StyleSheetEntry()
{
}
/**
* Returns the name of the style sheet source file
* for this StyleSheetEntry.
*/
public String getStyleSheetName()
{
return _name;
}
/**
* Returns the StyleSheetDocument for this
* StyleSheetEntry.
*/
public StyleSheetDocument getDocument()
{
return _document;
}
/**
* Checks whether the underlying style sheet source file
* has been modified and if so, reloads the StyleSheetDocument.
* Returns true if the document has been modified (and the
* StyleSheetDocument has been reloaded).
*/
public boolean checkModified(StyleContext context)
{
return false;
}
// Gets a File for the specified name, or returns null if no file exists
// Try the local styles directory.
public static File resolveLocalFile(File localStylesDir, String name)
{
// Try the local styles directory
File file = new File(localStylesDir, name);
if (file.exists())
return file;
return null;
}
// Gets an URL for the specified name using ClassLoaderUtils.getResource
public static URL resolveClassLoaderURL(String name)
{
if (name == null)
return null;
return ClassLoaderUtils.getResource(name);
}
// Gets an URL for the non static urls -- that is, urls that could change after the
// server has started.
public static URL resolveNonStaticURL(String name)
{
if (name == null)
return null;
FacesContext fContext = FacesContext.getCurrentInstance();
if (fContext != null)
{
try
{
if (name.startsWith("http:") ||
name.startsWith("https:") ||
name.startsWith("file:") ||
name.startsWith("ftp:") ||
name.startsWith("jar:"))
{
URL url = new URL(name);
if (url != null)
return url;
}
else
{
String rootName = _getRootName(name);
// Return a URL for the application resource mapped to the specified path,
// if it exists; otherwise, return null.
URL url = fContext.getExternalContext().getResource(rootName);
if (url != null)
return url;
}
}
catch (MalformedURLException e)
{
// Eat the MalformedURLException - maybe the name isn't an URL
;
}
}
return null;
}
// Called by CheckModifiedEntry when the style sheet has changed
void __setDocument(StyleSheetDocument document)
{
_document = document;
}
// Creates the SkinStyleSheet (a private static inner class that
// contains StyleSheetDocument plus a list
// of properties) from a CSS file
//
private static StyleSheetEntry _createSkinStyleSheet(
NameResolver resolver,
String styleSheetName
)
{
StyleSheetEntry skinStyleSheet = null;
if (styleSheetName.endsWith(".css"))
{
// this will parse a skin css file which allows icons, properties,
// and styles.
skinStyleSheet = _createSkinStyleSheetFromCSS(resolver,
styleSheetName);
} else {
String message = _LOG.getMessage("INVALID_STYLESHEET_TYPE", new Object[]{styleSheetName});
_LOG.severe(message);
}
return skinStyleSheet;
}
// Creates the StyleSheetEntry from a skinning file that ends in .css
private static StyleSheetEntry _createSkinStyleSheetFromCSS(
NameResolver resolver,
String styleSheetName
)
{
try
{
// This log is used by syntax error log in SkinCSSParser._handleBraceMismatch
_LOG.info("LOADING_STYLESHEET", styleSheetName);
// We simply use a ParseContext as a place to store parameters like
// inputStreamProviders and nameResolvers that will be reused when parsing
ParseContextImpl parseContext = new ParseContextImpl();
// if this is a utility that isn't in this file, then I can't return a SkinStyleSheet.
// I think instead this parseCSSSource should return a new instance of StyleSheetEntry.
return SkinStyleSheetParserUtils.parseCSSSource(
parseContext,
resolver,
styleSheetName,
StyleSheetEntry.class);
}
catch (Exception e)
{
if (RenderingContext.getCurrentInstance().isDesignTime())
{
_LOG.warning("CANNOT_LOAD_STYLESHEET", styleSheetName);
}
else
{
_LOG.severe("CANNOT_LOAD_STYLESHEET", styleSheetName);
_LOG.severe(e);
}
}
return null;
}
// Returns the NameResolver to use for locating and loading style sheet file.
// Depending upon what the styleSheetName is, we load the file different way: local file,
// url, etc.
private static NameResolver _getNameResolver(
StyleContext context,
String styleSheetName
)
{
// get localStylesDirectory
File localStylesDir = _getStylesDir(context);
// Make sure we have some styles directory
if ((localStylesDir == null))
{
_LOG.warning(_STYLES_DIR_ERROR);
return null;
}
NameResolver resolver = null;
try
{
resolver =
_getNameResolverForStyleSheetFile(context, localStylesDir, styleSheetName);
}
catch (IOException e)
{
if (_LOG.isWarning())
_LOG.warning("CANNOT_LOAD_STYLESHEET", styleSheetName);
_LOG.warning(e.getMessage());
}
if (resolver == null)
{
// If we can't get a NameResolver, something is seriously wrong.
// createResolver() logged the error already, so just return null.
return null;
}
// Wrap up the resolver in a CachingNameResolver that we can
// can use to check for updates to imported style sheets
return new CachingNameResolver(resolver, null, true);
}
/**
*
* This method tries to find the Skin's stylesheet file (e.g., purple-desktop.css).
* It creates a NameResolver object, and it returns the NameResolver object.
* A NameResolver object contains an
* InputStreamProvider (this object loads the file) and a sub- NameResolver
* that finds files that are relative to the base file, like an @import file in a .css file.
*
*
* This method tries to find the stylesheet file, first locally, or using an url, or a static url,
* then we create a StyleSheetNameResolver and we pass in the InputStreamProvider we created that
* we know can find the file. If we can't find the file any of these ways, then we check
* META-INF/services for a NameResolver service. This is how a third party can customize
* how they can find files, by supplying a META-INF/services NameResolver implementation.
*
* @param context
* @param localStylesDir File the local styles directory
* @param filename the stylesheet name
* @return NameResolver - either a StyleSheetNameResolver or the META-INF/services NameResolver
* implementation. The META-INF/services NameResolver implementation is the way a third party
* can customize the way they find and load files.
* @throws FileNotFoundException when the file could not be found in all of the ways we tried to find it.
*/
private static NameResolver _getNameResolverForStyleSheetFile(
StyleContext context,
File localStylesDir,
String filename) throws IOException
{
InputStreamProvider provider = null;
File file = StyleSheetEntry.resolveLocalFile(localStylesDir, filename);
if (file != null)
provider = new FileInputStreamProvider(file);
if (provider == null)
{
// Gets an URL for the specified name.
// Try a few different means to get the file as an url and then create the appropriate
// InputStreamProvider from that URL.
URL url = resolveNonStaticURL(filename);
if (url != null)
provider = new URLInputStreamProvider(url);
else
{
// see if it is an URL that can be loaded by the ClassLoader.
// We create a StaticURLInputStreamProvider from the url because we consider the
// url static because it can't be changed without restarting the server, so we don't
// need to check if the source has changed.
url = resolveClassLoaderURL(filename);
if (url != null)
provider = new StaticURLInputStreamProvider(url);
}
}
// If at this point we have found an InputStreamProvider, then we will create a
// StyleSheetNameResolver. Otherwise, we need to check for a custom NameResolver.
if (provider != null)
return StyleSheetNameResolver.createResolver(context, localStylesDir, provider);
// If we still can't locate the file at this point, then look for a custom
// NameResolver specified as a META-INF\services.
NameResolver servicesNameResolver = _loadNameResolverFromServices(filename);
if (servicesNameResolver != null)
{
if (_LOG.isFine())
{
_LOG.fine("Using the InputStreamProvider from META-INF\\services");
}
return servicesNameResolver;
}
// If we couldn't locate the file, throw an IOException
throw new FileNotFoundException(_getFileNotFoundMessage(localStylesDir, filename));
}
// Subclass of StyleSheetEntry which recreates the StyleSheetEntry
// if the skin is marked dirty (skin.isDirty()) or if the underlying
// source files have been modified and CHECK_FILE_MODIFICATION flag is set
// in web.xml.
private static class CheckModifiedEntry extends StyleSheetEntry
{
public CheckModifiedEntry(
String styleSheetName,
StyleSheetDocument document,
NameResolver resolver,
boolean checkFileModifiedFlagSet
)
{
super(styleSheetName, document);
// We need the InputStreamProvider in order to check
// for modifications. Get it from the NameResolver.
_provider = _getInputStreamProvider(resolver);
_checkFileModifiedFlagSet = checkFileModifiedFlagSet;
}
// Override of checkModified() which first checks if the file
// needs to be reparsed and a new CSS file generated.
// The conditions are if the skin is marked dirty, or if the
// web.xml's CHECK_FILE_MODIFICATION flag is set and the source
// has changed. The InputStreamProvider's hasSourceChanged method
// is called to see if the source has changed.
@Override
public boolean checkModified(StyleContext context)
{
// We would synchronize here, but at the moment synchronization
// is provided by Skin.getStyleSheetDocument().
if (context.isDirty() ||
(_checkFileModifiedFlagSet &&
((_provider != null) && (_provider.hasSourceChanged()))) )
{
// Throw away the old InputStreamProvider and StyleSheetDocument
_provider = null;
__setDocument(null);
// Get a new NameResolver
String name = getStyleSheetName();
NameResolver resolver = _getNameResolver(context, name);
if (resolver != null)
{
// Recreate the StyleSheetEntry for the styleSheet using the new NameResolver
// (e.g., if purpleSkin.css
// has changed, create the SkinStyleSheetEntry for purpleSkin.css)
// Using a new NameResolver like we do ensures that we don't get a
// cached result from the provider
// (see SkinStyleSheetParserUtils.parseCSSSource's getCachedResult)
StyleSheetEntry skinStyleSheet = _createSkinStyleSheet(resolver,
name);
if (skinStyleSheet != null)
{
_provider = _getInputStreamProvider(resolver);
__setDocument(skinStyleSheet.getDocument());
}
return true;
}
}
return false;
}
private InputStreamProvider _getInputStreamProvider(
NameResolver resolver
)
{
// Note: We assume that we are using a CachingNameResolver,
// and that the InputStreamProvider for the source file has
// already been retrieved. That way, when we call
// NameResolver.getProvider(), we are actually getting
// the same InputStreamProvider that was used to read the
// style sheet earlier.
assert (resolver instanceof CachingNameResolver);
try
{
return resolver.getProvider(getStyleSheetName());
}
catch (IOException e)
{
// We shouldn't get here - we know we were able to
// get the InputStreamProvider before - so we should be
// able to get the cached InputStreamProvider now
assert false;
}
return null;
}
private InputStreamProvider _provider;
private boolean _checkFileModifiedFlagSet;
}
// Construct error message for the specified file name
private static String _getFileNotFoundMessage(File localStylesDir, String name)
{
StringBuffer buffer = new StringBuffer();
buffer.append("Unable to locate the skin's style sheet \"");
buffer.append(name);
buffer.append("\" in ");
if (localStylesDir != null)
{
buffer.append("local styles directory (");
buffer.append(localStylesDir.getPath());
buffer.append("), ");
}
buffer.append("or on the class path.\n");
buffer.append("Please be sure that this style sheet is installed.");
return buffer.toString();
}
// Returns the File corresponding to the styles directory - either
// the local directory or the shared directory - depending on the
// shared value
private static File _getStylesDir(
StyleContext context)
{
String contextPath = context.getGeneratedFilesPath();
// We only need to look up the shared styles path if the shared
// context path is non-null. If the shared context path is null,
// we don't have a shared styles directory (and calling
// Configuration.getPath() may throw a DirectoryUnavailableException).
if (contextPath == null)
return null;
String stylesPath = contextPath + "/adf/styles";
// Convert the path to a File
File stylesDir = new File(stylesPath);
// Make sure the directory actually exists
if (stylesDir.exists())
return stylesDir;
return null;
}
// Returns a name which can be resolved relative to the
// ServletContext root.
private static String _getRootName(String name)
{
// Tack on a starting "/" if the name doesn't already have one -
// seems to be required by ServletContext.getRealPath() and
// ServletContext.getResource() - at least on OC4J.
return (name.startsWith("/")) ? name : ("/" + name);
}
/**
* Returns an instance of NameResolver that was set in META-INF\services.
* This is used only if the stylesheet cannot be found any other way.
* This way third party users can create their own way to find the file e.g., MDS.
*
* @return a NameResolver instance that has been defined in META-INF\services\
* org.apache.myfaces.trinidad.share.io.NameResolver
* In this file they will have a line like "org.mycompany.io.MyNameResolverImpl".
* null if no NameResolver is found.
*/
static private NameResolver _loadNameResolverFromServices(String name)
{
// first check if the cached NameResolver will serve the name requested
FacesContext context = FacesContext.getCurrentInstance();
Map appMap = context.getExternalContext().getApplicationMap();
// Is it stored on the application map already? can it serve the current request? If so, use it.
NameResolver savedResolver = (NameResolver)appMap.get(_SERVICES_RESOLVER_KEY);
if (savedResolver != null)
{
try
{
savedResolver.getProvider(name);
// NameResolver has to throw IOException if unable to serve a name. It should not return null.
// therefore, if the line above did not throw an exception that means we can proceed with this resolver
return savedResolver;
}
catch (IOException ex)
{
// proceed with checking other registered NameResolvers, if the cached one does not serve the name requested
if (_LOG.isFine())
_LOG.fine(_SERVICES_RESOLVER_IOEXCEPTION_MSG);
}
}
// Now, the cached resolver did not serve the name requested
// there fore, check other registered NameResolvers
List resolvers = ClassLoaderUtils.getServices(_NAME_RESOLVER_CLASS_NAME);
for (NameResolver customNameResolver : resolvers)
{
try
{
customNameResolver.getProvider(name);
// NameResolver has to throw IOException if unable to serve a name. It should not return null.
// therefore, if the line above did not throw an exception that means we can proceed with this resolver
// cache the resolver and return
appMap.put(_SERVICES_RESOLVER_KEY, customNameResolver);
return customNameResolver;
}
catch (IOException e)
{
// Log fine message. Try the next factory to get a provider
if (_LOG.isFine())
_LOG.fine(_SERVICES_RESOLVER_IOEXCEPTION_MSG);
}
}
return null;
}
// A subclass of URLInputStreamProvider which never checks for
// modifications
private static class StaticURLInputStreamProvider
extends URLInputStreamProvider
{
public StaticURLInputStreamProvider(URL url)
{
super(url);
}
@Override
public boolean hasSourceChanged()
{
return false;
}
}
// for META-INF\services\org.apache.myfaces.trinidad.share.io.NameResolver
// In this file they will have a line like "org.mycompany.io.MyNameResolverImpl"
static private final String _NAME_RESOLVER_CLASS_NAME =
NameResolver.class.getName();
// Error messages
private static final String _STYLES_DIR_ERROR =
"Could not locate the Trinidad styles directory."
+ "Please be sure that the Trinidad installable resources are installed.";
private static final String _SERVICES_RESOLVER_IOEXCEPTION_MSG =
"IOException when calling the META-INF/services NameResolver's getProvider method. " +
"Trying next nameResolver.";
private static final String _SERVICES_RESOLVER_KEY =
"org.apache.myfaces.trinidadinternal.skin.SERVICES_RESOLVER_KEY";
private static final TrinidadLogger _LOG = TrinidadLogger.createTrinidadLogger(StyleSheetEntry.class);
private String _name;
private StyleSheetDocument _document;
}