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

com.google.gwt.user.rebind.rpc.ProblemReport Maven / Gradle / Ivy

/*
 * Copyright 2008 Google Inc.
 * 
 * Licensed 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 com.google.gwt.user.rebind.rpc;

import com.google.gwt.core.ext.TreeLogger;
import com.google.gwt.core.ext.TreeLogger.Type;
import com.google.gwt.core.ext.typeinfo.JClassType;

import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;

/**
 * A collection of reported problems; these are accumulated during the
 * SerializableTypeOracleBuilder's isSerializable analysis, and what to do about
 * the problems is decided only later.
 */
public class ProblemReport {

  /**
   * Priority of problems. {@link #FATAL} problems will fail a build that would
   * otherwise have succeeded, for example because of a bad custom serializer
   * used only as a subclass of a superclass with other viable subtypes.
   * {@link #DEFAULT} problems might or might not be fatal, depending on overall
   * results accumulated later. {@link #AUXILIARY} problems are not fatal, and
   * often not even problems by themselves, but diagnostics related to default
   * problems (e.g. type filtration, which might suppress an
   * intended-to-serialize class).
   */
  public enum Priority {
    FATAL, DEFAULT, AUXILIARY
  }

  /**
   * An individual report, which may require multiple entries (expressed as logs
   * under a branchpoint), but relates to an individual issue.
   */
  public static class Problem {
    private String message;
    private List childMessages;

    private Problem(String message, String[] children) {
      this.message = message;
      // most problems don't have sub-messages, so init at zero size
      childMessages = new ArrayList(children.length);
      for (int i = 0; i < children.length; i++) {
        childMessages.add(children[i]);
      }
    }

    public void addChild(String message) {
      childMessages.add(message);
    }

    public String getPrimaryMessage() {
      return message;
    }

    public Iterable getSubMessages() {
      return childMessages;
    }

    public boolean hasSubMessages() {
      return !childMessages.isEmpty();
    }
  }

  private Map> allProblems;
  private Map> auxiliaries;
  private Map> fatalProblems;
  private JClassType contextType;

  /**
   * Creates a new, empty, context-less ProblemReport.
   */
  public ProblemReport() {
    Comparator comparator = new Comparator() {
      public int compare(JClassType o1, JClassType o2) {
        assert o1 != null;
        assert o2 != null;
        return o1.getParameterizedQualifiedSourceName().compareTo(
            o2.getParameterizedQualifiedSourceName());
      }
    };
    allProblems = new TreeMap>(comparator);
    auxiliaries = new TreeMap>(comparator);
    fatalProblems = new TreeMap>(comparator);
    contextType = null;
  }

  /**
   * Adds a problem for a given type. This also sorts the problems into
   * collections by priority.
   * 
   * @param type the problematic type
   * @param message the description of the problem
   * @param priority priority of the problem.
   * @param extraLines additional continuation lines for the message, usually
   *          for additional explanations.
   */
  public Problem add(JClassType type, String message, Priority priority, String... extraLines) {
    String contextString = "";
    if (contextType != null) {
      contextString = " (reached via " + contextType.getParameterizedQualifiedSourceName() + ")";
    }
    message = message + contextString;
    Problem entry = new Problem(message, extraLines);
    if (priority == Priority.AUXILIARY) {
      addToMap(type, entry, auxiliaries);
      return entry;
    }

    // both FATAL and DEFAULT problems go in allProblems...
    addToMap(type, entry, allProblems);

    // only FATAL problems go in fatalProblems...
    if (priority == Priority.FATAL) {
      addToMap(type, entry, fatalProblems);
    }
    return entry;
  }

  public String getWorstMessageForType(JClassType type) {
    List list = fatalProblems.get(type);
    if (list == null) {
      list = allProblems.get(type);
      if (list == null) {
        list = auxiliaries.get(type);
      }
    }
    if (list == null) {
      return null;
    }
    return list.get(0).getPrimaryMessage() + (list.size() > 1 ? ", etc." : "");
  }

  /**
   * Were any problems reported as "fatal"?
   */
  public boolean hasFatalProblems() {
    return !fatalProblems.isEmpty();
  }

  /**
   * Reports all problems to the logger supplied, at the log level supplied. The
   * problems are assured of being reported in lexographic order of type names.
   * 
   * @param logger logger to receive problem reports
   * @param problemLevel severity level at which to report problems.
   * @param auxLevel severity level at which to report any auxiliary messages.
   */
  public void report(TreeLogger logger, TreeLogger.Type problemLevel, TreeLogger.Type auxLevel) {
    doReport(logger, auxLevel, auxiliaries);
    doReport(logger, problemLevel, allProblems);
  }

  /**
   * Reports only urgent problems to the logger supplied, at the log level
   * supplied. The problems are assured of being reported in lexographic order
   * of type names.
   * 
   * @param logger logger to receive problem reports
   * @param level severity level at which to report problems.
   */
  public void reportFatalProblems(TreeLogger logger, TreeLogger.Type level) {
    doReport(logger, level, fatalProblems);
  }

  /**
   * Sets the context type currently being analyzed. Problems found will include
   * reference to this context, until reset with another call to this method.
   * Context may be canceled with a {@code null} value here.
   * 
   * @param newContext the type under analysis
   */
  public void setContextType(JClassType newContext) {
    contextType = newContext;
  }

  /**
   * Test accessor returning list of auxiliary "problems" logged against a given
   * type.
   * 
   * @param type type to fetch problems for
   * @return {@code null} if no auxiliaries were logged. Otherwise, a list of
   *         strings describing messages, including the context in which the
   *         problem was found.
   */
  List getAuxiliaryMessagesForType(JClassType type) {
    List list = auxiliaries.get(type);
    if (list == null) {
      list = new ArrayList(0);
    }
    return list;
  }

  /**
   * Test accessor returning list of problems logged against a given type.
   * 
   * @param type type to fetch problems for
   * @return {@code null} if no problems were logged. Otherwise, a list of
   *         strings describing problems, including the context in which the
   *         problem was found.
   */
  List getProblemsForType(JClassType type) {
    List list = allProblems.get(type);
    if (list == null) {
      list = new ArrayList(0);
    }
    return list;
  }

  /**
   * Adds an entry to one of the problem maps.
   * 
   * @param type the type to add
   * @param message the message to add for {@code type}
   * @param map the map to add to
   */
  private void addToMap(JClassType type, Problem problem, Map> map) {
    List list = map.get(type);
    if (list == null) {
      list = new ArrayList();
      map.put(type, list);
    }
    list.add(problem);
  }

  /**
   * Logs all of the problems from one of the problem maps.
   * 
   * @param logger the logger to log to
   * @param level the level for messages
   * @param problems the problems to log
   */
  private void doReport(TreeLogger logger, Type level, Map> problems) {
    if (!logger.isLoggable(level)) {
      return;
    }
    for (List problemList : problems.values()) {
      for (Problem problem : problemList) {
        if (problem.hasSubMessages()) {
          TreeLogger sublogger = logger.branch(level, problem.getPrimaryMessage());
          for (String sub : problem.getSubMessages()) {
            sublogger.log(level, sub);
          }
        } else {
          logger.log(level, problem.getPrimaryMessage());
        }
      }
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy