com.android.build.gradle.integration.common.truth.NativeAndroidProjectSubject.java Source code

Java tutorial

Introduction

Here is the source code for com.android.build.gradle.integration.common.truth.NativeAndroidProjectSubject.java

Source

/*
 * Copyright (C) 2016 The Android Open Source Project
 *
 * 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.android.build.gradle.integration.common.truth;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Preconditions.checkState;

import com.android.annotations.NonNull;
import com.android.builder.model.NativeAndroidProject;
import com.android.builder.model.NativeArtifact;
import com.android.utils.FileUtils;
import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.Lists;
import com.google.common.collect.Multimap;
import com.google.common.collect.Sets;
import com.google.common.truth.FailureStrategy;
import com.google.common.truth.Subject;
import com.google.common.truth.SubjectFactory;

import org.gradle.api.Nullable;

import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;

/**
 * A Truth subject for testing NativeAndroidProject
 */
public class NativeAndroidProjectSubject extends Subject<NativeAndroidProjectSubject, NativeAndroidProject> {
    static class Factory extends SubjectFactory<NativeAndroidProjectSubject, NativeAndroidProject> {

        @NonNull
        public static NativeAndroidProjectSubject.Factory get() {
            return new NativeAndroidProjectSubject.Factory();
        }

        private Factory() {
        }

        @NonNull
        @Override
        public NativeAndroidProjectSubject getSubject(@NonNull FailureStrategy failureStrategy,
                @NonNull NativeAndroidProject subject) {
            return new NativeAndroidProjectSubject(failureStrategy, subject);
        }
    }

    private NativeAndroidProjectSubject(@NonNull FailureStrategy failureStrategy,
            @NonNull NativeAndroidProject subject) {
        super(failureStrategy, subject);
    }

    @NonNull
    private Multimap<String, NativeArtifact> getArtifactsGroupedByGroupName() {
        Multimap<String, NativeArtifact> groupToArtifacts = ArrayListMultimap.create();
        for (NativeArtifact artifact : getSubject().getArtifacts()) {
            groupToArtifacts.put(artifact.getGroupName(), artifact);
        }
        return groupToArtifacts;
    }

    @NonNull
    private Multimap<String, NativeArtifact> getArtifactsByName() {
        Multimap<String, NativeArtifact> groupToArtifacts = ArrayListMultimap.create();
        for (NativeArtifact artifact : getSubject().getArtifacts()) {
            groupToArtifacts.put(artifact.getName(), artifact);
        }
        return groupToArtifacts;
    }

    @NonNull
    private List<File> getOutputs() {
        List<File> outputs = Lists.newArrayList();
        for (NativeArtifact artifact : getSubject().getArtifacts()) {
            outputs.add(artifact.getOutputFile());
        }
        return outputs;
    }

    /**
     * Given an artifact output file, find the root project directory.
     * @param outputFile the output artifact file to trim down to project directory.
     */
    @NonNull
    private File getProjectFolderOfOutputFile(@NonNull File outputFile) {
        checkArgument(outputFile.getAbsolutePath().toString().contains("project"), outputFile);
        File original = outputFile;
        while (!outputFile.getName().equals("project")) {
            outputFile = outputFile.getParentFile();
            checkNotNull(outputFile, original);
        }
        return outputFile;
    }

    /**
     * Find a folder like project/.externalNativeBuild or project/build.
     * @param baseFolder is like .externalNativeBuild or build.
     * @return null if there are no output files.
     */
    @Nullable
    private File getIntermediatesFolder(@NonNull String baseFolder) {
        Set<File> intermediatesFolders = Sets.newHashSet();
        for (NativeArtifact artifact : getSubject().getArtifacts()) {
            intermediatesFolders
                    .add(FileUtils.join(getProjectFolderOfOutputFile(artifact.getOutputFile()), baseFolder));
        }
        // Ensure there are no intermediates folder or they are all the same.
        checkState(intermediatesFolders.size() <= 1, intermediatesFolders);
        return intermediatesFolders.size() == 0 ? null : intermediatesFolders.iterator().next();
    }

    /**
     * Get all files under project/.externalNativeBuild (or project/build).
     * @param baseFolder is like .externalNativeBuild or build.
     */
    @NonNull
    private Set<Path> getIntermediateFiles(String baseFolder) throws IOException {
        File intermediatesFolder = getIntermediatesFolder(baseFolder);
        Set<Path> intermediates = Sets.newHashSet();
        if (intermediatesFolder == null || !intermediatesFolder.exists()) {
            // No intermediate files, return empty;
            return intermediates;
        }

        Path intermediatesPath = Paths.get(intermediatesFolder.getPath());
        intermediates.addAll(Files.find(intermediatesPath, 12, // Recursion depth
                (path, attributes) -> attributes.isRegularFile()).collect(Collectors.toSet()));
        return intermediates;
    }

    /**
     * Get the set of base file names of all files in project/.externalNativeBuild or project/build.
     * @param baseFolder is like .externalNativeBuild or build.
     */
    @NonNull
    private Set<String> getIntermediatesNames(String extension, String baseFolder) throws IOException {
        Set<Path> intermediates = getIntermediateFiles(baseFolder);
        Set<String> names = Sets.newHashSet();
        for (Path intermediate : intermediates) {
            if (intermediate.getFileName().toString().endsWith(extension)) {
                names.add(intermediate.getFileName().toString());
            }
        }
        return names;
    }

