blob: 87a76a79bc76c9b2e6727efeebdc0190e5171b51 [file] [log] [blame]
/*
* Copyright 2013 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.dev.javac;
import com.google.gwt.core.ext.TreeLogger;
import com.google.gwt.dev.CompilerContext;
import com.google.gwt.dev.util.UnitTestTreeLogger;
import junit.framework.TestCase;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* Tests for {@link CompilationUnitInvalidator}.
*/
public class CompilationUnitInvalidatorTest extends TestCase {
public void testRetainValidUnits() {
/*
* bad0 is a bad unit in validClasses
* bad1 is directly a bad unit
* bad2 depends on bad0
* bad3 depends on bad1
* bad4 has a missing dependency
* bad5 depends on good0, good1, bad3
* bad6 depends on bad5
*
* good0 is a good member of validClasses
* good1 depends on good0
* good2 depends on good1
* good3 depends on good0, good1, good2
*/
CompilationUnit bad0 = getUnitWithApiRefs(true, "bad0");
CompilationUnit bad1 = getUnitWithApiRefs(true, "bad1");
CompilationUnit bad2 = getUnitWithApiRefs(false, "bad2", "bad0");
CompilationUnit bad3 = getUnitWithApiRefs(false, "bad3", "bad1");
CompilationUnit bad4 = getUnitWithApiRefs(false, "bad4", "missing0");
CompilationUnit bad5 = getUnitWithApiRefs(false,
"bad5", "good0", "good1", "bad3");
CompilationUnit bad6 = getUnitWithApiRefs(false, "bad6", "bad5");
CompilationUnit good0 = getUnitWithApiRefs(false, "good0");
CompilationUnit good1 = getUnitWithApiRefs(false, "good1", "good0");
CompilationUnit good2 = getUnitWithApiRefs(false, "good2", "good1");
CompilationUnit good3 = getUnitWithApiRefs(false, "good3",
"good0", "good1", "good2");
Collection<CompilationUnit> units = new ArrayList<CompilationUnit>();
units.addAll(Arrays.asList(
bad1, bad2, bad3, bad4, bad5, bad6,
good1, good2, good3));
Map<String, CompiledClass> validClasses = new HashMap<String, CompiledClass>();
validClasses.put("bad0", bad0.getCompiledClasses().iterator().next());
validClasses.put("good0", good0.getCompiledClasses().iterator().next());
// At least some of the members of units also appear within validClasses
// By putting bad2 here, this test ensures that bad3 (and dependents)
// will be removed even though bad2 is in validClasses.
validClasses.put("bad2", bad2.getCompiledClasses().iterator().next());
validClasses.put("good1", good1.getCompiledClasses().iterator().next());
// Keep a copy so that we can check that the one passed-in isn't mutated
Map<String, CompiledClass> knownValidClasses =
new HashMap<String, CompiledClass>(validClasses);
// Collect more compilation errors where extra units are thrown out by unit invalidation.
CompilerContext compilerContext = new CompilerContext.Builder().build();
// Invoke the method under test
CompilationUnitInvalidator.retainValidUnits(TreeLogger.NULL, units, validClasses,
compilerContext.getCompilationErrorsIndex());
// Check that the compilation errors index was correctly populated.
UnitTestTreeLogger.Builder loggerBuilder = new UnitTestTreeLogger.Builder();
loggerBuilder.setLowestLogLevel(TreeLogger.TRACE);
loggerBuilder.expectTrace("Tracing compile failure path for type 'bad6'", null);
loggerBuilder.expectError("Errors in '/mock/bad6.java'", null);
loggerBuilder.expectError("bad5 cannot be resolved to a type", null);
loggerBuilder.expectError("Errors in '/mock/bad5.java'", null);
loggerBuilder.expectError("bad3 cannot be resolved to a type", null);
loggerBuilder.expectError("Errors in '/mock/bad3.java'", null);
loggerBuilder.expectError("bad1 cannot be resolved to a type", null);
UnitTestTreeLogger testLogger = loggerBuilder.createLogger();
CompilationProblemReporter.logErrorTrace(testLogger, TreeLogger.ERROR, compilerContext, "bad6",
false);
testLogger.assertCorrectLogEntries();
// Check that validClasses is not mutated
assertEquals(knownValidClasses, validClasses);
// Check that units is mutated
assertEquals(Arrays.asList(good1, good2, good3), units);
}
private static CompilationUnit getUnitWithApiRefs(
final boolean isError, final String simpleName, final String... apiRefs) {
return new MockCompilationUnit(simpleName, simpleName) {
@Override
public boolean isError() {
return isError;
}
@Override
public Collection<CompiledClass> getCompiledClasses() {
CompiledClass cc = new CompiledClass(new byte[1], null, false, simpleName, simpleName);
cc.initUnit(this);
return Collections.<CompiledClass>singletonList(cc);
}
@Override
public Dependencies getDependencies() {
return new Dependencies() {
@Override
public List<String> getApiRefs() {
return Arrays.asList(apiRefs);
}
};
}
};
}
}