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

org.apache.sshd.common.util.OsUtils Maven / Gradle / Ivy

Go to download

This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up with different versions on classes on the class path).

There is a newer version: 35.0.0.Beta1
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you 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.apache.sshd.common.util;

import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.Predicate;
import java.util.function.Supplier;

/**
 * Operating system dependent utility methods.
 *
 * @author Apache MINA SSHD Project
 */
public final class OsUtils {

    /**
     * Property that can be used to override the reported value from {@link #getCurrentUser()}. If not set then
     * "user.name" system property is used
     */
    public static final String CURRENT_USER_OVERRIDE_PROP = "org.apache.sshd.currentUser";

    /**
     * Property that can be used to override the reported value from {@link #getJavaVersion()}. If not set then
     * "java.version" system property is used
     */
    public static final String JAVA_VERSION_OVERRIDE_PROP = "org.apache.sshd.javaVersion";

    /**
     * Property that can be used to override the reported value from {@link #isWin32()}. If not set then
     * "os.name" system property is used
     */
    public static final String OS_TYPE_OVERRIDE_PROP = "org.apache.sshd.osType";

    /**
     * Property that can be used to override the reported value from {@link #isAndroid()}. If not set then
     * {@link #ANDROID_DETECTION_PROPERTIES} are used to determine its value. Otherwise, it must contain the string
     * "android" (case-insensitive)
     *
     * @see #ANDROID_PROPERTY_VALUE_MATCHER
     */
    public static final String ANDROID_MODE_OVERRIDE_PROP = "org.apache.sshd.androidMode";

    /**
     * Property that can be used to override the reported value from {@link #isDalvikMachine()}. If not set then
     * {@link #DALVIK_DETECTION_PROPERTIES} are used to determine its value. Otherwise, it must contain the string
     * "dalvik" (case-insensitive)
     */
    public static final String DALVIK_MACHINE_OVERRIDE_PROP = "org.apache.sshd.dalvikMachine";

    public static final String WINDOWS_SHELL_COMMAND_NAME = "cmd.exe";
    public static final String LINUX_SHELL_COMMAND_NAME = "/bin/sh";

    public static final String ROOT_USER = "root";

    public static final List LINUX_COMMAND
            = Collections.unmodifiableList(Arrays.asList(LINUX_SHELL_COMMAND_NAME, "-i", "-l"));
    public static final List WINDOWS_COMMAND
            = Collections.unmodifiableList(Collections.singletonList(WINDOWS_SHELL_COMMAND_NAME));

    /**
     * System properties consulted in order to detect {@link #isAndroid() Android O/S}.
     *
     * @see Android Developer
     */
    public static final List ANDROID_DETECTION_PROPERTIES
            = Collections.unmodifiableList(
                    Arrays.asList(
                            "java.vendor",
                            "java.specification.vendor",
                            "java.vm.vendor",
                            "java.vm.specification.vendor"));

    public static final Predicate ANDROID_PROPERTY_VALUE_MATCHER
            = v -> GenericUtils.trimToEmpty(v).toLowerCase().contains("android");

    /**
     * System properties consulted in order to detect {@link #isDalvikMachine() Dalvik machine}.
     *
     * @see Android Developer
     */
    public static final List DALVIK_DETECTION_PROPERTIES
            = Collections.unmodifiableList(
                    Arrays.asList(
                            "java.specification.name",
                            "java.vm.name",
                            "java.vm.specification.name"));

    public static final Predicate DALVIK_PROPERTY_VALUE_MATCHER
            = v -> GenericUtils.trimToEmpty(v).toLowerCase().contains("dalvik");

    private static final AtomicReference CURRENT_USER_HOLDER = new AtomicReference<>(null);
    private static final AtomicReference JAVA_VERSION_HOLDER = new AtomicReference<>(null);
    private static final AtomicReference OS_TYPE_HOLDER = new AtomicReference<>(null);

    private static final AtomicReference ANDROID_HOLDER = new AtomicReference<>(null);
    private static final AtomicReference DALVIK_HOLDER = new AtomicReference<>(null);

    private static final AtomicReference> CWD_PROVIDER_HOLDER = new AtomicReference<>();

    private OsUtils() {
        throw new UnsupportedOperationException("No instance allowed");
    }

    /**
     * @return {@code true} if currently running on Android. Note: {@link #isUNIX()} is also probably
     *         {@code true} as well, so special care must be taken in code that consults these values
     * @see    #ANDROID_DETECTION_PROPERTIES
     * @see    #ANDROID_MODE_OVERRIDE_PROP
     * @see    #ANDROID_PROPERTY_VALUE_MATCHER
     */
    public static boolean isAndroid() {
        return resolveAndroidSettingFlag(
                ANDROID_HOLDER, ANDROID_MODE_OVERRIDE_PROP, ANDROID_DETECTION_PROPERTIES, ANDROID_PROPERTY_VALUE_MATCHER);
    }

    /**
     * Override the value returned by {@link #isAndroid()} programmatically
     *
     * @param value Value to set if {@code null} then value is auto-detected
     */
    public static void setAndroid(Boolean value) {
        synchronized (ANDROID_HOLDER) {
            ANDROID_HOLDER.set(value);
        }
    }

    /**
     * @return {@code true} if currently running on a Dalvik machine. Note: {@link #isUNIX()} and/or
     *         {@link #isAndroid()} are also probably {@code true} as well, so special care must be taken in code that
     *         consults these values
     * @see    #DALVIK_DETECTION_PROPERTIES
     * @see    #DALVIK_MACHINE_OVERRIDE_PROP
     * @see    #DALVIK_PROPERTY_VALUE_MATCHER
     */
    public static boolean isDalvikMachine() {
        return resolveAndroidSettingFlag(
                DALVIK_HOLDER, DALVIK_MACHINE_OVERRIDE_PROP, DALVIK_DETECTION_PROPERTIES, DALVIK_PROPERTY_VALUE_MATCHER);
    }

    /**
     * Override the value returned by {@link #isDalvikMachine()} programmatically
     *
     * @param value Value to set if {@code null} then value is auto-detected
     */
    public static void setDalvikMachine(Boolean value) {
        synchronized (DALVIK_HOLDER) {
            DALVIK_HOLDER.set(value);
        }
    }

    /**
     * @return true if the host is a UNIX system (and not Windows). Note: this does not preclude
     *         {@link #isAndroid()} or {@link #isDalvikMachine()} from being {@code true} as well.
     */
    public static boolean isUNIX() {
        return !isWin32() && !isOSX();
    }

    /**
     * @return true if the host is a OSX (and not Windows or Unix).
     */
    public static boolean isOSX() {
        return getOS().contains("mac");
    }

    /**
     * @return true if the host is Windows (and not UNIX).
     * @see    #OS_TYPE_OVERRIDE_PROP
     * @see    #setOS(String)
     */
    public static boolean isWin32() {
        return getOS().contains("windows");
    }

    /**
     * Can be used to enforce Win32 or Linux report from {@link #isWin32()}, {@link #isOSX()} or {@link #isUNIX()}
     *
     * @param os The value to set - if {@code null} then O/S type is auto-detected
     * @see      #isWin32()
     * @see      #isOSX()
     * @see      #isUNIX()
     */
    public static void setOS(String os) {
        synchronized (OS_TYPE_HOLDER) {
            OS_TYPE_HOLDER.set(os);
        }
    }

    private static boolean resolveAndroidSettingFlag(
            AtomicReference flagHolder, String overrideProp,
            Collection detectionProps, Predicate detector) {
        synchronized (flagHolder) {
            Boolean value = flagHolder.get();
            if (value != null) {
                return value.booleanValue();
            }

            String propValue = System.getProperty(overrideProp);
            if (detector.test(propValue)) {
                flagHolder.set(Boolean.TRUE);
                return true;
            }

            for (String p : detectionProps) {
                String detectionPropValue = System.getProperty(p);
                if (detector.test(detectionPropValue)) {
                    flagHolder.set(Boolean.TRUE);
                    return true;
                }
            }

            flagHolder.set(Boolean.FALSE);
        }

        return false;
    }

    /**
     * @return The resolved O/S type string if not already set (lowercase)
     */
    private static String getOS() {
        String typeValue;
        synchronized (OS_TYPE_HOLDER) {
            typeValue = OS_TYPE_HOLDER.get();
            if (typeValue != null) { // is it the 1st time
                return typeValue;
            }

            String value = System.getProperty(OS_TYPE_OVERRIDE_PROP, System.getProperty("os.name"));
            typeValue = GenericUtils.trimToEmpty(value).toLowerCase();
            OS_TYPE_HOLDER.set(typeValue);
        }

        return typeValue;
    }

    public static String resolveDefaultInteractiveShellCommand() {
        return resolveDefaultInteractiveShellCommand(isWin32());
    }

    public static String resolveDefaultInteractiveShellCommand(boolean winOS) {
        return winOS ? WINDOWS_SHELL_COMMAND_NAME : LINUX_SHELL_COMMAND_NAME + " -i -l";
    }

    public static List resolveDefaultInteractiveCommandElements() {
        return resolveDefaultInteractiveCommandElements(isWin32());
    }

    public static List resolveDefaultInteractiveCommandElements(boolean winOS) {
        if (winOS) {
            return WINDOWS_COMMAND;
        } else {
            return LINUX_COMMAND;
        }
    }

    /**
     * @return The (C)urrent (W)orking (D)irectory {@link Path} - {@code null} if cannot resolve it. Resolution occurs
     *         as follows:
     *         
    *
  • Consult any currently registered {@link #setCurrentWorkingDirectoryResolver(Supplier) resolver}.
  • * *
  • If no resolver registered, then "user.dir" system property is consulted.
  • *
* @see #setCurrentWorkingDirectoryResolver(Supplier) */ public static Path getCurrentWorkingDirectory() { Supplier cwdProvider; synchronized (CWD_PROVIDER_HOLDER) { cwdProvider = CWD_PROVIDER_HOLDER.get(); } if (cwdProvider != null) { return cwdProvider.get(); } String cwdLocal = System.getProperty("user.dir"); return GenericUtils.isBlank(cwdLocal) ? null : Paths.get(cwdLocal); } /** * Allows the user to "plug-in" a resolver for the {@link #getCurrentWorkingDirectory()} method * * @param cwdProvider The {@link Supplier} of the (C)urrent (W)orking (D)irectory {@link Path} - if {@code null} * then "user.dir" system property is consulted */ public static void setCurrentWorkingDirectoryResolver(Supplier cwdProvider) { synchronized (CWD_PROVIDER_HOLDER) { CWD_PROVIDER_HOLDER.set(cwdProvider); } } /** * Get current user name * * @return Current user * @see #CURRENT_USER_OVERRIDE_PROP */ public static String getCurrentUser() { String username; synchronized (CURRENT_USER_HOLDER) { username = CURRENT_USER_HOLDER.get(); if (username != null) { // have we already resolved it ? return username; } username = getCanonicalUser(System.getProperty(CURRENT_USER_OVERRIDE_PROP, System.getProperty("user.name"))); ValidateUtils.checkNotNullAndNotEmpty(username, "No username available"); CURRENT_USER_HOLDER.set(username); } return username; } /** * Remove {@code Windows} domain and/or group prefix as well as "(User);" suffix * * @param user The original username - ignored if {@code null}/empty * @return The canonical user - unchanged if {@code Unix} O/S */ public static String getCanonicalUser(String user) { if (GenericUtils.isEmpty(user)) { return user; } // Windows owner sometime has the domain and/or group prepended to it if (isWin32()) { int pos = user.lastIndexOf('\\'); if (pos > 0) { user = user.substring(pos + 1); } pos = user.indexOf(' '); if (pos > 0) { user = user.substring(0, pos).trim(); } } return user; } /** * Attempts to resolve canonical group name for {@code Windows} * * @param group The original group name - used if not {@code null}/empty * @param user The owner name - sometimes it contains a group name * @return The canonical group name */ public static String resolveCanonicalGroup(String group, String user) { if (isUNIX()) { return group; } // we reach this code only for Windows if (GenericUtils.isEmpty(group)) { int pos = GenericUtils.isEmpty(user) ? -1 : user.lastIndexOf('\\'); return (pos > 0) ? user.substring(0, pos) : group; } int pos = group.indexOf(' '); return (pos < 0) ? group : group.substring(0, pos).trim(); } /** * Can be used to programmatically set the username reported by {@link #getCurrentUser()} * * @param username The username to set - if {@code null} then {@link #CURRENT_USER_OVERRIDE_PROP} will be consulted */ public static void setCurrentUser(String username) { synchronized (CURRENT_USER_HOLDER) { CURRENT_USER_HOLDER.set(username); } } /** * Resolves the reported Java version by consulting {@link #JAVA_VERSION_OVERRIDE_PROP}. If not set, then * "java.version" property is used * * @return The resolved {@link VersionInfo} - never {@code null} * @see #setJavaVersion(VersionInfo) */ public static VersionInfo getJavaVersion() { VersionInfo version; synchronized (JAVA_VERSION_HOLDER) { version = JAVA_VERSION_HOLDER.get(); if (version != null) { // first time ? return version; } String value = System.getProperty(JAVA_VERSION_OVERRIDE_PROP, System.getProperty("java.version")); // e.g.: 1.7.5_30 value = ValidateUtils.checkNotNullAndNotEmpty(value, "No configured Java version value").replace('_', '.'); // clean up any non-digits - in case something like 1.6.8_25-b323 for (int index = 0; index < value.length(); index++) { char ch = value.charAt(index); if ((ch == '.') || ((ch >= '0') && (ch <= '9'))) { continue; } value = value.substring(0, index); break; } version = ValidateUtils.checkNotNull(VersionInfo.parse(value), "No version parsed for %s", value); JAVA_VERSION_HOLDER.set(version); } return version; } /** * Set programmatically the reported Java version * * @param version The version - if {@code null} then it will be automatically resolved */ public static void setJavaVersion(VersionInfo version) { synchronized (JAVA_VERSION_HOLDER) { JAVA_VERSION_HOLDER.set(version); } } /** * @param path The original path * @return A path that can be compared with another one where case sensitivity of the underlying O/S has been * taken into account - never {@code null} */ public static String getComparablePath(String path) { String p = (path == null) ? "" : path; return isWin32() ? p.toLowerCase() : p; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy