All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.plugin.devel.tasks.ValidateTaskProperties Maven / Gradle / Ivy

/*
 * Copyright 2016 the original author or authors.
 *
 * 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 org.gradle.plugin.devel.tasks;

import com.google.common.base.Charsets;
import com.google.common.base.Function;
import com.google.common.base.Joiner;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.io.Files;
import org.gradle.api.DefaultTask;
import org.gradle.api.GradleException;
import org.gradle.api.Incubating;
import org.gradle.api.InvalidUserDataException;
import org.gradle.api.Task;
import org.gradle.api.UncheckedIOException;
import org.gradle.api.file.FileCollection;
import org.gradle.api.file.FileVisitDetails;
import org.gradle.api.file.FileVisitor;
import org.gradle.api.internal.DocumentationRegistry;
import org.gradle.api.internal.file.collections.DirectoryFileTreeFactory;
import org.gradle.api.tasks.CacheableTask;
import org.gradle.api.tasks.Classpath;
import org.gradle.api.tasks.Input;
import org.gradle.api.tasks.InputDirectory;
import org.gradle.api.tasks.Optional;
import org.gradle.api.tasks.OutputFile;
import org.gradle.api.tasks.ParallelizableTask;
import org.gradle.api.tasks.PathSensitive;
import org.gradle.api.tasks.PathSensitivity;
import org.gradle.api.tasks.SkipWhenEmpty;
import org.gradle.api.tasks.TaskAction;
import org.gradle.api.tasks.TaskValidationException;
import org.gradle.api.tasks.VerificationTask;
import org.gradle.internal.Cast;
import org.gradle.internal.classloader.ClassLoaderFactory;
import org.gradle.internal.classloader.ClassLoaderUtils;
import org.gradle.internal.classpath.ClassPath;
import org.gradle.internal.classpath.DefaultClassPath;
import org.gradle.util.internal.Java9ClassReader;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.Opcodes;

import javax.inject.Inject;
import java.io.File;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;

/**
 * Validates task property annotations.
 *
 * 

* Task properties must be annotated with one of: *

* *
    *
  • * Properties taken into account during up-to-date checks: *
      *
    • * {@literal @}{@link org.gradle.api.tasks.Input}, * {@literal @}{@link org.gradle.api.tasks.Nested}, * {@literal @}{@link org.gradle.api.tasks.InputFile}, * {@literal @}{@link org.gradle.api.tasks.InputDirectory}, * {@literal @}{@link org.gradle.api.tasks.InputFiles} * to mark it as an input to the task. *
    • *
    • * {@literal @}{@link org.gradle.api.tasks.OutputFile}, * {@literal @}{@link org.gradle.api.tasks.OutputDirectory} * to mark it as an output of the task. *
    • *
    *
  • *
  • * Properties ignored during up-to-date checks: *
      *
    • {@literal @}{@link javax.inject.Inject} marks a Gradle service used by the task.
    • *
    • {@literal @}{@link org.gradle.api.tasks.Console Console} marks a property that only influences the console output of the task.
    • *
    • {@literal @}{@link org.gradle.api.tasks.Internal Internal} mark an internal property of the task.
    • *
    *
  • *
* * @since 3.0 */ @Incubating @ParallelizableTask @CacheableTask @SuppressWarnings("WeakerAccess") public class ValidateTaskProperties extends DefaultTask implements VerificationTask { private File classesDir; private FileCollection classpath; private Object outputFile; private boolean ignoreFailures; private boolean failOnWarning; @TaskAction public void validateTaskClasses() throws IOException { ClassLoader previousContextClassLoader = Thread.currentThread().getContextClassLoader(); ClassPath classPath = new DefaultClassPath(Iterables.concat(Collections.singleton(getClassesDir()), getClasspath())); ClassLoader classLoader = getClassLoaderFactory().createIsolatedClassLoader(classPath); Thread.currentThread().setContextClassLoader(classLoader); try { validateTaskClasses(classLoader); } finally { Thread.currentThread().setContextClassLoader(previousContextClassLoader); ClassLoaderUtils.tryClose(classLoader); } } private void validateTaskClasses(final ClassLoader classLoader) throws IOException { final Map taskValidationProblems = Maps.newTreeMap(); final Class taskInterface; final Method validatorMethod; try { taskInterface = classLoader.loadClass(Task.class.getName()); Class validatorClass = classLoader.loadClass("org.gradle.api.internal.project.taskfactory.TaskPropertyValidationAccess"); validatorMethod = validatorClass.getMethod("collectTaskValidationProblems", Class.class, Map.class); } catch (ClassNotFoundException e) { throw new RuntimeException(e); } catch (NoSuchMethodException e) { throw new RuntimeException(e); } getServices().get(DirectoryFileTreeFactory.class).create(getClassesDir()).visit(new FileVisitor() { @Override public void visitDir(FileVisitDetails dirDetails) { } @Override public void visitFile(FileVisitDetails fileDetails) { if (!fileDetails.getPath().endsWith(".class")) { return; } ClassReader reader; try { reader = new Java9ClassReader(Files.asByteSource(fileDetails.getFile()).read()); } catch (IOException e) { throw new UncheckedIOException(e); } List classNames = Lists.newArrayList(); reader.accept(new TaskNameCollectorVisitor(classNames), ClassReader.SKIP_CODE); for (String className : classNames) { Class clazz; try { clazz = classLoader.loadClass(className); } catch (IllegalAccessError e) { throw new GradleException("Could not load class: " + className, e); } catch (ClassNotFoundException e) { throw new GradleException("Could not load class: " + className, e); } catch (NoClassDefFoundError e) { throw new GradleException("Could not load class: " + className, e); } if (!Modifier.isPublic(clazz.getModifiers())) { continue; } if (Modifier.isAbstract(clazz.getModifiers())) { continue; } if (!taskInterface.isAssignableFrom(clazz)) { continue; } Class taskClass = Cast.uncheckedCast(clazz); try { validatorMethod.invoke(null, taskClass, taskValidationProblems); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (InvocationTargetException e) { throw new RuntimeException(e); } } } }); List problemMessages = toProblemMessages(taskValidationProblems); storeResults(problemMessages, getOutputFile()); communicateResult(problemMessages, taskValidationProblems.values().contains(Boolean.TRUE)); } private void storeResults(List problemMessages, File outputFile) throws IOException { if (outputFile != null) { //noinspection ResultOfMethodCallIgnored outputFile.createNewFile(); Files.asCharSink(outputFile, Charsets.UTF_8).write(Joiner.on('\n').join(problemMessages)); } } private void communicateResult(List problemMessages, boolean hasErrors) { if (problemMessages.isEmpty()) { getLogger().info("Task property validation finished without warnings."); } else { if (hasErrors || getFailOnWarning()) { if (getIgnoreFailures()) { getLogger().warn("Task property validation finished with errors. See {} for more information on how to annotate task properties.{}", getDocumentationRegistry().getDocumentationFor("more_about_tasks", "sec:task_input_output_annotations"), toMessageList(problemMessages)); } else { throw new TaskValidationException(String.format("Task property validation failed. See %s for more information on how to annotate task properties.", getDocumentationRegistry().getDocumentationFor("more_about_tasks", "sec:task_input_output_annotations")), toExceptionList(problemMessages)); } } else { getLogger().warn("Task property validation finished with warnings:{}", toMessageList(problemMessages)); } } } private static List toProblemMessages(Map problems) { ImmutableList.Builder builder = ImmutableList.builder(); for (Map.Entry entry : problems.entrySet()) { String problem = entry.getKey(); Boolean error = entry.getValue(); builder.add(String.format("%s: %s", Boolean.TRUE.equals(error) ? "Error" : "Warning", problem )); } return builder.build(); } private static CharSequence toMessageList(List problemMessages) { StringBuilder builder = new StringBuilder(); for (String problemMessage : problemMessages) { builder.append(String.format("%n - %s", problemMessage)); } return builder; } private static List toExceptionList(List problemMessages) { return Lists.transform(problemMessages, new Function() { @Override public InvalidUserDataException apply(String problemMessage) { return new InvalidUserDataException(problemMessage); } }); } /** * {@inheritDoc} */ @Input @Override public boolean getIgnoreFailures() { return ignoreFailures; } /** * {@inheritDoc} */ @Override public void setIgnoreFailures(boolean ignoreFailures) { this.ignoreFailures = ignoreFailures; } /** * The directory containing the classes to validate. */ @PathSensitive(PathSensitivity.RELATIVE) @InputDirectory @SkipWhenEmpty public File getClassesDir() { return classesDir; } /** * Sets the directory containing the classes to validate. */ public void setClassesDir(File classesDir) { this.classesDir = classesDir; } /** * The classpath used to load the classes under validation. */ @Classpath public FileCollection getClasspath() { return classpath; } /** * Sets the classpath used to load the classes under validation. */ public void setClasspath(FileCollection classpath) { this.classpath = classpath; } /** * Returns whether the build should break when the verifications performed by this task detects a warning. */ @Input public boolean getFailOnWarning() { return failOnWarning; } /** * Returns the output file to store the report in. */ @Optional @OutputFile public File getOutputFile() { return outputFile == null ? null : getProject().file(outputFile); } /** * Sets the output file to store the report in. */ public void setOutputFile(Object outputFile) { this.outputFile = outputFile; } /** * Specifies whether the build should break when the verifications performed by this task detects a warning. * * @param failOnWarning {@code true} to break the build on warning, {@code false} to ignore warnings. The default is {@code false}. */ @SuppressWarnings("unused") public void setFailOnWarning(boolean failOnWarning) { this.failOnWarning = failOnWarning; } @Inject protected ClassLoaderFactory getClassLoaderFactory() { throw new UnsupportedOperationException(); } @Inject protected DocumentationRegistry getDocumentationRegistry() { throw new UnsupportedOperationException(); } private static class TaskNameCollectorVisitor extends ClassVisitor { private final Collection classNames; public TaskNameCollectorVisitor(Collection classNames) { super(Opcodes.ASM5); this.classNames = classNames; } @Override public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) { if ((access & Opcodes.ACC_PUBLIC) != 0) { classNames.add(name.replace('/', '.')); } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy