org.gradle.plugin.devel.plugins.JavaGradlePluginPlugin Maven / Gradle / Ivy
Show all versions of gradle-api Show documentation
/*
* Copyright 2014 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.plugins;
import com.google.common.collect.Sets;
import org.gradle.api.Action;
import org.gradle.api.NonNullApi;
import org.gradle.api.Plugin;
import org.gradle.api.Project;
import org.gradle.api.Task;
import org.gradle.api.artifacts.Configuration;
import org.gradle.api.artifacts.dsl.DependencyHandler;
import org.gradle.api.file.CopySpec;
import org.gradle.api.file.FileCollection;
import org.gradle.api.file.FileCopyDetails;
import org.gradle.api.internal.artifacts.ivyservice.projectmodule.ProjectPublicationRegistry;
import org.gradle.api.internal.plugins.PluginDescriptor;
import org.gradle.api.internal.project.ProjectInternal;
import org.gradle.api.logging.Logger;
import org.gradle.api.logging.Logging;
import org.gradle.api.plugins.AppliedPlugin;
import org.gradle.api.plugins.JavaBasePlugin;
import org.gradle.api.plugins.JavaPlugin;
import org.gradle.api.plugins.JavaPluginConvention;
import org.gradle.api.tasks.ClasspathNormalizer;
import org.gradle.api.tasks.Copy;
import org.gradle.api.tasks.SourceSet;
import org.gradle.api.tasks.TaskProvider;
import org.gradle.api.tasks.bundling.Jar;
import org.gradle.api.tasks.testing.Test;
import org.gradle.internal.Describables;
import org.gradle.internal.DisplayName;
import org.gradle.plugin.devel.GradlePluginDevelopmentExtension;
import org.gradle.plugin.devel.PluginDeclaration;
import org.gradle.plugin.devel.tasks.GeneratePluginDescriptors;
import org.gradle.plugin.devel.tasks.PluginUnderTestMetadata;
import org.gradle.plugin.devel.tasks.ValidateTaskProperties;
import org.gradle.plugin.use.PluginId;
import org.gradle.plugin.use.internal.DefaultPluginId;
import org.gradle.plugin.use.resolve.internal.local.PluginPublication;
import javax.annotation.Nullable;
import java.io.File;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.concurrent.Callable;
/**
* A plugin for building java gradle plugins. Automatically generates plugin descriptors. Emits warnings for common error conditions. Provides a direct integration with TestKit by declaring the
* {@code gradleTestKit()} dependency for the test compile configuration and a dependency on the plugin classpath manifest generation task for the test runtime configuration. Default conventions can
* be customized with the help of {@link GradlePluginDevelopmentExtension}.
*
* Integrates with the 'maven-publish' and 'ivy-publish' plugins to automatically publish the plugins so they can be resolved using the `pluginRepositories` and `plugins` DSL.
*/
@NonNullApi
public class JavaGradlePluginPlugin implements Plugin {
private static final Logger LOGGER = Logging.getLogger(JavaGradlePluginPlugin.class);
static final String COMPILE_CONFIGURATION = "compile";
static final String JAR_TASK = "jar";
static final String PROCESS_RESOURCES_TASK = "processResources";
static final String GRADLE_PLUGINS = "gradle-plugins";
static final String PLUGIN_DESCRIPTOR_PATTERN = "META-INF/" + GRADLE_PLUGINS + "/*.properties";
static final String CLASSES_PATTERN = "**/*.class";
static final String BAD_IMPL_CLASS_WARNING_MESSAGE = "%s: A valid plugin descriptor was found for %s but the implementation class %s was not found in the jar.";
static final String INVALID_DESCRIPTOR_WARNING_MESSAGE = "%s: A plugin descriptor was found for %s but it was invalid.";
static final String NO_DESCRIPTOR_WARNING_MESSAGE = "%s: No valid plugin descriptors were found in META-INF/" + GRADLE_PLUGINS + "";
static final String DECLARED_PLUGIN_MISSING_MESSAGE = "%s: Could not find plugin descriptor of %s at META-INF/" + GRADLE_PLUGINS + "/%s.properties";
static final String DECLARATION_MISSING_ID_MESSAGE = "Missing id for %s";
static final String DECLARATION_MISSING_IMPLEMENTATION_MESSAGE = "Missing implementationClass for %s";
static final String EXTENSION_NAME = "gradlePlugin";
static final String PLUGIN_UNDER_TEST_METADATA_TASK_NAME = "pluginUnderTestMetadata";
static final String GENERATE_PLUGIN_DESCRIPTORS_TASK_NAME = "pluginDescriptors";
static final String VALIDATE_TASK_PROPERTIES_TASK_NAME = "validateTaskProperties";
/**
* The task group used for tasks created by the Java Gradle plugin development plugin.
*
* @since 4.0
*/
static final String PLUGIN_DEVELOPMENT_GROUP = "Plugin development";
/**
* The description for the task generating metadata for plugin functional tests.
*
* @since 4.0
*/
static final String PLUGIN_UNDER_TEST_METADATA_TASK_DESCRIPTION = "Generates the metadata for plugin functional tests.";
/**
* The description for the task generating plugin descriptors from plugin declarations.
*
* @since 4.0
*/
static final String GENERATE_PLUGIN_DESCRIPTORS_TASK_DESCRIPTION = "Generates plugin descriptors from plugin declarations.";
/**
* The description for the task validating task property annotations for the plugin.
*
* @since 4.0
*/
static final String VALIDATE_TASK_PROPERTIES_TASK_DESCRIPTION = "Validates task property annotations for the plugin.";
public void apply(Project project) {
project.getPluginManager().apply(JavaPlugin.class);
applyDependencies(project);
GradlePluginDevelopmentExtension extension = createExtension(project);
configureJarTask(project, extension);
configureTestKit(project, extension);
configurePublishing(project);
registerPlugins(project, extension);
configureDescriptorGeneration(project, extension);
validatePluginDeclarations(project, extension);
configureTaskPropertiesValidation(project);
}
private void registerPlugins(Project project, GradlePluginDevelopmentExtension extension) {
ProjectInternal projectInternal = (ProjectInternal) project;
ProjectPublicationRegistry registry = projectInternal.getServices().get(ProjectPublicationRegistry.class);
extension.getPlugins().all(new Action() {
@Override
public void execute(PluginDeclaration pluginDeclaration) {
registry.registerPublication(projectInternal, new LocalPluginPublication(pluginDeclaration));
}
});
}
private void applyDependencies(Project project) {
DependencyHandler dependencies = project.getDependencies();
dependencies.add(COMPILE_CONFIGURATION, dependencies.gradleApi());
}
private void configureJarTask(Project project, final GradlePluginDevelopmentExtension extension) {
project.getTasks().named(JAR_TASK, Jar.class, new Action() {
@Override
public void execute(Jar jarTask) {
List descriptors = new ArrayList();
Set classList = new HashSet();
PluginDescriptorCollectorAction pluginDescriptorCollector = new PluginDescriptorCollectorAction(descriptors);
ClassManifestCollectorAction classManifestCollector = new ClassManifestCollectorAction(classList);
PluginValidationAction pluginValidationAction = new PluginValidationAction(extension.getPlugins(), descriptors, classList);
jarTask.filesMatching(PLUGIN_DESCRIPTOR_PATTERN, pluginDescriptorCollector);
jarTask.filesMatching(CLASSES_PATTERN, classManifestCollector);
jarTask.appendParallelSafeAction(pluginValidationAction);
}
});
}
private GradlePluginDevelopmentExtension createExtension(Project project) {
JavaPluginConvention javaConvention = project.getConvention().getPlugin(JavaPluginConvention.class);
SourceSet defaultPluginSourceSet = javaConvention.getSourceSets().getByName(SourceSet.MAIN_SOURCE_SET_NAME);
SourceSet defaultTestSourceSet = javaConvention.getSourceSets().getByName(SourceSet.TEST_SOURCE_SET_NAME);
return project.getExtensions().create(EXTENSION_NAME, GradlePluginDevelopmentExtension.class, project, defaultPluginSourceSet, defaultTestSourceSet);
}
private void configureTestKit(Project project, GradlePluginDevelopmentExtension extension) {
TaskProvider pluginUnderTestMetadataTask = createAndConfigurePluginUnderTestMetadataTask(project, extension);
establishTestKitAndPluginClasspathDependencies(project, extension, pluginUnderTestMetadataTask);
}
private TaskProvider createAndConfigurePluginUnderTestMetadataTask(final Project project, final GradlePluginDevelopmentExtension extension) {
return project.getTasks().register(PLUGIN_UNDER_TEST_METADATA_TASK_NAME, PluginUnderTestMetadata.class, new Action() {
@Override
public void execute(final PluginUnderTestMetadata pluginUnderTestMetadataTask) {
pluginUnderTestMetadataTask.setGroup(PLUGIN_DEVELOPMENT_GROUP);
pluginUnderTestMetadataTask.setDescription(PLUGIN_UNDER_TEST_METADATA_TASK_DESCRIPTION);
pluginUnderTestMetadataTask.getOutputDirectory().set(project.getLayout().getBuildDirectory().dir(pluginUnderTestMetadataTask.getName()));
pluginUnderTestMetadataTask.getPluginClasspath().from(new Callable