blob: 26eb517b057040f18a194952c589b98083147f5f [file] [log] [blame]
// Copyright 2017 The Bazel Authors. All rights reserved.
//
// 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.devtools.build.android;
import static com.google.common.truth.Fact.fact;
import static com.google.common.truth.Fact.simpleFact;
import static com.google.common.truth.Truth.assertThat;
import static java.util.Arrays.asList;
import static java.util.stream.Collectors.joining;
import com.google.common.base.Joiner;
import com.google.common.truth.FailureMetadata;
import com.google.common.truth.Subject;
import com.google.devtools.build.android.aapt2.ResourceCompiler.CompiledType;
import java.io.IOException;
import java.io.StringReader;
import java.io.StringWriter;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import javax.annotation.Nullable;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.OutputKeys;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import org.w3c.dom.Document;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
/** A testing utility that allows assertions against Paths. */
public class PathsSubject extends Subject {
private static final String PATH_NORMALIZER =
String.format(
"(%s)/(.*/)(javatests)",
Arrays.stream(CompiledType.values())
.map(CompiledType::asPrefix)
.map(p -> String.format("(?:%s)", p))
.collect(joining("|")));
private final Path actual;
PathsSubject(FailureMetadata failureMetadata, @Nullable Path subject) {
super(failureMetadata, subject);
this.actual = subject;
}
void exists() {
if (actual == null) {
failWithoutActual(simpleFact("expected not to be null"));
}
if (!Files.exists(actual)) {
failWithActual(simpleFact("expected to exist"));
}
}
void containsExactlyUncompressedFilesIn(String... paths) throws IOException {
if (actual == null) {
failWithoutActual(simpleFact("expected not to be null"));
}
exists();
assertThat(
new ZipFile(actual.toFile())
.stream()
.filter(entry -> entry.getMethod() == ZipEntry.STORED)
.map(ZipEntry::getName)
.map(n -> n.replaceAll(PATH_NORMALIZER, "$1/$3"))
.collect(Collectors.toSet()))
.containsExactlyElementsIn(paths);
}
void containsAllArchivedFilesIn(String... paths) throws IOException {
if (actual == null) {
failWithoutActual(simpleFact("expected not to be null"));
}
exists();
assertThat(
new ZipFile(actual.toFile())
.stream()
.map(ZipEntry::getName)
.map(n -> n.replaceAll(PATH_NORMALIZER, "$1/$3"))
.collect(Collectors.toSet()))
.containsAtLeastElementsIn(Arrays.asList(paths));
}
void containsExactlyArchivedFilesIn(String... paths) throws IOException {
if (actual == null) {
failWithoutActual(simpleFact("expected not to be null"));
}
exists();
assertThat(
new ZipFile(actual.toFile())
.stream()
.map(ZipEntry::getName)
.map(n -> n.replaceAll(PATH_NORMALIZER, "$1/$3"))
.collect(Collectors.toSet()))
.containsExactly(Arrays.asList(paths));
}
void containsNoArchivedFilesIn(String... paths) throws IOException {
if (actual == null) {
failWithoutActual(simpleFact("expected not to be null"));
}
exists();
assertThat(
new ZipFile(actual.toFile())
.stream()
.map(ZipEntry::getName)
.map(n -> n.replaceAll(PATH_NORMALIZER, "$1/$3"))
.collect(Collectors.toSet()))
.containsNoneIn(Arrays.asList(paths));
}
void xmlContentsIsEqualTo(String... contents) {
if (actual == null) {
failWithoutActual(simpleFact("expected not to be null"));
}
exists();
try {
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
Transformer transformer = TransformerFactory.newInstance().newTransformer();
transformer.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "yes");
transformer.setOutputProperty(OutputKeys.INDENT, "no");
assertThat(
normalizeXml(
newXmlDocument(factory, Files.readAllLines(actual, StandardCharsets.UTF_8)),
transformer))
.isEqualTo(normalizeXml(newXmlDocument(factory, Arrays.asList(contents)), transformer));
} catch (IOException | SAXException | ParserConfigurationException | TransformerException e) {
failWithoutActual(
fact("expected to have contents", asList(contents)),
fact("but failed to read file with", e),
fact("path was", actual));
}
}
private Document newXmlDocument(DocumentBuilderFactory factory, List<String> contents)
throws SAXException, IOException, ParserConfigurationException {
return factory
.newDocumentBuilder()
.parse(new InputSource(new StringReader(Joiner.on("").join(contents))));
}
private String normalizeXml(Document doc, Transformer transformer) throws TransformerException {
StringWriter writer = new StringWriter();
transformer.transform(new DOMSource(doc), new StreamResult(writer));
return writer.toString().replaceAll("\n|\r", "").replaceAll(">\\s+<", "><");
}
public void containsExactlyTheseLines(String... lines) throws IOException {
assertThat(Files.readAllLines(actual, StandardCharsets.UTF_8)).containsExactlyElementsIn(lines);
}
}