Считать вывод команды внутри процесса su - PullRequest
12 голосов
/ 01 августа 2011

Сначала я представлю свою ситуацию.Мне нужно выполнить команду "su" в моем приложении для Android, и она работает хорошо.Затем мне нужно выполнить команду "ls" и прочитать вывод.Я делаю это, получая поток вывода из процесса "su" и записывая в него свою команду.

И здесь возникает вопрос.Как прочитать вывод процесса "ls"?Все, что у меня есть, это объект процесса "su".Получение потока ввода из него ничего не дает, потому что "su" ничего не пишет.Но «ls» делает, и я не знаю, как получить доступ к его выходным сообщениям.

Я искал много сайтов, но не нашел никакого решения.Может быть, кто-то поможет мне:)

С уважением

Ответы [ 3 ]

24 голосов
/ 05 августа 2011

Хорошо, я нашел решение. Это должно выглядеть так:

Process p = Runtime.getRuntime().exec(new String[]{"su", "-c", "system/bin/sh"});
DataOutputStream stdin = new DataOutputStream(p.getOutputStream());
//from here all commands are executed with su permissions
stdin.writeBytes("ls /data\n"); // \n executes the command
InputStream stdout = p.getInputStream();
byte[] buffer = new byte[BUFF_LEN];
int read;
String out = new String();
//read method will wait forever if there is nothing in the stream
//so we need to read it in another way than while((read=stdout.read(buffer))>0)
while(true){
    read = stdout.read(buffer);
    out += new String(buffer, 0, read);
    if(read<BUFF_LEN){
        //we have read everything
        break;
    }
}
//do something with the output

Надеюсь, это будет кому-то полезно

5 голосов
/ 01 августа 2011
public String ls () {
    Class<?> execClass = Class.forName("android.os.Exec");
    Method createSubprocess = execClass.getMethod("createSubprocess", String.class, String.class, String.class, int[].class);
    int[] pid = new int[1];
    FileDescriptor fd = (FileDescriptor)createSubprocess.invoke(null, "/system/bin/ls", "/", null, pid);

    BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(fd)));
    String output = "";
    try {
        String line;
        while ((line = reader.readLine()) != null) {
            output += line + "\n";
        }
    }
    catch (IOException e) {}
    return output;
}

Проверьте этот код, упомянутый здесь:

Как запустить команду терминала в приложении Android?


try {
// Executes the command.
Process process = Runtime.getRuntime().exec("/system/bin/ls /sdcard");

// Reads stdout.
// NOTE: You can write to stdin of the command using
//       process.getOutputStream().
BufferedReader reader = new BufferedReader(
        new InputStreamReader(process.getInputStream()));
int read;
char[] buffer = new char[4096];
StringBuffer output = new StringBuffer();
while ((read = reader.read(buffer)) > 0) {
    output.append(buffer, 0, read);
}
reader.close();

// Waits for the command to finish.
process.waitFor();

return output.toString();
} catch (IOException e) {
throw new RuntimeException(e);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}

Ссылки

этот код GScript

2 голосов
/ 03 ноября 2017

Я изменил принятый ответ @glodos для следующих проблем:

  1. потоки закрыты, в противном случае процесс exec навсегда зависает в открытом потоке.Если вы выполните ps в оболочке (то есть adb shell) после нескольких выполнений, вы увидите несколько su процессов живыми.Они должны быть правильно завершены.
  2. добавлено waitFor(), чтобы убедиться, что процесс завершен.
  3. Добавлена ​​обработка для read=-1, теперь команды с пустыми stdout могут выполняться.Ранее они разбивались на new String(buffer, 0, read)
  4. Использование StringBuffer для более эффективной обработки строк.

    private String execCommand(String cmd) throws IOException, InterruptedException {
        Process p = Runtime.getRuntime().exec(new String[]{"su", "-c", "system/bin/sh"});
        DataOutputStream stdout = new DataOutputStream(p.getOutputStream());
    
        stdout.writeBytes(cmd);
        stdout.writeByte('\n');
        stdout.flush();
        stdout.close();
    
        BufferedReader stdin = new BufferedReader(new InputStreamReader(p.getInputStream()));
        char[] buffer = new char[1024];
        int read;
        StringBuffer out = new StringBuffer();
    
        while((read = stdin.read(buffer)) > 0) {
            out.append(buffer, 0, read);
        }
        stdin.close();
        p.waitFor();
        return out.toString();
    }
    

Некоторые кредиты идут на @Sherif elKhatib))

...