com.google.javascript.refactoring.FixingErrorManager.java Source code

Java tutorial

Introduction

Here is the source code for com.google.javascript.refactoring.FixingErrorManager.java

Source

/*
 * Copyright 2015 The Closure Compiler Authors.
 *
 * 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.javascript.refactoring;

import static com.google.javascript.jscomp.CheckMissingAndExtraRequires.EXTRA_REQUIRE_WARNING;
import static com.google.javascript.jscomp.CheckMissingAndExtraRequires.MISSING_REQUIRE_STRICT_WARNING;
import static com.google.javascript.jscomp.CheckMissingAndExtraRequires.MISSING_REQUIRE_WARNING;
import static com.google.javascript.jscomp.ClosureCheckModule.JSDOC_REFERENCE_TO_SHORT_IMPORT_BY_LONG_NAME_INCLUDING_SHORT_NAME;
import static com.google.javascript.jscomp.ClosureCheckModule.REFERENCE_TO_SHORT_IMPORT_BY_LONG_NAME_INCLUDING_SHORT_NAME;

import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ListMultimap;
import com.google.javascript.jscomp.AbstractCompiler;
import com.google.javascript.jscomp.BasicErrorManager;
import com.google.javascript.jscomp.CheckLevel;
import com.google.javascript.jscomp.DiagnosticType;
import com.google.javascript.jscomp.JSError;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;

/**
 * <p>An error manager that finds a SuggestedFix for all errors if possible.
 */
public class FixingErrorManager extends BasicErrorManager {
    private AbstractCompiler compiler;
    private final HashMap<JSError, SuggestedFix> sureFixes = new HashMap<>();
    private final ListMultimap<JSError, SuggestedFix> multiFixes = ArrayListMultimap.create();
    private final ImmutableSet<DiagnosticType> unfixableErrors;

    private enum FixTypes {
        ONE_FIX, ONE_FIX_AND_MULTI_FIXES,
    }

    public FixingErrorManager() {
        this(ImmutableSet.of());
    }

    public FixingErrorManager(ImmutableSet<DiagnosticType> unfixableErrors) {
        this.unfixableErrors = unfixableErrors;
    }

    public void setCompiler(AbstractCompiler compiler) {
        this.compiler = compiler;
    }

    @Override
    public void report(CheckLevel level, JSError error) {
        super.report(level, error);
        if (!unfixableErrors.contains(error.getType())) {
            ImmutableList<SuggestedFix> fixes = ErrorToFixMapper.getFixesForJsError(error, compiler);
            if (fixes.size() == 1) {
                sureFixes.put(error, fixes.get(0));
            } else {
                multiFixes.putAll(error, fixes);
            }
        }
    }

    private boolean containsFixableShorthandModuleWarning() {
        for (JSError error : sureFixes.keySet()) {
            if (error.getType().equals(JSDOC_REFERENCE_TO_SHORT_IMPORT_BY_LONG_NAME_INCLUDING_SHORT_NAME)
                    || error.getType().equals(REFERENCE_TO_SHORT_IMPORT_BY_LONG_NAME_INCLUDING_SHORT_NAME)) {
                return true;
            }
        }
        for (JSError error : multiFixes.keySet()) {
            if (error.getType().equals(JSDOC_REFERENCE_TO_SHORT_IMPORT_BY_LONG_NAME_INCLUDING_SHORT_NAME)
                    || error.getType().equals(REFERENCE_TO_SHORT_IMPORT_BY_LONG_NAME_INCLUDING_SHORT_NAME)) {
                return true;
            }
        }
        return false;
    }

    public List<SuggestedFix> getFixesForJsError(JSError error) {
        if (sureFixes.containsKey(error)) {
            return ImmutableList.of(sureFixes.get(error));
        } else {
            return multiFixes.get(error);
        }
    }

    private Collection<SuggestedFix> getFixes(FixTypes fixTypes) {
        boolean containsFixableShorthandModuleWarning = containsFixableShorthandModuleWarning();
        Collection<SuggestedFix> fixes = new ArrayList<>();
        for (JSError error : getErrors()) {
            // Sometimes code will produce a spurious extra-require or missing-require error,
            // as well as a warning about using a full namespace instead of a shorthand type. In this case
            // don't apply the extra/missing require fix.
            if (containsFixableShorthandModuleWarning && (error.getType().equals(EXTRA_REQUIRE_WARNING)
                    || error.getType().equals(MISSING_REQUIRE_STRICT_WARNING)
                    || error.getType().equals(MISSING_REQUIRE_WARNING))) {
                // Don't apply this fix.
            } else {
                if (fixTypes == FixTypes.ONE_FIX && sureFixes.containsKey(error)) {
                    fixes.add(sureFixes.get(error));
                } else if (fixTypes == FixTypes.ONE_FIX_AND_MULTI_FIXES) {
                    fixes.addAll(getFixesForJsError(error));
                }
            }
        }
        for (JSError warning : getWarnings()) {
            if (warning.getType().equals(EXTRA_REQUIRE_WARNING) && containsFixableShorthandModuleWarning) {
                // As above, don't apply the extra-require fix.
            } else {
                if (fixTypes == FixTypes.ONE_FIX && sureFixes.containsKey(warning)) {
                    fixes.add(sureFixes.get(warning));
                } else if (fixTypes == FixTypes.ONE_FIX_AND_MULTI_FIXES) {
                    fixes.addAll(getFixesForJsError(warning));
                }
            }
        }
        return fixes;
    }

    /** Returns fixes for errors and warnings that only have one 'sure' guaranteed fix. */
    public Collection<SuggestedFix> getSureFixes() {
        return getFixes(FixTypes.ONE_FIX);
    }

    /**
     * Returns fixes for errors first, then fixes for warnings. This includes 'sure' fixes with only
     * one option, and 'multi' fixes which have multiple choices.
     */
    public Collection<SuggestedFix> getAllFixes() {
        return getFixes(FixTypes.ONE_FIX_AND_MULTI_FIXES);
    }

    @Override
    public void printSummary() {
        // No-op
    }

    @Override
    public void println(CheckLevel level, JSError error) {
        // No-op
    }
}