Наконец-то нашли решение, которое работает на меня. Это может быть полезно для других в этом стеке переполнения вопросов и ответов под названием Sudo In Java Process .
Решение включает в себя довольно много Java, поэтому я буду ссылаться только на ответ в исходной ссылке, упомянутой выше. Вот пример использования кода из этого ответа.
public static void main(String[] args) throws IOException, InterruptedException {
String password = LinuxCommand.getPasswdForRoot();
System.out.println("stdout of 'id':");
Process p = LinuxCommand.runFromRoot("id",password);
System.out.print(streamToString(p.getInputStream()));
System.out.println("stdout of 'fdisk -l':");
p = LinuxCommand.runFromRoot("fdisk -l",password);
System.out.print(streamToString(p.getInputStream()));
}
Фактическая реализация api, использованного выше:
import java.awt.BorderLayout;
import java.io.IOException;
import java.io.InputStream;
import javax.swing.JLabel;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JPasswordField;
public class LinuxCommand {
static InputStream is;
static byte[] buff = new byte[8192];
static int n;
public static String getPasswdForRoot() throws IOException {
Process p = Runtime.getRuntime().exec(new String[]{"sh","-c","sudo -S id"});
is = p.getErrorStream();
n = is.read(buff, 0, 8192);
String text = new String(buff,0,n);
if(text.contains("root"))return null; //not set password
JPanel panel = new JPanel(new BorderLayout());
JLabel lab = new JLabel(text);
panel.add(lab,BorderLayout.NORTH);
JPasswordField password = new JPasswordField();
panel.add(password,BorderLayout.SOUTH);
JOptionPane.showMessageDialog(null, panel);
byte[] passwd = (new String(password.getPassword())+"\r\n").getBytes();
p.getOutputStream().write(passwd);
p.getOutputStream().flush();
n = is.read(buff, 0, 8192);
if(n==-1) return new String(password.getPassword());
text = new String(buff,0,n);
while(true) {
lab.setText(text);
JOptionPane.showMessageDialog(null, panel);
p = Runtime.getRuntime().exec(new String[]{"sh","-c","sudo -S id"});
is = p.getErrorStream();
n = is.read(buff, 0, 8192);
passwd = (new String(password.getPassword())+"\n").getBytes();
p.getOutputStream().write(passwd);
p.getOutputStream().flush();
n = is.read(buff, 0, 8192);
if(n==-1) return new String(password.getPassword());
text = new String(buff,0,n);
}
}
public static Process runFromRoot(String command, String password) throws IOException {
byte[] passwd = (password+"\n").getBytes(); //for OutputStream better is byte[]
Process p = Runtime.getRuntime().exec(new String[]{"sh","-c","sudo -S "+command});
p.getOutputStream().write(passwd);
p.getOutputStream().flush();
return p;
}
}
Наряду с выводом при выполнении этого кода:
stdout of 'id':
uid=0(root) gid=0(root) grupy=0(root)
stdout of 'fdisk -l':
Disk /dev/sda: 640.1 GB, 640135028736 bytes
głowic: 255, sektorów/ścieżkę: 63, cylindrów: 77825, w sumie sektorów: 1250263728
Jednostka = sektorów, czyli 1 * 512 = 512 bajtów
Rozmiar sektora (logiczny/fizyczny) w bajtach: 512 / 4096
Rozmiar we/wy (minimalny/optymalny) w bajtach: 4096 / 4096
Identyfikator dysku: 0xc56b9eef
Urządzenie Rozruch Początek Koniec Bloków ID System
/dev/sda1 2048 37064703 18531328 27 Hidden NTFS WinRE
/dev/sda2 * 37064704 37269503 102400 7 HPFS/NTFS/exFAT
/dev/sda3 37269504 456711884 209721190+ 7 HPFS/NTFS/exFAT
/dev/sda4 456711946 1250258624 396773339+ f W95 Rozsz. (LBA)
Partycja 4 nie zaczyna się na granicy bloku fizycznego.
/dev/sda5 456711948 810350729 176819391 7 HPFS/NTFS/exFAT
Partycja 5 nie zaczyna się na granicy bloku fizycznego.
/dev/sda6 810350793 862802954 26226081 7 HPFS/NTFS/exFAT
Partycja 6 nie zaczyna się na granicy bloku fizycznego.
/dev/sda7 862803018 1020078408 78637695+ 83 Linux
Partycja 7 nie zaczyna się na granicy bloku fizycznego.
/dev/sda8 1020079368 1229791814 104856223+ 7 HPFS/NTFS/exFAT
/dev/sda9 1229791878 1250258624 10233373+ 7 HPFS/NTFS/exFAT
Partycja 9 nie zaczyna się na granicy bloku fizycznego.
-121--186962-
Это объясняется в статье, которую вы связали...
Можно создавать переменные среды с специально созданными значениями перед вызовом оболочки bash. Эти переменные могут содержать код, который выполняется сразу после вызова оболочки.
Это означает, что bash, вызываемый с помощью -c «echo this is test»
, выполняет код в одиночных кавычках при его вызове.
У Баша есть функции, хотя и в несколько ограниченной реализации, и эти функции bash можно поместить в переменные среды. Этот недостаток возникает при добавлении дополнительного кода в конец определения функций (внутри переменной enivronment).
Означает, что приведенный вами пример кода использует тот факт, что вызываемый bash не прекращает оценку этой последовательности после делать назначения. Назначение функции в этом случае.
На самом деле особая вещь в фрагменте кода, который вы опубликовали, как я понимаю, заключается в том, что, используя определение функции перед кодом, который мы хотим выполнить, некоторые механизмы безопасности можно обойти.
-121--900-
Изменение wait
на no
, скорее всего, решит проблему. На странице «man»:
Если его значение «yes», то услуга является однопоточной; это означает, что xinetd запустит сервер, а затем прекратит обработку запросов на обслуживание до тех пор, пока сервер не умрет и программное обеспечение сервера не примет подключение.
Бит ключа заключается в том, что если значение ожидания равно «да», ожидается, что серверное программное обеспечение примет соединение, чего не делает сценарий.
Я использовал настройки PulseAudio для настройки сетевой установки:
sudo apt-get install paprefs
Когда я деактивирую и повторно активирую две опции в «Доступе к сети»:
"Make discoverable PulseAudio network sound devices available locally"
"Make discoverable Apple AirTunes sound devices available locally"
«фантомный поток» исчезает.