org.gradle.api.internal.initialization.DefaultScriptClassPathResolver Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
The newest version!
/*
* Copyright 2017 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.api.internal.initialization;
import org.gradle.api.Action;
import org.gradle.api.JavaVersion;
import org.gradle.api.artifacts.ArtifactCollection;
import org.gradle.api.artifacts.ArtifactView;
import org.gradle.api.artifacts.Configuration;
import org.gradle.api.artifacts.component.ComponentIdentifier;
import org.gradle.api.artifacts.component.ProjectComponentIdentifier;
import org.gradle.api.artifacts.dsl.DependencyHandler;
import org.gradle.api.attributes.Attribute;
import org.gradle.api.attributes.AttributeContainer;
import org.gradle.api.attributes.Bundling;
import org.gradle.api.attributes.Category;
import org.gradle.api.attributes.LibraryElements;
import org.gradle.api.attributes.Usage;
import org.gradle.api.attributes.java.TargetJvmVersion;
import org.gradle.api.attributes.plugin.GradlePluginApiVersion;
import org.gradle.api.file.FileCollection;
import org.gradle.api.internal.artifacts.dsl.DependencyHandlerInternal;
import org.gradle.api.internal.artifacts.dsl.dependencies.DependencyFactoryInternal.ClassPathNotation;
import org.gradle.api.internal.initialization.transform.registration.InstrumentationTransformRegisterer;
import org.gradle.api.internal.initialization.transform.services.CacheInstrumentationDataBuildService;
import org.gradle.api.internal.initialization.transform.services.CacheInstrumentationDataBuildService.ResolutionScope;
import org.gradle.api.internal.initialization.transform.utils.InstrumentationClasspathMerger;
import org.gradle.api.internal.initialization.transform.utils.InstrumentationClasspathMerger.FileType;
import org.gradle.api.internal.model.NamedObjectInstantiator;
import org.gradle.api.invocation.Gradle;
import org.gradle.internal.classpath.ClassPath;
import org.gradle.internal.classpath.TransformedClassPath;
import org.gradle.internal.component.local.model.OpaqueComponentIdentifier;
import org.gradle.internal.instrumentation.agent.AgentStatus;
import org.gradle.internal.instrumentation.reporting.MethodInterceptionReportCollector;
import org.gradle.internal.instrumentation.reporting.PropertyUpgradeReportConfig;
import org.gradle.internal.lazy.Lazy;
import org.gradle.internal.logging.util.Log4jBannedVersion;
import org.gradle.util.GradleVersion;
import java.io.File;
import java.util.Collections;
import java.util.EnumSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import static org.gradle.api.attributes.LibraryElements.LIBRARY_ELEMENTS_ATTRIBUTE;
import static org.gradle.api.internal.initialization.DefaultScriptClassPathResolver.InstrumentationPhase.ANALYZED_ARTIFACT;
import static org.gradle.api.internal.initialization.DefaultScriptClassPathResolver.InstrumentationPhase.INSTRUMENTED_AND_UPGRADED;
import static org.gradle.api.internal.initialization.DefaultScriptClassPathResolver.InstrumentationPhase.INSTRUMENTED_ONLY;
import static org.gradle.api.internal.initialization.DefaultScriptClassPathResolver.InstrumentationPhase.NOT_INSTRUMENTED;
import static org.gradle.api.internal.initialization.transform.utils.InstrumentationClasspathMerger.FileType.ARTIFACT;
import static org.gradle.api.internal.initialization.transform.utils.InstrumentationClasspathMerger.FileType.INTERCEPTED_METHODS_REPORT;
public class DefaultScriptClassPathResolver implements ScriptClassPathResolver {
private static final Set GRADLE_API_NOTATIONS = EnumSet.of(
ClassPathNotation.GRADLE_API,
ClassPathNotation.LOCAL_GROOVY
);
public enum InstrumentationPhase {
NOT_INSTRUMENTED("not-instrumented"),
ANALYZED_ARTIFACT("analyzed-artifact"),
MERGED_ARTIFACT_ANALYSIS("merged-artifact-analysis"),
INSTRUMENTED_AND_UPGRADED("instrumented-and-upgraded"),
INSTRUMENTED_ONLY("instrumented-only");
private final String value;
InstrumentationPhase(String value) {
this.value = value;
}
public String getValue() {
return value;
}
}
public static final Attribute INSTRUMENTED_ATTRIBUTE = Attribute.of("org.gradle.internal.instrumented", String.class);
private final NamedObjectInstantiator instantiator;
private final InstrumentationTransformRegisterer instrumentationTransformRegisterer;
private final PropertyUpgradeReportConfig propertyUpgradeReportConfig;
public DefaultScriptClassPathResolver(
NamedObjectInstantiator instantiator,
AgentStatus agentStatus,
Gradle gradle,
PropertyUpgradeReportConfig propertyUpgradeReportConfig
) {
this.instantiator = instantiator;
// Shared services must be provided lazily, otherwise they are instantiated too early and some cases can fail
this.instrumentationTransformRegisterer = new InstrumentationTransformRegisterer(
agentStatus,
propertyUpgradeReportConfig,
Lazy.atomic().of(gradle::getSharedServices)
);
this.propertyUpgradeReportConfig = propertyUpgradeReportConfig;
}
@Override
public ScriptClassPathResolutionContext prepareDependencyHandler(DependencyHandler dependencyHandler) {
((DependencyHandlerInternal) dependencyHandler).getDefaultArtifactAttributes()
.attribute(INSTRUMENTED_ATTRIBUTE, NOT_INSTRUMENTED.value);
// Register instrumentation pipelines
return instrumentationTransformRegisterer.registerTransforms(dependencyHandler);
}
@Override
public void prepareClassPath(Configuration configuration, ScriptClassPathResolutionContext resolutionContext) {
// should ideally reuse the `JvmPluginServices` but this code is too low level
// and this service is therefore not available!
AttributeContainer attributes = configuration.getAttributes();
attributes.attribute(Usage.USAGE_ATTRIBUTE, instantiator.named(Usage.class, Usage.JAVA_RUNTIME));
attributes.attribute(Category.CATEGORY_ATTRIBUTE, instantiator.named(Category.class, Category.LIBRARY));
attributes.attribute(LIBRARY_ELEMENTS_ATTRIBUTE, instantiator.named(LibraryElements.class, LibraryElements.JAR));
attributes.attribute(Bundling.BUNDLING_ATTRIBUTE, instantiator.named(Bundling.class, Bundling.EXTERNAL));
attributes.attribute(TargetJvmVersion.TARGET_JVM_VERSION_ATTRIBUTE, Integer.parseInt(JavaVersion.current().getMajorVersion()));
attributes.attribute(GradlePluginApiVersion.GRADLE_PLUGIN_API_VERSION_ATTRIBUTE, instantiator.named(GradlePluginApiVersion.class, GradleVersion.current().getVersion()));
DependencyHandler dependencyHandler = resolutionContext.getDependencyHandler();
configuration.getDependencyConstraints().add(dependencyHandler.getConstraints().create(Log4jBannedVersion.LOG4J2_CORE_COORDINATES, constraint -> constraint.version(version -> {
version.require(Log4jBannedVersion.LOG4J2_CORE_REQUIRED_VERSION);
version.reject(Log4jBannedVersion.LOG4J2_CORE_VULNERABLE_VERSION_RANGE);
})));
}
@Override
public ClassPath resolveClassPath(Configuration classpathConfiguration, ScriptClassPathResolutionContext resolutionContext) {
// We clear resolution scope from service after the resolution is done, so data is not reused between invocations.
long contextId = resolutionContext.getContextId();
CacheInstrumentationDataBuildService buildService = resolutionContext.getBuildService().get();
try (ResolutionScope resolutionScope = buildService.newResolutionScope(contextId)) {
ArtifactView originalDependencies = getOriginalDependencies(classpathConfiguration);
resolutionScope.setTypeHierarchyAnalysisResult(getAnalysisResult(classpathConfiguration));
resolutionScope.setOriginalClasspath(originalDependencies.getFiles());
ArtifactCollection instrumentedExternalDependencies = getInstrumentedExternalDependencies(classpathConfiguration);
ArtifactCollection instrumentedProjectDependencies = getInstrumentedProjectDependencies(classpathConfiguration);
Map> instrumentedClasspath = InstrumentationClasspathMerger.mergeToClasspath(
originalDependencies.getArtifacts(),
instrumentedExternalDependencies,
instrumentedProjectDependencies
);
MethodInterceptionReportCollector reportCollector = propertyUpgradeReportConfig.getReportCollector();
instrumentedClasspath.getOrDefault(INTERCEPTED_METHODS_REPORT, Collections.emptyList()).forEach(reportCollector::collect);
return TransformedClassPath.handleInstrumentingArtifactTransform(instrumentedClasspath.getOrDefault(ARTIFACT, Collections.emptyList()));
}
}
private FileCollection getAnalysisResult(Configuration classpathConfiguration) {
return classpathConfiguration.getIncoming().artifactView((Action super ArtifactView.ViewConfiguration>) config -> {
config.attributes(attributes -> {
attributes.attribute(INSTRUMENTED_ATTRIBUTE, ANALYZED_ARTIFACT.value);
attributes.attribute(LIBRARY_ELEMENTS_ATTRIBUTE, instantiator.named(LibraryElements.class, LibraryElements.CLASSES));
});
// We have to analyze external and project dependencies to get full hierarchies, since
// for example user could use dependency substitution to replace external dependency with project dependency.
config.componentFilter(componentId -> !isGradleApi(componentId));
}).getFiles();
}
private static ArtifactView getOriginalDependencies(Configuration classpathConfiguration) {
return classpathConfiguration.getIncoming().artifactView((Action super ArtifactView.ViewConfiguration>) config -> {
config.componentFilter(it -> !isGradleApi(it));
});
}
private static ArtifactCollection getInstrumentedExternalDependencies(Configuration classpathConfiguration) {
return classpathConfiguration.getIncoming().artifactView((Action super ArtifactView.ViewConfiguration>) config -> {
config.attributes(it -> it.attribute(INSTRUMENTED_ATTRIBUTE, INSTRUMENTED_AND_UPGRADED.value));
config.componentFilter(DefaultScriptClassPathResolver::isExternalDependency);
}).getArtifacts();
}
private static ArtifactCollection getInstrumentedProjectDependencies(Configuration classpathConfiguration) {
return classpathConfiguration.getIncoming().artifactView((Action super ArtifactView.ViewConfiguration>) config -> {
config.attributes(it -> it.attribute(INSTRUMENTED_ATTRIBUTE, INSTRUMENTED_ONLY.value));
config.componentFilter(DefaultScriptClassPathResolver::isProjectDependency);
}).getArtifacts();
}
private static boolean isGradleApi(ComponentIdentifier componentId) {
if (componentId instanceof OpaqueComponentIdentifier) {
ClassPathNotation classPathNotation = ((OpaqueComponentIdentifier) componentId).getClassPathNotation();
return DefaultScriptClassPathResolver.GRADLE_API_NOTATIONS.contains(classPathNotation);
}
return false;
}
private static boolean isProjectDependency(ComponentIdentifier componentId) {
if (componentId instanceof OpaqueComponentIdentifier) {
return ((OpaqueComponentIdentifier) componentId).getClassPathNotation() == ClassPathNotation.LOCAL_PROJECT_AS_OPAQUE_DEPENDENCY;
}
return componentId instanceof ProjectComponentIdentifier;
}
private static boolean isExternalDependency(ComponentIdentifier componentId) {
return !isGradleApi(componentId) && !isProjectDependency(componentId);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy