blob: 0ff7516a13ce4ad483fd865c0b664a773949ee27 [file] [log] [blame]
// Copyright 2015 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 com.android.builder.dependency.SymbolFileProvider;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.android.aapt2.CompiledResources;
import java.io.File;
import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.nio.file.Path;
import java.util.Objects;
import java.util.regex.Pattern;
/**
* Contains the assets, resources, manifest and resource symbols for an android_library dependency.
*
* <p>This class serves the role of both a processed MergedAndroidData and a dependency exported
* from another invocation of the AndroidResourcesProcessorAction. Since it's presumed to be cheaper
* to only pass the derived artifact (rTxt) rather that the entirety of the processed dependencies
* (png crunching and resource processing should be saved for the final
* AndroidResourcesProcessorAction invocation) AndroidData can have multiple roots for resources and
* assets.
*/
class DependencyAndroidData extends SerializedAndroidData {
// From the start of the line,
// 1) match any number of characters that isn't ":" until a ":" (twice for resources and assets)
// 2) match at least one character that isn't ":" until a ":" (twice for manifest and R.txt)
// 3) match anything that isn't ":" until end of line (symbols)
private static final Pattern VALID_REGEX = Pattern.compile("^[^:]*:[^:]*(:[^:]+){2,3}$");
public static final String EXPECTED_FORMAT =
"[resources[#resources]]:[assets[#assets]]:manifest:R.txt:symbols";
public static DependencyAndroidData valueOf(String text) {
return valueOf(text, FileSystems.getDefault());
}
@VisibleForTesting
static DependencyAndroidData valueOf(String text, FileSystem fileSystem) {
if (!VALID_REGEX.matcher(text).matches()) {
throw new IllegalArgumentException(text + " is not in the format '" + EXPECTED_FORMAT + "'");
}
String[] parts = text.split(":");
Path rTxt = exists(fileSystem.getPath(parts[3]));
ImmutableList<Path> assetDirs = splitPaths(parts[1], fileSystem);
Path manifest = exists(fileSystem.getPath(parts[2]));
Path symbols;
CompiledResources compiledSymbols;
if (parts.length == 5) {
// This is called "symbols.zip" in aapt2-land, or "merged.bin" in aapt1-land, with formats
// that are to be understood from context.
symbols = exists(fileSystem.getPath(parts[4]));
compiledSymbols = CompiledResources.from(symbols, manifest);
} else {
// Legacy code only used by aapt1 tests assuming the obsolete "android_resources" rule.
// No production code actually uses this.
symbols = null;
compiledSymbols = null;
}
return new DependencyAndroidData(
splitPaths(parts[0], fileSystem), assetDirs, manifest, rTxt, symbols, compiledSymbols);
}
private final Path manifest;
private final Path rTxt;
private final CompiledResources compiledSymbols;
public DependencyAndroidData(
ImmutableList<Path> resourceDirs,
ImmutableList<Path> assetDirs,
Path manifest,
Path rTxt,
Path symbols,
CompiledResources compiledSymbols) {
// Use the manifest as a label for now.
super(resourceDirs, assetDirs, manifest.toString(), symbols);
this.manifest = manifest;
this.rTxt = rTxt;
this.compiledSymbols = compiledSymbols;
}
public SymbolFileProvider asSymbolFileProvider() {
return new SymbolFileProvider() {
@Override
public File getManifest() {
return manifest.toFile();
}
@Override
public File getSymbolFile() {
return rTxt == null ? null : rTxt.toFile();
}
@Override
public boolean isOptional() {
return false;
}
@Override
public int hashCode() {
return Objects.hash(getManifest(), getSymbolFile());
}
@Override
public boolean equals(Object obj) {
if (obj instanceof SymbolFileProvider) {
SymbolFileProvider other = (SymbolFileProvider) obj;
return Objects.equals(getManifest(), other.getManifest())
&& Objects.equals(getSymbolFile(), other.getSymbolFile());
}
return false;
}
};
}
public CompiledResources getCompiledSymbols() {
return compiledSymbols;
}
@Override
public String toString() {
return String.format(
"AndroidData(%s, %s, %s, %s, %s)", resourceDirs, assetDirs, manifest, rTxt, symbols);
}
@Override
public int hashCode() {
return Objects.hash(resourceDirs, assetDirs, manifest, rTxt, symbols);
}
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (obj == null) {
return false;
}
if (!(obj instanceof DependencyAndroidData)) {
return false;
}
DependencyAndroidData other = (DependencyAndroidData) obj;
return Objects.equals(other.resourceDirs, resourceDirs)
&& Objects.equals(other.assetDirs, assetDirs)
&& Objects.equals(other.rTxt, rTxt)
&& Objects.equals(other.symbols, symbols)
&& Objects.equals(other.manifest, manifest);
}
}