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

org.arquillian.cube.kubernetes.impl.utils.ProcessUtil Maven / Gradle / Ivy

There is a newer version: 2.0.0.Alpha1
Show newest version
/*
 * Copyright 2016 Red Hat, Inc.
 *
 * Red Hat 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.arquillian.cube.kubernetes.impl.utils;

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.URL;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import org.apache.commons.io.FileUtils;
import org.arquillian.cube.impl.util.Strings;
import org.arquillian.cube.kubernetes.api.Logger;

/**
 * A helper class for running external processes
 * 

* This class is a cut down version taken from the [Fabric8 Maven Plugin](https://github.com/fabric8io/fabric8-maven-plugin/blob/master/core/src/main/java/io/fabric8/maven/core/util/ProcessUtil.java) */ public class ProcessUtil { public static int runCommand(final Logger log, URL scriptUrl) throws IOException { return runCommand(log, scriptUrl, Collections.emptyMap()); } public static int runCommand(final Logger log, URL scriptUrl, Map env) throws IOException { File scriptFile = File.createTempFile("arquillian-cube-script", getSuffix()); FileUtils.copyURLToFile(scriptUrl, scriptFile); scriptFile.setExecutable(true, false); return runCommand(log, getCommand(), Arrays.asList(new String[] {scriptFile.getAbsolutePath()}), env, true); } public static int runCommand(final Logger log, String command, List args, Map env, boolean withShutdownHook) throws IOException { String[] commandWithArgs = prepareCommandArray(command, args); String[] envp = prepareEnvp(env); Process process = Runtime.getRuntime().exec(commandWithArgs, envp); if (withShutdownHook) { addShutdownHook(log, process, command); } List threads = startLoggingThreads(process, log, command + " " + Strings.join(args, " ")); try { int answer = process.waitFor(); joinThreads(threads, log); return answer; } catch (InterruptedException e) { return process.exitValue(); } } private static void joinThreads(List threads, Logger log) { for (Thread thread : threads) { try { thread.join(); } catch (InterruptedException e) { log.warn("Caught " + e.getMessage()); } } } // ========================================================================================================== private static void addShutdownHook(final Logger log, final Process process, final String command) { Runtime.getRuntime().addShutdownHook(new Thread(command) { @Override public void run() { if (process != null) { log.info("Terminating process [" + command + "]"); try { process.destroy(); } catch (Exception e) { log.error("Failed to terminate process [" + command + "]"); } } } }); } private static String[] prepareCommandArray(String command, List args) { List nCmd = Strings.splitAndTrimAsList(command, " "); List nArgs = args != null ? args : new ArrayList(); String[] commandWithArgs = new String[nCmd.size() + nArgs.size()]; for (int i = 0; i < nCmd.size(); i++) { commandWithArgs[i] = nCmd.get(i); } for (int i = 0; i < nArgs.size(); i++) { commandWithArgs[i + nCmd.size()] = nArgs.get(i); } return commandWithArgs; } private static String[] prepareEnvp(Map env) { ; String[] envp = new String[env.size()]; int i = 0; for (Map.Entry entry : env.entrySet()) { envp[i++] = entry.getKey() + "=" + entry.getValue(); } return envp; } private static void processOutput(InputStream inputStream, Function function) throws IOException { try (BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream))) { String line = null; while ((line = reader.readLine()) != null) { function.apply(line); } } } private static List startLoggingThreads(final Process process, final Logger log, final String commandDesc) { List threads = new ArrayList<>(); threads.add(startOutputLoggingThread(process, log, commandDesc)); threads.add(startErrorLoggingThread(process, log, commandDesc)); return threads; } private static Thread startErrorLoggingThread(final Process process, final Logger log, final String commandDesc) { Thread logThread = new Thread("[ERR] " + commandDesc) { @Override public void run() { try { processOutput(process.getErrorStream(), createErrorHandler(log)); } catch (IOException e) { log.error("Failed to read error stream from [" + commandDesc + "] : [" + e.getMessage() + "]"); } } }; logThread.setDaemon(true); logThread.start(); return logThread; } private static Thread startOutputLoggingThread(final Process process, final Logger log, final String commandDesc) { Thread logThread = new Thread("[OUT] " + commandDesc) { @Override public void run() { try { processOutput(process.getInputStream(), createOutputHandler(log)); } catch (IOException e) { log.error("Failed to read output stream from [" + commandDesc + "] : [" + e.getMessage() + "]"); } } }; logThread.setDaemon(true); logThread.start(); return logThread; } private static Function createOutputHandler(final Logger log) { return new Function() { @Override public Void apply(String outputLine) { log.info(outputLine); return null; } }; } private static Function createErrorHandler(final Logger log) { return new Function() { @Override public Void apply(String outputLine) { log.error(outputLine); return null; } }; } public static boolean isWindows() { return System.getProperty("os.name").toLowerCase().contains("windows"); } private static final String getCommand() { if (isWindows()) { return "cmd /c start"; } else { return "/bin/sh -c"; } } private static final String getSuffix() { if (isWindows()) { return ".bat"; } else { return ".sh"; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy