org.gradle.api.services.internal.DefaultBuildServicesRegistry Maven / Gradle / Ivy
/*
* Copyright 2019 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.services.internal;
import org.gradle.BuildAdapter;
import org.gradle.BuildResult;
import org.gradle.api.Action;
import org.gradle.api.NamedDomainObjectSet;
import org.gradle.api.NonExtensible;
import org.gradle.api.artifacts.component.BuildIdentifier;
import org.gradle.api.internal.collections.DomainObjectCollectionFactory;
import org.gradle.api.provider.Provider;
import org.gradle.api.services.BuildService;
import org.gradle.api.services.BuildServiceParameters;
import org.gradle.api.services.BuildServiceRegistration;
import org.gradle.api.services.BuildServiceSpec;
import org.gradle.internal.event.ListenerManager;
import org.gradle.internal.instantiation.InstantiatorFactory;
import org.gradle.internal.isolated.IsolationScheme;
import org.gradle.internal.isolation.IsolatableFactory;
import org.gradle.internal.reflect.Instantiator;
import org.gradle.internal.resources.ResourceLock;
import org.gradle.internal.resources.SharedResource;
import org.gradle.internal.resources.SharedResourceLeaseRegistry;
import org.gradle.internal.service.ServiceRegistry;
import javax.annotation.Nullable;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.function.Function;
import java.util.function.Supplier;
import static org.gradle.internal.Cast.uncheckedCast;
import static org.gradle.internal.Cast.uncheckedNonnullCast;
public class DefaultBuildServicesRegistry implements BuildServiceRegistryInternal {
private final BuildIdentifier buildIdentifier;
private final NamedDomainObjectSet> registrations;
private final Lock registrationsLock = new ReentrantLock();
private final InstantiatorFactory instantiatorFactory;
private final ServiceRegistry services;
private final ListenerManager listenerManager;
private final IsolatableFactory isolatableFactory;
private final SharedResourceLeaseRegistry leaseRegistry;
private final IsolationScheme isolationScheme = new IsolationScheme<>(BuildService.class, BuildServiceParameters.class, BuildServiceParameters.None.class);
private final Instantiator paramsInstantiator;
private final Instantiator specInstantiator;
private final BuildServiceProvider.Listener listener;
public DefaultBuildServicesRegistry(
BuildIdentifier buildIdentifier,
DomainObjectCollectionFactory factory,
InstantiatorFactory instantiatorFactory,
ServiceRegistry services,
ListenerManager listenerManager,
IsolatableFactory isolatableFactory,
SharedResourceLeaseRegistry leaseRegistry,
BuildServiceProvider.Listener listener
) {
this.buildIdentifier = buildIdentifier;
this.registrations = uncheckedCast(factory.newNamedDomainObjectSet(BuildServiceRegistration.class));
this.instantiatorFactory = instantiatorFactory;
this.services = services;
this.listenerManager = listenerManager;
this.isolatableFactory = isolatableFactory;
this.leaseRegistry = leaseRegistry;
this.paramsInstantiator = instantiatorFactory.decorateScheme().withServices(services).instantiator();
this.specInstantiator = instantiatorFactory.decorateLenientScheme().withServices(services).instantiator();
this.listener = listener;
}
private U withRegistrations(Function>, U> function) {
registrationsLock.lock();
try {
return function.apply(registrations);
} finally {
registrationsLock.unlock();
}
}
@Override
public NamedDomainObjectSet> getRegistrations() {
return registrations;
}
@Override
public SharedResource forService(Provider extends BuildService>> service) {
if (!(service instanceof BuildServiceProvider)) {
throw new IllegalArgumentException("The given provider is not a build service provider.");
}
BuildServiceProvider, ?> provider = (BuildServiceProvider, ?>) service;
DefaultServiceRegistration, ?> registration = getByName(provider.getName());
return registration.asSharedResource(() -> {
// Prevent further changes to registration
registration.getMaxParallelUsages().finalizeValue();
int maxUsages = registration.getMaxParallelUsages().getOrElse(-1);
if (maxUsages > 0) {
leaseRegistry.registerSharedResource(provider.getName(), maxUsages);
}
return new ServiceBackedSharedResource(provider.getName(), maxUsages, leaseRegistry);
});
}
private DefaultServiceRegistration, ?> getByName(String name) {
return (DefaultServiceRegistration, ?>) withRegistrations(registrations -> registrations.getByName(name));
}
@Override
public , P extends BuildServiceParameters> Provider registerIfAbsent(String name, Class implementationType, Action super BuildServiceSpec> configureAction) {
return withRegistrations(registrations -> {
BuildServiceRegistration, ?> existing = registrations.findByName(name);
if (existing != null) {
// TODO - assert same type
// TODO - assert same parameters
return uncheckedNonnullCast(existing.getService());
}
// TODO - extract some shared infrastructure to take care of parameter instantiation (eg strict vs lenient, which services are visible)
P parameters = instantiateParametersOf(implementationType);
// TODO - should defer execution of the action, to match behaviour for other container `register()` methods.
DefaultServiceSpec
spec = uncheckedNonnullCast(specInstantiator.newInstance(DefaultServiceSpec.class, parameters));
configureAction.execute(spec);
Integer maxParallelUsages = spec.getMaxParallelUsages().getOrNull();
// TODO - finalize the parameters during isolation
// TODO - need to lock the project during isolation - should do this the same way as artifact transforms
return doRegister(name, implementationType, parameters, maxParallelUsages, registrations);
});
}
@Nullable
private , P extends BuildServiceParameters> P instantiateParametersOf(Class implementationType) {
Class parameterType = isolationScheme.parameterTypeFor(implementationType);
return parameterType != null
? paramsInstantiator.newInstance(parameterType)
: null;
}
@Override
public BuildServiceProvider, ?> register(String name, Class extends BuildService>> implementationType, @Nullable BuildServiceParameters parameters, int maxUsages) {
return withRegistrations(registrations -> {
if (registrations.findByName(name) != null) {
throw new IllegalArgumentException(String.format("Service '%s' has already been registered.", name));
}
return doRegister(name, uncheckedNonnullCast(implementationType), parameters, maxUsages <= 0 ? null : maxUsages, registrations);
});
}
private , P extends BuildServiceParameters> BuildServiceProvider doRegister(
String name,
Class implementationType,
@Nullable P parameters,
Integer maxParallelUsages,
NamedDomainObjectSet> registrations
) {
BuildServiceProvider provider = new BuildServiceProvider<>(
buildIdentifier,
name,
implementationType,
parameters,
isolationScheme,
instantiatorFactory.injectScheme(),
isolatableFactory,
services,
listener
);
DefaultServiceRegistration registration = uncheckedNonnullCast(specInstantiator.newInstance(DefaultServiceRegistration.class, name, parameters, provider));
registration.getMaxParallelUsages().set(maxParallelUsages);
registrations.add(registration);
// TODO - should stop the service after last usage (ie after the last task that uses it) instead of at the end of the build
// TODO - should reuse service across build invocations, until the parameters change (which contradicts the previous item)
listenerManager.addListener(new ServiceCleanupListener(provider));
return provider;
}
private static class ServiceBackedSharedResource implements SharedResource {
private final String name;
private final int maxUsages;
private final SharedResourceLeaseRegistry leaseRegistry;
public ServiceBackedSharedResource(String name, int maxUsages, SharedResourceLeaseRegistry leaseRegistry) {
this.name = name;
this.maxUsages = maxUsages;
this.leaseRegistry = leaseRegistry;
}
@Override
public int getMaxUsages() {
return maxUsages;
}
@Override
public ResourceLock getResourceLock() {
return leaseRegistry.getResourceLock(name);
}
}
public static abstract class DefaultServiceRegistration, P extends BuildServiceParameters> implements BuildServiceRegistration {
private final String name;
private final P parameters;
private final BuildServiceProvider provider;
private SharedResource resourceWrapper;
public DefaultServiceRegistration(String name, P parameters, BuildServiceProvider provider) {
this.name = name;
this.parameters = parameters;
this.provider = provider;
}
@Override
public String getName() {
return name;
}
@Override
public P getParameters() {
return parameters;
}
@Override
public Provider getService() {
return provider;
}
public SharedResource asSharedResource(Supplier factory) {
if (resourceWrapper == null) {
resourceWrapper = factory.get();
}
return resourceWrapper;
}
}
@NonExtensible
public abstract static class DefaultServiceSpec implements BuildServiceSpec
{
private final P parameters;
public DefaultServiceSpec(P parameters) {
this.parameters = parameters;
}
@Override
public P getParameters() {
return parameters;
}
@Override
public void parameters(Action super P> configureAction) {
configureAction.execute(parameters);
}
}
private static class ServiceCleanupListener extends BuildAdapter {
private final BuildServiceProvider, ?> provider;
ServiceCleanupListener(BuildServiceProvider, ?> provider) {
this.provider = provider;
}
@SuppressWarnings("deprecation")
@Override
public void buildFinished(BuildResult result) {
provider.maybeStop();
}
}
}