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

org.apache.myfaces.trinidadinternal.style.util.StableNameUtils 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.style.util;

import java.util.Arrays;
import java.util.Collection;

import java.util.Collections;

import java.util.HashSet;
import java.util.Iterator;
import java.util.Locale;

import java.util.TreeSet;

import org.apache.myfaces.trinidad.context.Version;
import org.apache.myfaces.trinidad.util.Range;
import org.apache.myfaces.trinidad.logging.TrinidadLogger;
import org.apache.myfaces.trinidadinternal.agent.TrinidadAgent.Application;
import org.apache.myfaces.trinidadinternal.skin.AgentAtRuleMatcher;
import org.apache.myfaces.trinidadinternal.style.StyleContext;
import org.apache.myfaces.trinidadinternal.style.util.StyleSheetVisitUtils.StyleSheetVisitor;
import org.apache.myfaces.trinidadinternal.style.xml.XMLConstants;
import org.apache.myfaces.trinidadinternal.style.xml.parse.StyleSheetDocument;
import org.apache.myfaces.trinidadinternal.style.xml.parse.StyleSheetNode;
import org.apache.myfaces.trinidadinternal.util.nls.LocaleUtils;

/**
 * This class serves one purpose: it provides a replacement for NameUtils.getContextName()
 * that produces "stable" file names.  This ensures that generated style sheets will be
 * named in a consistent manner that is not dependent on the order in which requests arrives.
 * 
 * See the "Structure of Stable Style Sheet Names" section in the Skinning chapter of the
 * Trinidad Developer's Guide for the specification of stable style sheet names.
 */
public final class StableNameUtils
{
  /**
   * Similar to NameUtils.getContextName(), but produces stable names.
   * 
   * @param context the current style context
   * @param document the skin document
   * @return a name suitable for inclusion in a generated style sheet file name.
   */
  public static String getContextName(
    StyleContext       context,
    StyleSheetDocument document
    )
  {
    return new ContextNameProducer(context, document).getName();
  }
  
  // Helper class that produces context-specific file names.
  private static final class ContextNameProducer
  {
    // The ContextNameProducer implementation is designed for modularity/code 
    // cleanliness more than performance.  Each name that is produced generates
    // a fairly large number of temporary objects (most iterators).  This seems
    // acceptable as we only generate each name once per generated file, when the
    // file is initially written.  If performance/temporary object creation
    // ever become an issue, we can optimize/refactor.

    public ContextNameProducer(
      StyleContext       context,
      StyleSheetDocument document
      )
    {
      _context = context;
      _documentId = document.getDocumentId(context);
      _styleSheets = document.getStyleSheetsAsCollection(context);
    }

    /**
     * Returns the context-specific file name.
     */
    public String getName()
    {
      // We're going to collect all required info (eg. platform, agent,
      // locale, etc...) in a single StyleSheetVisitUtils.visit(), during
      // which we delegate to a number of "naming" StyleSheetVisitors
      // to collect the info that we need to produce a name.
      Collection visitors = _getVisitors();

      _visitStyleSheets(visitors);

      return _toName(visitors);
    }

    private Collection _getVisitors()
    {
      return Arrays.asList(
               new PlatformNameExtractor(),
               new ApplicationNameExtractor(_context.getAgent().getVersion()),
               new LocaleNameExtractor(),
               new DirectionNameExtractor(),
               new AccessibilityNameExtractor());
    }

    private void _visitStyleSheets(Collection visitors)
    {
      StyleSheetVisitor compoundVisitor = StyleSheetVisitUtils.compoundStyleSheetVisitor(visitors);
      StyleSheetVisitUtils.visitStyleSheets(_styleSheets, compoundVisitor);      
    }

    private String _toName(Collection visitors)
    {
      // Picking a slightly large initial size in the hopes that this will
      // avoid reallocations.
      StringBuilder builder = new StringBuilder(100);
      
      _appendSkinIdentifierSection(builder);
      _appendVariantsSection(builder, visitors);
      _appendContextualSection(builder);
      _appendSuffix(builder);
      
      return builder.toString();
    }

    private void _appendSkinIdentifierSection(StringBuilder builder)
    {
      // The document id is actually just the content/version hash.  The skin
      // id is later prepended in SkinStyleProvider.getTargetStyleSheetName().
      builder.append(_documentId);
      
      // Double-separator to make it easier to write regular
      // expressions against the interesting portion of the name
      // (eg. for uri rewriting purposes).
      builder.append(_SEPARATOR);
      builder.append(_SEPARATOR);      
    }
    
    private void _appendVariantsSection(
      StringBuilder builder,
      Collection visitors
      )
    {
      for (NamingStyleSheetVisitor visitor : visitors)
      {
        visitor.appendName(builder);
        builder.append(_SEPARATOR);
      }
      
      // Another double-dash separator to aid regular expression
      // targeting.  This may come in handy if we ever end up
      // introducing new variants that we want to add to the
      // visit-based names section.
      builder.append(_SEPARATOR);
    }

    private void _appendContextualSection(StringBuilder builder)
    {
      builder.append(_context.isPortletMode() ? "p" : "s");
      builder.append(_SEPARATOR);
      builder.append(_context.isRequestSecure() ? "s" : "n");
      builder.append(_SEPARATOR);
      builder.append(_context.isDisableStyleCompression() ? "u" : "c");
    }
    
    private void _appendSuffix(StringBuilder builder)
    {
      builder.append(".css");
    }

    private final StyleContext               _context;
    private final String                     _documentId;
    private final Collection _styleSheets;    
  }
  
  // Extension of StyleSheetVisitor that contributes to name generation
  private interface NamingStyleSheetVisitor extends StyleSheetVisitor
  {
    /**
     * Appends the the file name that is being built up in the specified
     * StringBuilder.
     */
    public void appendName(StringBuilder builder);
  }

  // Abstract base class for NamingStyleSheetVisitors that operate on the 
  // collections coughed up by StyleSheetNode.  Handles three possible
  // scenarios:
  //
  // 1. There is no explicit match - eg. the skin does not define any @locale
  //    rules, so StyleSheetNode.getLocales() is always empty.  In this case,
  //    we always append the default match path token.
  // 2. There is a single match - eg. the skin defines "@locale ja" and this
  //    StyleSheetNode is visited.  In this case, we append the string representation
  //    of the matched value.
  // 3. There is a multiple match (and no single match) - eg. the skin defines
  //    "@locale ja, cz, ko" and this StyleSheetNode is visited.  In this case,
  //    we append a concatenation of the string representation of each value,
  //    separated by the underscore character.
  private abstract static class CollectionNameExtractor implements NamingStyleSheetVisitor
  {
    /**
     * Returns the collection of values that contribute to the name.
     */
    abstract protected Collection getCollectionValues(StyleSheetNode styleSheet);
    
    @Override
    public void visit(StyleSheetNode styleSheet)
    {
      Collection newValues = getCollectionValues(styleSheet);
      if (!newValues.isEmpty())
      {
        if (_values == null)
        {
          // Need to make a copy since we'll be modifying this collection.
          _values = new HashSet(newValues);
        }
        else
        {
          mergeValues(styleSheet, _values, newValues);

          // We should never reach a state where the collected values
          // is empty.  This would indicate that our style sheet node
          // matching logic is wrong - eg. we matched a style sheet
          // node with "@locale ja" and a second style sheet node with
          // "@locale ko".  Make some noise if we hit this.
          if (_values.isEmpty())
          {
            _fail();
          }
        }
      }
    }

    /**
     * Merges previously collected values with a new collection of values,
     * storing the result in oldValues.
     * 
     * By default, the intersection of the two collections is retained.
     */
    protected void mergeValues(
      StyleSheetNode styleSheet,
      Collection  oldValues,
      Collection  newValues
      )
    {
      oldValues.retainAll(newValues);
    }

    @Override
    public void appendName(StringBuilder builder)
    {
      if (_values == null)
      {
        builder.append(_DEFAULT_PATH_TOKEN);
      }
      else
      {
        assert(_values != null);
        assert(!_values.isEmpty());
        
        Collection names = _toNames(_values);
        _appendNames(builder, names);
      }
    }
    
    private Collection _toNames(Collection values)
    {
      Collection names = new TreeSet();
      
      for (E value : values)
      {
        String name = toName(value);
        assert(name != null);
        
        names.add(name);
      }
      
      return names;
    }

    /**
     * Converts the specified value to a name to include in the 
     * file name.
     */
    protected String toName(E value)
    {
      return value.toString();
    }

    private void _appendNames(StringBuilder builder, Collection names)
    {
      assert(names != null);
      assert(!names.isEmpty());
      
      Iterator iter = names.iterator();
      while (iter.hasNext())
      {
        builder.append(iter.next());
        
        if (iter.hasNext())
        {
          builder.append("_");
        }
      }
    }
    
    private Collection _values = null;
  }

  // NamingStyleSheetVisitor that extracts the platform name.
  private static final class PlatformNameExtractor extends CollectionNameExtractor
  {
    @Override
    protected Collection getCollectionValues(StyleSheetNode styleSheet)
    {
      return styleSheet.getPlatforms();
    }
    
    @Override
    protected String toName(Integer platform)
    {
      return NameUtils.getPlatformName(platform);
    }
  }

  // NamingStyleSheetVisitor that extracts the agent application name
  private static class ApplicationNameExtractor extends CollectionNameExtractor
  {
    public ApplicationNameExtractor(Version agentVersion)
    {
      assert(agentVersion != null);

      _agentVersion = agentVersion;
      _matchedVersions = Version.ALL_VERSIONS;
    }

    @Override
    protected String toName(Application agentApplication)
    {
      return agentApplication.getAgentName();
    }

    @Override
    protected Collection getCollectionValues(StyleSheetNode styleSheet)
    {
      AgentAtRuleMatcher agentMatcher = styleSheet.getAgentMatcher();
      if (agentMatcher == null)
      {
        return Collections.emptySet();
      }
      
      return agentMatcher.getAllApplications();
    }

    @Override
    protected void mergeValues(
      StyleSheetNode          styleSheet,
      Collection oldAgentApplications,
      Collection newAgentApplications
      )
    {
      super.mergeValues(styleSheet, oldAgentApplications, newAgentApplications);

      // In addition to merging application values, we also need to keep track of
      // versions that we have seen, since this may be included in the generated
      // file name.  Note that we only care about versions for cases where we've
      // got an exact agent application match (ie. in cases where we actually know
      // which application we are targeted.)
      Application newAgentApplication = _getSingleAgentApplication(newAgentApplications);
      if (newAgentApplication != null)
      {
        _mergeVersions(styleSheet.getAgentMatcher(), newAgentApplication);
      }
    }

    // If the collection contains a single entry, returns the single value.
    // Otherwise, returns null;
    private Application _getSingleAgentApplication(Collection agentApplications)
    {
      if (agentApplications.size() == 1)
      {
         Iterator iter = agentApplications.iterator();
         if (iter.hasNext())
         {
           return iter.next();
         }
      }
      
      return null;
    }

    private void _mergeVersions(
      AgentAtRuleMatcher agentMatcher,
      Application        agentApplication
      )
    {
      assert(agentMatcher != null);
      assert(agentApplication != null);

      Range matchedVersions =
        agentMatcher.getMatchedVersionsForApplication(agentApplication, _agentVersion);
      _matchedVersions = _matchedVersions.intersect(matchedVersions);

      // We should never see an empty range at this point since the two
      // ranges that we intersected both contains _agentVersion.
      if (_matchedVersions.isEmpty())
      {
        _fail();
      }
    }

    @Override
    public void appendName(StringBuilder builder)
    {
      // This super call will handle appending the agent name segment.
      // However, we also need to ensure that the agent version is added.
      super.appendName(builder);
      
      // Tack on a value for the version field.  (All stable names need to
      // have the same # of fields in order to support allow uris/names to
      // be easily targeted by uri rewriting regular expressions.)
      builder.append(_SEPARATOR);
      _appendVersionName(builder);      
    }

    private void _appendVersionName(StringBuilder builder)
    {
      Version startVersion = _matchedVersions.getStart();
      Version endVersion = _matchedVersions.getEnd();
      boolean startMin = startVersion.equals(Version.MIN_VERSION);
      boolean endMax = endVersion.equals(Version.MAX_VERSION);
      
      if (startMin && endMax)
      {
        builder.append(_DEFAULT_PATH_TOKEN);
      }
      else if (startMin)
      {
        builder.append(endVersion);
        builder.append("m");
      }
      else if (endMax)
      {
        builder.append(startVersion);
        builder.append("p");
      }
      else
      {
        builder.append(startVersion);
      }
    }

    private final Version  _agentVersion;
    private Range _matchedVersions;
  }

  // NamingStyleSheetVisitor that extracts the locale name.
  private static final class LocaleNameExtractor extends CollectionNameExtractor
  {
    @Override
    protected Collection getCollectionValues(StyleSheetNode styleSheet)
    {
      return styleSheet.getLocales();
    }

    @Override
    protected void mergeValues(
      StyleSheetNode     styleSheet,
      Collection oldLocales,
      Collection newLocales
      )
    {
      assert(oldLocales != null);
      assert(newLocales != null);

      // We can't simply use Collection.retainAll() because we need to
      // compensate for partial locales - ie. if "ja" is present in
      // oldLocales and we encounter "ja_JP", Collection.retainAll()
      // would result in the empty set, where as we want to replace "ja"
      // with "ja_JP".
      
      for (Locale newLocale : newLocales)
      {
        _retainLocale(oldLocales, newLocale); 
      }
    }
    
    // We only want to retain the new locale if:
    //
    // a) it exist in the old locales, or...
    // b) one of its "super" locales exists in the old locales.
    //
    // In the case of b), we want to remove the super-locale and
    // repalce it with the new, more-specific locale.
    private void _retainLocale(Collection oldLocales, Locale newLocale)
    {
      // We could optimize by explicitly checking whether the new locale actually
      // specifies a country/variant, but prefer to keep the code simple.
      Locale langOnlyLocale = _toLanguageOnlyLocale(newLocale);
      Locale langAndCountryLocale = _toLanguageAndCountryLocale(newLocale);
      
      if (oldLocales.contains(langOnlyLocale) || oldLocales.contains(langAndCountryLocale))
      {
        oldLocales.remove(langOnlyLocale);
        oldLocales.remove(langAndCountryLocale);
        oldLocales.add(newLocale);
      }
    }
    
    private Locale _toLanguageOnlyLocale(Locale locale)
    {
      return new Locale(locale.getLanguage());
    }

    private Locale _toLanguageAndCountryLocale(Locale locale)
    {
      return new Locale(locale.getLanguage(), locale.getCountry());
    }
  }
  
  // NamingStyleSheetVisitor that extracts the direction name
  private static class DirectionNameExtractor implements NamingStyleSheetVisitor
  {
    @Override
    public void visit(StyleSheetNode styleSheet)
    {
      int direction = styleSheet.getReadingDirection();   
      if (direction != LocaleUtils.DIRECTION_DEFAULT)
      {
        assert((_direction == LocaleUtils.DIRECTION_DEFAULT) || (_direction == direction));
        _direction = direction;
      }
    }

    @Override
    public void appendName(StringBuilder builder)
    {
      String name = (_direction == LocaleUtils.DIRECTION_DEFAULT) ? 
                      _DEFAULT_PATH_TOKEN : 
                      NameUtils.getDirectionName(_direction);
      
      builder.append(name);
    }
    
    private int _direction = LocaleUtils.DIRECTION_DEFAULT;
  }

  // NamingStyleSheetVisitor that extracts the accessibility name
  private static final class AccessibilityNameExtractor extends CollectionNameExtractor
  {
    @Override
    protected Collection getCollectionValues(StyleSheetNode styleSheet)
    {
      return styleSheet.getAccessibilityProperties();
    }
    
    protected void mergeValues(
      StyleSheetNode styleSheet,
      Collection oldAccessibilityProperties,
      Collection newAccessibilityProperties
      )
    {
      oldAccessibilityProperties.addAll(newAccessibilityProperties);
    }
    
    @Override
    protected String toName(String accessibilityProperty)
    {
      String name = null;
      
      if (XMLConstants.ACC_HIGH_CONTRAST.equals(accessibilityProperty))
      {
        name = "hc";
      }
      
      if (XMLConstants.ACC_LARGE_FONTS.equals(accessibilityProperty))
      {
        name = "lf";
      }
      
      assert(name != null);

      return name;
    }
  }
  
  private static void _fail()
  {
    String message = _LOG.getMessage("SKIN_GENERATION_ERROR");
    throw new IllegalStateException(message);
  }

  private StableNameUtils()
  {
  }
  
  private static final char _SEPARATOR = '-';

  private static final String _DEFAULT_PATH_TOKEN = "d";
  
  private static final TrinidadLogger _LOG =
    TrinidadLogger.createTrinidadLogger(StableNameUtils.class);  
  
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy