org.gradle.api.services.internal.BuildServiceProvider Maven / Gradle / Ivy
Show all versions of gradle-api Show documentation
/*
* 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.api.internal.provider.AbstractReadOnlyProvider;
import org.gradle.api.services.BuildService;
import org.gradle.api.services.BuildServiceParameters;
import org.gradle.internal.Try;
import org.gradle.internal.instantiation.InstantiationScheme;
import org.gradle.internal.isolation.IsolatableFactory;
import org.gradle.internal.service.DefaultServiceRegistry;
import org.gradle.internal.state.Managed;
import javax.annotation.Nullable;
// TODO - complain when used at configuration time, except when opted in to this
public class BuildServiceProvider, P extends BuildServiceParameters> extends AbstractReadOnlyProvider implements Managed {
private final String name;
private final Class implementationType;
private final InstantiationScheme instantiationScheme;
private final IsolatableFactory isolatableFactory;
private final Class parametersType;
private final P parameters;
private Try instance;
public BuildServiceProvider(String name, Class implementationType, Class parametersType, @Nullable P parameters, InstantiationScheme instantiationScheme, IsolatableFactory isolatableFactory) {
this.name = name;
this.implementationType = implementationType;
this.parametersType = parametersType;
this.parameters = parameters;
this.instantiationScheme = instantiationScheme;
this.isolatableFactory = isolatableFactory;
}
public String getName() {
return name;
}
public Class getImplementationType() {
return implementationType;
}
@Nullable
public P getParameters() {
return parameters;
}
@Nullable
@Override
public Class getType() {
return implementationType;
}
@Override
public boolean isPresent() {
return true;
}
@Override
public boolean immutable() {
return true;
}
@Override
public Object unpackState() {
throw new UnsupportedOperationException("Build services cannot be serialized.");
}
// TODO - rename this method
@Override
public boolean isValueProducedByTask() {
return true;
}
@Nullable
@Override
public T getOrNull() {
synchronized (this) {
if (instance == null) {
// TODO - extract a ServiceLookup implementation to reuse
// TODO - extract some shared infrastructure to take care of instantiaton (eg which services are visible, strict vs lenient, decorated or not?)
DefaultServiceRegistry services = new DefaultServiceRegistry();
// TODO - should hold the project lock to do the isolation. Should work the same way as artifact transforms (a work node does the isolation, etc)
P isolatedParameters = isolatableFactory.isolate(parameters).isolate();
if (isolatedParameters != null) {
services.add(parametersType, isolatedParameters);
}
try {
instance = Try.successful(instantiationScheme.withServices(services).instantiator().newInstance(implementationType));
} catch (Exception e) {
instance = Try.failure(new ServiceLifecycleException("Failed to create service '" + name + "'.", e));
}
}
return instance.get();
}
}
public void maybeStop() {
synchronized (this) {
if (instance != null) {
instance.ifSuccessful(t -> {
if (t instanceof AutoCloseable) {
try {
((AutoCloseable) t).close();
} catch (Exception e) {
throw new ServiceLifecycleException("Failed to stop service '" + name + "'.", e);
}
}
});
}
}
}
}