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

org.seaborne.dboe.sys.ProcessUtils Maven / Gradle / Ivy

/*
 *  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.
 *
 *  See the NOTICE file distributed with this work for additional
 *  information regarding copyright ownership.
 */

package org.seaborne.dboe.sys;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.lang.management.ManagementFactory;
import java.util.ArrayList;
import java.util.List;

public class ProcessUtils {

    private static int myPid = -1;

    /**
     * Tries to get the PID of the current process caching it for future calls
     * since it won't change throughout the life of the process
     * 
     * @param fallback
     *            Fallback PID to return if unable to determine the PID i.e. an
     *            error code to return
     * 
     * @return PID of current process or provided {@code fallback} if unable to
     *         determine PID
     */
    public static int getPid(int fallback) {
        if (myPid != -1)
            return myPid;

        String runtimeBeanName = ManagementFactory.getRuntimeMXBean().getName();
        if (runtimeBeanName == null) {
            return useFallbackPid(fallback);
        }

        // Bean name will have format PID@hostname so we try to parse the PID
        // portion
        int index = runtimeBeanName.indexOf("@");
        if (index < 0)
            return useFallbackPid(fallback);
        try {
            // Parse and cache for future reuse
            String pidData = runtimeBeanName.substring(0, index);
            myPid = Integer.parseInt(pidData);
            return myPid;
        } catch (NumberFormatException e) {
            // Invalid PID
            return useFallbackPid(fallback);
        }
    }

    private static int useFallbackPid(int fallback) {
        // In the case where we can't determine our PID then treat ourselves as
        // no owner and cache for future use
        myPid = fallback;
        return myPid;
    }

    /**
     * Determines whether a given PID is alive
     * 
     * @param pid
     *            PID
     * @return True if the given PID is alive or unknown, false if it is dead
     */
    public static boolean isAlive(int pid) {
        String pidStr = Integer.toString(pid);
        try {
            List data = getProcessInfo(pidStr);

            // Expect a line to contain the PID to indicate the process is
            // alive
            for (String lineData : data) {
                if (lineData.contains(pidStr))
                    return true;
            }

            // Did not find any lines mentioning the PID so we can safely
            // assume that process is dead
            return false;
        } catch (IOException e) {
            // If any error running the process to check for the live process
            // then our check failed and for safety we assume the process is
            // alive

            Sys.errlog
                    .warn("Your platform does not support checking process liveness so TDB disk locations cannot be reliably locked to prevent possible corruption due to unsafe multi-JVM usage");
            return true;
        }
    }

    /**
     * Gets whether the platform we are running on will cause us to treat
     * negative (i.e. invalid) PIDs as alive because of the format in which the
     * command line process monitor application for the system returns errors
     * for invalid PIDs
     * 
     * @return True if a negative PID is treated as alive on this platform or we
     *         cannot determine liveness for PIDs on this platform, false
     *         otherwise
     */
    public static boolean negativePidsTreatedAsAlive() {
        return isAlive(-1);
    }

    /**
     * Gets process information based on the given PID string
     * 
     * @param pidStr
     *            PID string
     * @return Output of running the OSes appropriate command line task info
     *         application
     * @throws IOException
     *             Thrown if there is a problem running the command line
     *             application or reading the data returned
     */
    private static List getProcessInfo(String pidStr) throws IOException {
        Process p;
        if (Sys.isWindows) {
            // Use the Windows tasklist utility
            ProcessBuilder builder = new ProcessBuilder("tasklist", "/FI", "PID eq " + pidStr);
            builder.redirectErrorStream(true);
            p = builder.start();
        } else {
            // Use the ps utility
            ProcessBuilder builder = new ProcessBuilder("ps", "-p", pidStr);
            builder.redirectErrorStream(true);
            p = builder.start();
        }

        // Run and read data from the process
        try(BufferedReader reader = new BufferedReader(new InputStreamReader(p.getInputStream()))) {
            List data = new ArrayList<>();
            String line = null;
            while ((line = reader.readLine()) != null) {
                data.add(line);
            }
            return data;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy