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

com.adobe.epubcheck.vocab.VocabUtil Maven / Gradle / Ivy

Go to download

EPUBCheck is a tool to validate the conformance of EPUB publications against the EPUB specifications. EPUBCheck can be run as a standalone command-line tool or used as a Java library.

There is a newer version: 5.1.0
Show newest version
package com.adobe.epubcheck.vocab;

import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import com.adobe.epubcheck.api.EPUBLocation;
import com.adobe.epubcheck.api.Report;
import com.adobe.epubcheck.messages.LocalizedMessages;
import com.adobe.epubcheck.messages.MessageId;
import com.adobe.epubcheck.opf.ValidationContext;
import com.google.common.base.Optional;
import com.google.common.base.Preconditions;
import com.google.common.base.Splitter;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Maps;

/**
 * Utilities related to property values, vocabularies, and prefix declarations.
 * 
 * @author Romain Deltour
 *
 */
public final class VocabUtil
{
  public static Vocab EMPTY_VOCAB = new EnumVocab(EMPTY.class, "");

  private enum EMPTY
  {
  }

  private static Pattern propertyPattern = Pattern.compile("(([^:]*):)?(.*)");
  private static Splitter whitespaceSplitter = Splitter.onPattern("\\s+").omitEmptyStrings();

  /**
   * Parses a single property value and report validation errors on the fly.
   * 
   * @param value
   *          the value to parse.
   * @param vocabs
   *          a map of prefix to vocabularies.
   * @param context
   *          the validation context (report, locale, path, etc).
   * @param location
   *          the location in the validated file.
   * @return an {@link Optional} containing the property if it was parsed
   *         successfully or nothing if there was a parsing error
   */
  public static Optional parseProperty(String value, Map vocabs,
      ValidationContext context, EPUBLocation location)
  {

    return Optional.fromNullable(
        Iterables.get(parseProperties(value, vocabs, false, context, location), 0, null));
  }

  /**
   * Parses a space-separated list of property values, and report validation
   * errors on the fly.
   * 
   * @param value
   *          the value to parse.
   * @param vocabs
   *          a map of prefix to vocabularies.
   * @param context
   *          the validation context (report, locale, path, etc).
   * @param location
   *          the location in the validated file.
   * @return
   */
  public static Set parsePropertyList(String value, Map vocabs,
      ValidationContext context, EPUBLocation location)
  {
    return parseProperties(value, vocabs, true, context, location);
  }

  private static Set parseProperties(String value, Map vocabs,
      boolean isList, ValidationContext context, EPUBLocation location)
  {
    Preconditions.checkNotNull(vocabs);
    Preconditions.checkNotNull(context);
    Preconditions.checkNotNull(location);
    if (value == null)
    {
      return ImmutableSet.of();
    }

    ImmutableSet.Builder builder = ImmutableSet.builder();

    // split properties, report error if a list is found but not allowed
    Iterable properties = whitespaceSplitter.split(value);
    if (!isList && !Iterables.isEmpty(Iterables.skip(properties, 1)))
    {
      context.report.message(MessageId.OPF_025, location, value);
      return ImmutableSet.of();
    }

    for (String property : properties)
    {
      // parse prefix and local name, report error if malformed
      Matcher matcher = propertyPattern.matcher(property);
      matcher.matches();
      if (matcher.group(1) != null && (matcher.group(2).isEmpty() || matcher.group(3).isEmpty()))
      {
        context.report.message(MessageId.OPF_026, location, property);
        continue;
      }
      String prefix = Strings.nullToEmpty(matcher.group(2));
      String name = matcher.group(3);

      // lookup property in the vocab for its prefix
      // report error if not found
      try
      {
        Optional found = vocabs.get(prefix).lookup(name);
        if (found.isPresent())
        {
          if (found.get().isDeprecated())
          {
            MessageId messageId = (StructureVocab.URI.equals(found.get().getVocabURI()))
                ? MessageId.OPF_086b
                : MessageId.OPF_086;
            // Replacement suggestions for deprecated properties are given
            // in message strings of the form OPF_086_SUG.property-name
            String suggestion = LocalizedMessages.getInstance(context.locale)
                .getSuggestion(messageId, found.get().getName());
            context.report.message(messageId, location, property, suggestion);
          }
          if (!found.get().isAllowed(context))
          {
            context.report.message(MessageId.OPF_087, location, property, context.mimeType);
          }
          builder.add(found.get());
        }
        else
        {
          context.report.message(MessageId.OPF_027, location, property);
          continue;
        }
      } catch (NullPointerException e)
      {
        // vocab not found (i.e. prefix undeclared), report warning
        context.report.message(MessageId.OPF_028, location, prefix);
        continue;
      }
    }
    return builder.build();
  }

  /**
   * Parses a prefix attribute value and returns a map of prefixes to
   * vocabularies, given a pre-existing set of reserved prefixes, known
   * vocabularies, and default vocabularies that cannot be re-declared.
   * 
   * @param value
   *          the prefix declaration to parse.
   * @param predefined
   *          a map of reserved prefixes to associated vocabularies.
   * @param known
   *          a map of known URIs to known vocabularies.
   * @param forbidden
   *          a set of URIs of default vocabularies that cannot be re-declared.
   * @param report
   *          to report errors on the fly.
   * @param location
   *          the location of the attribute in the source file.
   * @return
   */
  public static Map parsePrefixDeclaration(String value,
      Map predefined, Map known,
      Set forbidden, Report report, EPUBLocation location)
  {
    Map vocabs = Maps.newHashMap(predefined);
    Map mappings = PrefixDeclarationParser.parsePrefixMappings(value, report,
        location);
    for (Entry mapping : mappings.entrySet())
    {
      String prefix = mapping.getKey();
      String uri = mapping.getValue();
      if ("_".equals(prefix))
      {
        // must not define the '_' prefix
        report.message(MessageId.OPF_007a, location);
      }
      else if (forbidden.contains(uri))
      {
        // must not declare a default vocab
        report.message(MessageId.OPF_007b, location, prefix);
      }
      else
      {
        if (predefined.containsKey(prefix)
            && !Strings.nullToEmpty(predefined.get(prefix).getURI()).equals(uri))
        {
          // re-declaration of reserved prefix
          report.message(MessageId.OPF_007, location, prefix);
        }
        Vocab vocab = known.get(uri);
        vocabs.put(mapping.getKey(), (vocab == null) ? new UncheckedVocab(uri, prefix) : vocab);
      }
    }
    return ImmutableMap.copyOf(vocabs);
  }

  private VocabUtil()
  {
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy