| // 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); |
| } |
| } |