org.gradle.internal.nativeintegration.services.NativeServices 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.
/*
* Copyright 2011 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.internal.nativeintegration.services;
import net.rubygrapefruit.platform.NativeException;
import net.rubygrapefruit.platform.NativeIntegrationUnavailableException;
import net.rubygrapefruit.platform.PosixFiles;
import net.rubygrapefruit.platform.Process;
import net.rubygrapefruit.platform.ProcessLauncher;
import net.rubygrapefruit.platform.SystemInfo;
import net.rubygrapefruit.platform.Terminals;
import net.rubygrapefruit.platform.WindowsRegistry;
import net.rubygrapefruit.platform.internal.DefaultProcessLauncher;
import org.gradle.internal.SystemProperties;
import org.gradle.internal.jvm.Jvm;
import org.gradle.internal.nativeintegration.ProcessEnvironment;
import org.gradle.internal.nativeintegration.console.ConsoleDetector;
import org.gradle.internal.nativeintegration.console.NativePlatformConsoleDetector;
import org.gradle.internal.nativeintegration.console.NoOpConsoleDetector;
import org.gradle.internal.nativeintegration.console.WindowsConsoleDetector;
import org.gradle.internal.nativeintegration.filesystem.services.FileSystemServices;
import org.gradle.internal.nativeintegration.filesystem.services.UnavailablePosixFiles;
import org.gradle.internal.nativeintegration.jna.UnsupportedEnvironment;
import org.gradle.internal.nativeintegration.processenvironment.NativePlatformBackedProcessEnvironment;
import org.gradle.internal.os.OperatingSystem;
import org.gradle.internal.service.DefaultServiceRegistry;
import org.gradle.internal.service.ServiceRegistry;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* Provides various native platform integration services.
*/
public class NativeServices extends DefaultServiceRegistry implements ServiceRegistry {
private static final Logger LOGGER = LoggerFactory.getLogger(NativeServices.class);
private static boolean useNativePlatform = "true".equalsIgnoreCase(System.getProperty("org.gradle.native", "true"));
private static final NativeServices INSTANCE = new NativeServices();
private static boolean initialized;
private static File nativeBaseDir;
public static final String NATIVE_DIR_OVERRIDE = "org.gradle.native.dir";
/**
* Initializes the native services to use the given user home directory to store native libs and other resources. Does nothing if already initialized. Will be implicitly initialized on first usage
* of a native service. Also initializes the Native-Platform library using the given user home directory.
*/
public static synchronized void initialize(File userHomeDir) {
if (!initialized) {
nativeBaseDir = getNativeServicesDir(userHomeDir);
if (useNativePlatform) {
try {
net.rubygrapefruit.platform.Native.init(nativeBaseDir);
} catch (NativeIntegrationUnavailableException ex) {
LOGGER.debug("Native-platform is not available.");
useNativePlatform = false;
} catch (NativeException ex) {
if (ex.getCause() instanceof UnsatisfiedLinkError && ex.getCause().getMessage().toLowerCase().contains("already loaded in another classloader")) {
LOGGER.debug("Unable to initialize native-platform. Failure: {}", format(ex));
useNativePlatform = false;
} else {
throw ex;
}
}
}
initialized = true;
LOGGER.info("Initialized native services in: " + nativeBaseDir);
}
}
public static File getNativeServicesDir(File userHomeDir) {
String overrideProperty = getNativeDirOverride();
if (overrideProperty == null) {
return new File(userHomeDir, "native");
} else {
return new File(overrideProperty);
}
}
private static String getNativeDirOverride() {
return System.getProperty(NATIVE_DIR_OVERRIDE, System.getenv(NATIVE_DIR_OVERRIDE));
}
public static synchronized NativeServices getInstance() {
if (!initialized) {
// If this occurs while running gradle or running integration tests, it is indicative of a problem.
// If this occurs while running unit tests, then either use the NativeServicesTestFixture or the '@UsesNativeServices' annotation.
throw new IllegalStateException("Cannot get an instance of NativeServices without first calling initialize().");
}
return INSTANCE;
}
private NativeServices() {
addProvider(new FileSystemServices());
}
@Override
public void close() {
// Don't close
}
protected OperatingSystem createOperatingSystem() {
return OperatingSystem.current();
}
protected Jvm createJvm() {
return Jvm.current();
}
protected ProcessEnvironment createProcessEnvironment(OperatingSystem operatingSystem) {
if (useNativePlatform) {
try {
net.rubygrapefruit.platform.Process process = net.rubygrapefruit.platform.Native.get(Process.class);
return new NativePlatformBackedProcessEnvironment(process);
} catch (NativeIntegrationUnavailableException ex) {
LOGGER.debug("Native-platform process integration is not available. Continuing with fallback.");
}
}
return new UnsupportedEnvironment();
}
protected ConsoleDetector createConsoleDetector(OperatingSystem operatingSystem) {
if (useNativePlatform) {
try {
Terminals terminals = net.rubygrapefruit.platform.Native.get(Terminals.class);
return new NativePlatformConsoleDetector(terminals);
} catch (NativeIntegrationUnavailableException ex) {
LOGGER.debug("Native-platform terminal integration is not available. Continuing with fallback.");
} catch (NativeException ex) {
LOGGER.debug("Unable to load from native-platform backed ConsoleDetector. Continuing with fallback. Failure: {}", format(ex));
}
}
try {
if (operatingSystem.isWindows()) {
return new WindowsConsoleDetector();
}
} catch (LinkageError e) {
// Thrown when jna cannot initialize the native stuff
LOGGER.debug("Unable to load native library. Continuing with fallback. Failure: {}", format(e));
}
return new NoOpConsoleDetector();
}
protected WindowsRegistry createWindowsRegistry(OperatingSystem operatingSystem) {
if (useNativePlatform && operatingSystem.isWindows()) {
return net.rubygrapefruit.platform.Native.get(WindowsRegistry.class);
}
return notAvailable(WindowsRegistry.class);
}
protected SystemInfo createSystemInfo() {
if (useNativePlatform) {
try {
return net.rubygrapefruit.platform.Native.get(SystemInfo.class);
} catch (NativeIntegrationUnavailableException e) {
LOGGER.debug("Native-platform system info is not available. Continuing with fallback.");
}
}
return notAvailable(SystemInfo.class);
}
protected ProcessLauncher createProcessLauncher() {
if (useNativePlatform) {
try {
return net.rubygrapefruit.platform.Native.get(ProcessLauncher.class);
} catch (NativeIntegrationUnavailableException e) {
LOGGER.debug("Native-platform process launcher is not available. Continuing with fallback.");
}
}
return new DefaultProcessLauncher();
}
protected PosixFiles createPosixFiles() {
if (useNativePlatform) {
try {
return net.rubygrapefruit.platform.Native.get(PosixFiles.class);
} catch (NativeIntegrationUnavailableException e) {
LOGGER.debug("Native-platform posix files is not available. Continuing with fallback.");
}
}
return notAvailable(UnavailablePosixFiles.class);
}
private T notAvailable(Class type) {
return (T) Proxy.newProxyInstance(type.getClassLoader(), new Class[]{type}, new BrokenService(type.getSimpleName()));
}
private static String format(Throwable throwable) {
StringBuilder builder = new StringBuilder();
builder.append(throwable.toString());
for (Throwable current = throwable.getCause(); current != null; current = current.getCause()) {
builder.append(SystemProperties.getInstance().getLineSeparator());
builder.append("caused by: ");
builder.append(current.toString());
}
return builder.toString();
}
private static class BrokenService implements InvocationHandler {
private final String type;
private BrokenService(String type) {
this.type = type;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
throw new org.gradle.internal.nativeintegration.NativeIntegrationUnavailableException(String.format("%s is not supported on this operating system.", type));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy