Como invocar um comando shell Linux a partir de Java

93

Estou tentando executar alguns comandos Linux de Java usando redirecionamento (> &) e tubos (|). Como o Java pode invocar cshou bashcomandos?

Tentei usar isto:

Process p = Runtime.getRuntime().exec("shell command");

Mas não é compatível com redirecionamentos ou canais.

Narek
fonte
2
cate cshnão têm nada a ver um com o outro.
Bombe
4
eu posso entender a pergunta para outros comandos, mas para cat: por que diabos você simplesmente não lê o arquivo?
Atmocreations
8
Todo mundo errou na primeira vez - o exec () do Java não usa o shell do sistema subjacente para executar o comando (como kts aponta). O redirecionamento e a tubulação são recursos de um shell real e não estão disponíveis no exec () do Java.
SteveD
stevendick: Muito obrigado, estava tendo problemas por causa de redirecionamento e tubulação!
Narek
System.exit (0) não está dentro da verificação condicional se o processo foi concluído, então ele sempre sairá sem erros de saída. Nunca escreva condicionais sem chaves, para evitar exatamente esse tipo de erro.

Respostas:

96

exec não executa um comando em seu shell

experimentar

Process p = Runtime.getRuntime().exec(new String[]{"csh","-c","cat /home/narek/pk.txt"});

em vez de.

EDIT :: Não tenho csh no meu sistema, então usei o bash. O seguinte funcionou para mim

Process p = Runtime.getRuntime().exec(new String[]{"bash","-c","ls /home/XXX"});
KitsuneYMG
fonte
@Narek. Me desculpe por isso. Consertei removendo os extras. Aparentemente, eles não são necessários. Não tenho csh no meu sistema, mas funciona com o bash.
KitsuneYMG
3
Como outros mencionaram, isso deve ser independente se você tem csh ou bash, não é?
Narek,
@Narek. Deveria, mas não sei como csh lida com argumentos.
KitsuneYMG
1
Obrigado. Isso funcionou. Na verdade, usei "sh" em vez de "csh".
Farshid
5
Aviso: esta solução provavelmente irá encontrar o problema típico de travamento porque você não leu a saída e os fluxos de erro. Por exemplo: stackoverflow.com/questions/8595748/java-runtime-exec
Evgeni Sergeev
32

Use ProcessBuilder para separar comandos e argumentos em vez de espaços. Isso deve funcionar independentemente do shell usado:

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.List;

public class Test {

    public static void main(final String[] args) throws IOException, InterruptedException {
        //Build command 
        List<String> commands = new ArrayList<String>();
        commands.add("/bin/cat");
        //Add arguments
        commands.add("/home/narek/pk.txt");
        System.out.println(commands);

        //Run macro on target
        ProcessBuilder pb = new ProcessBuilder(commands);
        pb.directory(new File("/home/narek"));
        pb.redirectErrorStream(true);
        Process process = pb.start();

        //Read output
        StringBuilder out = new StringBuilder();
        BufferedReader br = new BufferedReader(new InputStreamReader(process.getInputStream()));
        String line = null, previous = null;
        while ((line = br.readLine()) != null)
            if (!line.equals(previous)) {
                previous = line;
                out.append(line).append('\n');
                System.out.println(line);
            }

        //Check result
        if (process.waitFor() == 0) {
            System.out.println("Success!");
            System.exit(0);
        }

        //Abnormal termination: Log command parameters and output and throw ExecutionException
        System.err.println(commands);
        System.err.println(out.toString());
        System.exit(1);
    }
}
Tim
fonte
Mesmo depois de java.util. *; foi importado corretamente, não consigo executar o exemplo acima.
Steve K de
@Stephan Eu atualizei o código de exemplo acima para remover as instruções de log e variáveis ​​que foram declaradas fora do código colado e agora ele deve compilar e executar. Sinta-se à vontade para tentar e me diga como funciona.
Tim
15

Com base no exemplo de @ Tim para fazer um método independente:

import java.io.BufferedReader;
import java.io.File;
import java.io.InputStreamReader;
import java.util.ArrayList;

public class Shell {

    /** Returns null if it failed for some reason.
     */
    public static ArrayList<String> command(final String cmdline,
    final String directory) {
        try {
            Process process = 
                new ProcessBuilder(new String[] {"bash", "-c", cmdline})
                    .redirectErrorStream(true)
                    .directory(new File(directory))
                    .start();

            ArrayList<String> output = new ArrayList<String>();
            BufferedReader br = new BufferedReader(
                new InputStreamReader(process.getInputStream()));
            String line = null;
            while ( (line = br.readLine()) != null )
                output.add(line);

            //There should really be a timeout here.
            if (0 != process.waitFor())
                return null;

            return output;

        } catch (Exception e) {
            //Warning: doing this is no good in high quality applications.
            //Instead, present appropriate error messages to the user.
            //But it's perfectly fine for prototyping.

            return null;
        }
    }

    public static void main(String[] args) {
        test("which bash");

        test("find . -type f -printf '%T@\\\\t%p\\\\n' "
            + "| sort -n | cut -f 2- | "
            + "sed -e 's/ /\\\\\\\\ /g' | xargs ls -halt");

    }

    static void test(String cmdline) {
        ArrayList<String> output = command(cmdline, ".");
        if (null == output)
            System.out.println("\n\n\t\tCOMMAND FAILED: " + cmdline);
        else
            for (String line : output)
                System.out.println(line);

    }
}

(O exemplo de teste é um comando que lista todos os arquivos em um diretório e seus subdiretórios, recursivamente, em ordem cronológica .)

A propósito, se alguém puder me dizer por que preciso de quatro e oito barras invertidas ali, em vez de duas e quatro, posso aprender alguma coisa. Há mais um nível de acontecimento sem escape do que o que estou contando.

Edit: acabei de tentar este mesmo código no Linux, e aí descobri que preciso da metade das barras invertidas no comando de teste! (Ou seja: o número esperado de dois e quatro.) Agora não é mais apenas estranho, é um problema de portabilidade.

Evgeni Sergeev
fonte
Você deve fazer sua pergunta como uma nova pergunta do StackOverflow, não como parte de sua resposta.
vog
Funciona com dois e quatro em OSX / Oracle java8. Parece que há um problema com seu ambiente java original (do qual você não mencionou a natureza)
TheMadsen