    @SuppressWarnings("NonBooleanMethodNameMayNotStartWithQuestion")
    private void hasExactOutputFiles(String extension, String baseFolder, String... baseName) throws IOException {
        Set<String> intermediateNames = getIntermediatesNames(extension, baseFolder);
        Set<String> expected = Sets.newHashSet(baseName);
        Set<String> expectedNotFound = Sets.newHashSet();
        expectedNotFound.addAll(expected);
        expectedNotFound.removeAll(intermediateNames);
        if (!expectedNotFound.isEmpty()) {
            failWithRawMessage("Not true that %s build intermediates was %s. Set %s was missing %s",
                    getIntermediatesFolder(baseFolder), expected, intermediateNames, expectedNotFound);
        }

        Set<String> foundNotExpected = Sets.newHashSet();
        foundNotExpected.addAll(intermediateNames);
        foundNotExpected.removeAll(expected);
        if (!foundNotExpected.isEmpty()) {
            failWithRawMessage("Not true that %s build intermediates was %s. It had extras %s",
                    getIntermediatesFolder(baseFolder), expected, foundNotExpected);
        }
    }

    @SuppressWarnings("NonBooleanMethodNameMayNotStartWithQuestion")
    public void hasExactObjectFilesInBuildFolder(String... baseName) throws IOException {
        hasExactOutputFiles(".o", "build", baseName);
    }

    @SuppressWarnings("NonBooleanMethodNameMayNotStartWithQuestion")
    public void hasExactObjectFilesInExternalNativeBuildFolder(String... baseName) throws IOException {
        hasExactOutputFiles(".o", ".externalNativeBuild", baseName);
    }

    @SuppressWarnings("NonBooleanMethodNameMayNotStartWithQuestion")
    public void hasExactSharedObjectFilesInBuildFolder(String... baseName) throws IOException {
        hasExactOutputFiles(".so", "build", baseName);
    }

    @SuppressWarnings("NonBooleanMethodNameMayNotStartWithQuestion")
    public void hasBuildOutputCountEqualTo(int expectedCount) {
        List<File> buildOutputs = getOutputs();

        if (buildOutputs.size() != expectedCount) {
            failWithRawMessage("Not true that %s build output count was %s. It was %s", getDisplaySubject(),
                    expectedCount, buildOutputs.size());
        }
    }

    public void allBuildOutputsExist() {
        List<File> exist = Lists.newArrayList();
        List<File> dontExist = Lists.newArrayList();
        for (File buildOutput : getOutputs()) {
            if (!buildOutput.isFile()) {
                dontExist.add(buildOutput);
            } else {
                exist.add(buildOutput);
            }
        }
        if (!dontExist.isEmpty()) {
            failWithRawMessage("Not true that %s build outputs <%s> exist. Existing build outputs are <%s>",
                    getDisplaySubject(), dontExist, exist);
        }
    }

    public void noBuildOutputsExist() {
        List<File> exist = Lists.newArrayList();
        List<File> dontExist = Lists.newArrayList();
        for (File buildOutput : getOutputs()) {
            if (!buildOutput.isFile()) {
                dontExist.add(buildOutput);
            } else {
                exist.add(buildOutput);
            }
        }
        if (!exist.isEmpty()) {
            failWithRawMessage(
                    "Not true that %s build outputs <%s> don't exist. Nonexistent build outputs are <%s>",
                    getDisplaySubject(), exist, dontExist);
        }
    }

    @SuppressWarnings("NonBooleanMethodNameMayNotStartWithQuestion")
    public void hasArtifactGroupsNamed(String... artifacts) {
        Set<String> expected = Sets.newHashSet(artifacts);
        Multimap<String, NativeArtifact> groups = getArtifactsGroupedByGroupName();
        if (!groups.keySet().equals(expected)) {
            failWithRawMessage("Not true that %s artifact groups are <%s>. They are <%s>", getDisplaySubject(),
                    expected, groups.keySet());
        }
    }

    @SuppressWarnings("NonBooleanMethodNameMayNotStartWithQuestion")
    public void hasTargetsNamed(String... artifacts) {
        Set<String> expected = Sets.newHashSet(artifacts);
        Multimap<String, NativeArtifact> groups = getArtifactsByName();
        if (!groups.keySet().equals(expected)) {
            failWithRawMessage("Not true that %s that qualified targets are <%s>. They are <%s>",
                    getDisplaySubject(), expected, groups.keySet());
        }
    }

    @SuppressWarnings("NonBooleanMethodNameMayNotStartWithQuestion")
    public void hasArtifactGroupsOfSize(long size) {
        Multimap<String, NativeArtifact> groups = getArtifactsGroupedByGroupName();
        for (String groupName : groups.keySet()) {
            if (groups.get(groupName).size() != size) {
                failWithRawMessage("Not true that %s artifact group %s has size %s. " + "Actual size is <%s>",
                        getDisplaySubject(), groupName, size, groups.get(groupName).size());
            }
        }
    }
}