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

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 com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import org.apache.commons.lang.StringUtils;
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.internal.project.HoldsProjectState;
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.Cast;
import org.gradle.internal.build.ExecutionResult;
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.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Set;
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.api.services.internal.BuildServiceProvider.asBuildServiceProvider;
import static org.gradle.internal.Cast.uncheckedCast;
import static org.gradle.internal.Cast.uncheckedNonnullCast;

public class DefaultBuildServicesRegistry implements BuildServiceRegistryInternal, HoldsProjectState {

    private final BuildIdentifier buildIdentifier;
    private final Lock registrationsLock = new ReentrantLock();
    private NamedDomainObjectSet> registrations;
    private final DomainObjectCollectionFactory collectionFactory;
    private final InstantiatorFactory instantiatorFactory;
    private final ServiceRegistry services;
    private final IsolatableFactory isolatableFactory;
    private final SharedResourceLeaseRegistry leaseRegistry;
    private final IsolationScheme, BuildServiceParameters> isolationScheme = new IsolationScheme<>(
        Cast.uncheckedCast(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 collectionFactory,
        InstantiatorFactory instantiatorFactory,
        ServiceRegistry services,
        ListenerManager listenerManager,
        IsolatableFactory isolatableFactory,
        SharedResourceLeaseRegistry leaseRegistry,
        BuildServiceProvider.Listener listener
    ) {
        this.buildIdentifier = buildIdentifier;
        this.registrations = uncheckedCast(collectionFactory.newNamedDomainObjectSet(BuildServiceRegistration.class));
        this.collectionFactory = collectionFactory;
        this.instantiatorFactory = instantiatorFactory;
        this.services = services;
        this.isolatableFactory = isolatableFactory;
        this.leaseRegistry = leaseRegistry;
        this.paramsInstantiator = instantiatorFactory.decorateScheme().withServices(services).instantiator();
        this.specInstantiator = instantiatorFactory.decorateLenientScheme().withServices(services).instantiator();
        this.listener = listener;
        listenerManager.addListener(new ServiceCleanupListener());
    }

    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(BuildServiceProvider service) {
        DefaultServiceRegistration registration = findRegistration(service.getType(), service.getName());
        if (registration == null) {
            // no corresponding service registered
            return null;
        }
        return registration.asSharedResource(() -> {
            // Prevent further changes to registration
            registration.getMaxParallelUsages().finalizeValue();
            int maxUsages = registration.getMaxParallelUsages().getOrElse(-1);

            if (maxUsages > 0) {
                leaseRegistry.registerSharedResource(registration.getName(), maxUsages);
            }
            return new ServiceBackedSharedResource(registration.getName(), maxUsages, leaseRegistry);
        });
    }

    @Nullable
    @Override
    public DefaultServiceRegistration findRegistration(Class type, String name) {
        return uncheckedCast(!name.isEmpty() ?
            findByName(name) :
            findByType(type)
        );
    }

    @Override
    public Set> findRegistrations(Class type, @Nullable String name) {
        return withRegistrations(registrations ->
            ImmutableSet.>builder().addAll(registrations.matching(it ->
                type.isAssignableFrom(BuildServiceProvider.getProvidedType(it.getService()))
                    &&
                (StringUtils.isEmpty(name) || it.getName().equals(name))
            )).build()
        );
    }

    @Override
    @Nullable
    public BuildServiceRegistration findByName(String name) {
        return withRegistrations(registrations -> registrations.findByName(name));
    }

    @Nullable
    @Override
    public BuildServiceRegistration findByType(Class type) {
        return findRegistrations(type, null).stream().findFirst().orElse(null);
    }

    @Override
    public , P extends BuildServiceParameters> Provider registerIfAbsent(String name, Class implementationType, Action> configureAction) {
        return doRegisterIfAbsent(name, implementationType, () -> {
            // 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); return spec; }); } @Override public BuildServiceProvider registerIfAbsent(String name, Class> implementationType, @Nullable BuildServiceParameters parameters, int maxUsages) { Supplier> buildServiceSpecSupplier = () -> { DefaultServiceSpec spec = uncheckedNonnullCast(specInstantiator.newInstance(DefaultServiceSpec.class, parameters)); spec.getMaxParallelUsages().set(maxUsages); return spec; }; return doRegisterIfAbsent(name, uncheckedNonnullCast(implementationType), uncheckedNonnullCast(buildServiceSpecSupplier)); } private , P extends BuildServiceParameters> BuildServiceProvider doRegisterIfAbsent(String name, Class implementationType, Supplier> specSupplier) { return withRegistrations(registrations -> { BuildServiceRegistration existing = registrations.findByName(name); if (existing != null) { // TODO - assert same type // TODO - assert same parameters return uncheckedNonnullCast(existing.getService()); } // TODO - finalize the parameters during isolation // TODO - need to lock the project during isolation - should do this the same way as artifact transforms BuildServiceSpec

spec = specSupplier.get(); return doRegister(name, implementationType, spec.getParameters(), spec.getMaxParallelUsages().getOrNull(), registrations); }); } @Override public List getSharedResources(Set>> services) { if (services.isEmpty()) { return Collections.emptyList(); } ImmutableList.Builder locks = ImmutableList.builder(); for (Provider> service : services) { if (!service.isPresent()) { continue; } SharedResource resource = forService(asBuildServiceProvider(service)); if (resource != null && resource.getMaxUsages() > 0) { locks.add(resource.getResourceLock()); } } return locks.build(); } @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> implementationType, @Nullable BuildServiceParameters parameters, int maxUsages) { return withRegistrations(registrations -> { DefaultServiceRegistration registration = Cast.uncheckedCast(registrations.findByName(name)); if (registration != null) { if (registration.provider.isKeepAlive()) { // Reuse the service instance return registration.provider; } throw new IllegalArgumentException(String.format("Service '%s' has already been registered.", name)); } return doRegister(name, uncheckedNonnullCast(implementationType), parameters, maxUsages <= 0 ? null : maxUsages, registrations); }); } @Override public BuildServiceProvider consume(String name, Class> implementationType) { return doConsume(name, uncheckedCast(implementationType)); } private > BuildServiceProvider doConsume(String name, Class implementationType) { return new ConsumedBuildServiceProvider<>(buildIdentifier, name, implementationType, services); } private , P extends BuildServiceParameters> BuildServiceProvider doRegister( String name, Class implementationType, @Nullable P parameters, @Nullable Integer maxParallelUsages, NamedDomainObjectSet> registrations ) { RegisteredBuildServiceProvider provider = new RegisteredBuildServiceProvider<>( buildIdentifier, name, implementationType, parameters, isolationScheme, instantiatorFactory.injectScheme(), isolatableFactory, services, listener, maxParallelUsages ); 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) return provider; } @Override public void discardAll() { discardAll(false); } private void discardAll(boolean forceAll) { withRegistrations(registrations -> { List> preserved = new ArrayList<>(); try { ExecutionResult.forEach(registrations, registration -> { DefaultServiceRegistration serviceRegistration = (DefaultServiceRegistration) registration; // Do not stop services that are to be retained beyond configuration time (e.g. build event listeners) if (forceAll || !serviceRegistration.provider.isKeepAlive()) { serviceRegistration.provider.maybeStop(); } else { preserved.add(serviceRegistration); } }).rethrow(); } finally { // Replace the entire container, rather than clear it, to discard all the service instances and because it may contain configuration actions and // other state that can affect the service instances when they are registered again this.registrations = uncheckedCast(collectionFactory.newNamedDomainObjectSet(BuildServiceRegistration.class)); } this.registrations.addAll(preserved); return null; }); } 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 RegisteredBuildServiceProvider provider; private SharedResource resourceWrapper; public DefaultServiceRegistration(String name, P parameters, RegisteredBuildServiceProvider 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 configureAction) { configureAction.execute(parameters); } } private class ServiceCleanupListener extends BuildAdapter { @SuppressWarnings("deprecation") @Override public void buildFinished(BuildResult result) { discardAll(true); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy