Can Java launch the Windows UAC?

前端 未结 4 1035
眼角桃花
眼角桃花 2021-02-10 07:40

As the title says, I\'m wondering if it is possible for a program written in Java (and only java) to relaunch himself (preferably a .jar) with administrator privileges, s

相关标签:
4条回答
  • 2021-02-10 07:47

    I doubt "only Java". At best you would have to have a JNI wrapper around the MSFT module. Unless just invoking the exe using ProcessBuilder counts as "only Java" -- your code to bring up the user console would be only Java but not what it invokes. IOW, Win does not come with a Java API

    0 讨论(0)
  • 2021-02-10 07:49

    It cannot be done in pure java.

    Best bet would be to write this to a file:

    @echo Set objShell = CreateObject("Shell.Application") > %temp%\sudo.tmp.vbs
    @echo args = Right("%*", (Len("%*") - Len("%1"))) >> %temp%\sudo.tmp.vbs
    @echo objShell.ShellExecute "%1", args, "", "runas" >> %temp%\sudo.tmp.vbs
    @cscript %temp%\sudo.tmp.vbs
    

    and save it as something.bat in Windows temp directory (as we have access to this).

    You would then execute this from your application using Runtime or ProcessBuilder and exit your application (System.exit(0);).

    You should add an immediate start up check to your application that checks if the program has elevation, if it has proceed if not re-run the batch and exit.

    Here is an example I made (this must be run when compiled as a Jar or it wont work):

    import java.io.File;
    import java.io.FileWriter;
    import java.io.IOException;
    import javax.swing.JOptionPane;
    
    /**
     *
     * @author David
     */
    public class UacTest {
    
        public static String jarName = "UacTest.jar", batName = "elevate.bat";
    
        /**
         * @param args the command line arguments
         */
        public static void main(String[] args) {
            if (checkForUac()) {//uac is on
                JOptionPane.showMessageDialog(null, "I am not elevated");
                //attempt elevation
                new UacTest().elevate();
                System.exit(0);
            } else {//uac is not on
                //if we get here we are elevated
                JOptionPane.showMessageDialog(null, "I am elevated");
            }
    
        }
    
        private static boolean checkForUac() {
            File dummyFile = new File("c:/aaa.txt");
            dummyFile.deleteOnExit();
    
            try {
                //attempt to craete file in c:/
                try (FileWriter fw = new FileWriter(dummyFile, true)) {
                }
            } catch (IOException ex) {//we cannot UAC muts be on
                //ex.printStackTrace();
                return true;
            }
            return false;
        }
    
        private void elevate() {
            //create batch file in temporary directory as we have access to it regardless of UAC on or off
            File file = new File(System.getProperty("java.io.tmpdir") + "/" + batName);
            file.deleteOnExit();
    
            createBatchFile(file);
    
            runBatchFile();
    
        }
    
        private String getJarLocation() {
            return getClass().getProtectionDomain().getCodeSource().getLocation().getPath().substring(1);
        }
    
        private void runBatchFile() {
            //JOptionPane.showMessageDialog(null, getJarLocation());
    
            Runtime runtime = Runtime.getRuntime();
            String[] cmd = new String[]{"cmd.exe", "/C",
                System.getProperty("java.io.tmpdir") + "/" + batName + " java -jar " + getJarLocation()};
            try {
                Process proc = runtime.exec(cmd);
                //proc.waitFor();
            } catch (Exception ex) {
                ex.printStackTrace();
            }
        }
    
        private void createBatchFile(File file) {
            try {
                try (FileWriter fw = new FileWriter(file, true)) {
                    fw.write(
                            "@echo Set objShell = CreateObject(\"Shell.Application\") > %temp%\\sudo.tmp.vbs\r\n"
                            + "@echo args = Right(\"%*\", (Len(\"%*\") - Len(\"%1\"))) >> %temp%\\sudo.tmp.vbs\r\n"
                            + "@echo objShell.ShellExecute \"%1\", args, \"\", \"runas\" >> %temp%\\sudo.tmp.vbs\r\n"
                            + "@cscript %temp%\\sudo.tmp.vbs\r\n"
                            + "del /f %temp%\\sudo.tmp.vbs\r\n");
                }
            } catch (IOException ex) {
                //ex.printStackTrace();
            }
        }
    }
    
    0 讨论(0)
  • 2021-02-10 08:07

    To relaunch your application elevated, you have to call ShellExecute or ShellExecuteEx function from Windows API and use runas verb.

    You can use these API in pure Java with JNA library.

    To relaunch yourself, you would have to know the full path to java.exe or javaw.exe, the command-line parameters (class path, if any, and the path to your jar). Obviously you can get this information by using Windows API.


    What do you mean by remote case?
    You cannot start remote elevated process this way.

    You can re-launch your application elevated from a network share. Yet it won't work with mapped drives: after elevation there's no access to user's mapped drives.


    No, this can't work on other platforms. UAC is a Windows feature. It's similar to sudo in Linux in some ways, so for Linux you can use sudo $pathtojava/java.exe <yourparameters>. However this won't work nicely if your application is not started from a console. Window Managers usually have wrappers which prompt for password in a GUI dialog.

    0 讨论(0)
  • 2021-02-10 08:09

    Use a batch file and the runas command.

    0 讨论(0)
提交回复
热议问题