Skip to content

Commit

Permalink
Merge pull request #18305 from smowton/smowton/admin/agent-extracted-…
Browse files Browse the repository at this point in the history
…file-test

Java: Add test for a JavacTool-based compiler that doesn't use standard JavaFileObjects
  • Loading branch information
smowton authored Jan 14, 2025
2 parents 3e10e78 + 060161c commit b2bb143
Show file tree
Hide file tree
Showing 5 changed files with 110 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,93 @@
import javax.lang.model.element.Modifier;
import javax.lang.model.element.NestingKind;
import javax.tools.JavaCompiler;
import javax.tools.JavaFileObject;
import javax.tools.ToolProvider;
import java.io.*;
import java.net.URI;
import java.util.List;
import java.util.Objects;

public class Compiler {
public static void main(String[] args) {

JavaCompiler.CompilationTask jc = ToolProvider.getSystemJavaCompiler().getTask(
null, null, null, null, null,
List.of(
new JavaFileObject() {
@Override
public Kind getKind() {
return Kind.SOURCE;
}

@Override
public boolean isNameCompatible(String simpleName, Kind kind) {
return Objects.equals(simpleName, "Main");
}

@Override
public NestingKind getNestingKind() {
return null;
}

@Override
public Modifier getAccessLevel() {
return null;
}

@Override
public URI toUri() {
return URI.create("https://nonesuch.imaginary/somedir/Main.java");
}

@Override
public String getName() {
return "Main.java";
}

@Override
public InputStream openInputStream() throws IOException {
return new ByteArrayInputStream(this.getCharContent(true).toString().getBytes());
}

@Override
public OutputStream openOutputStream() throws IOException {
throw new IOException("No output allowed");
}

@Override
public Reader openReader(boolean ignoreEncodingErrors) throws IOException {
return new StringReader(this.getCharContent(ignoreEncodingErrors).toString());
}

@Override
public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {
return "public class Main { }";
}

@Override
public Writer openWriter() throws IOException {
throw new IOException("No output allowed");
}

@Override
public long getLastModified() {
return 0;
}

@Override
public boolean delete() {
return false;
}

@Override
public String toString() {
return "In-memory file with URI " + this.toUri();
}
}
)
);

jc.call();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
| Main |
5 changes: 5 additions & 0 deletions java/ql/integration-tests/java/javac-tool-custom-file/test.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
import commands

def test(codeql, java):
commands.run("javac Compiler.java")
codeql.database.create(command = "java Compiler")
5 changes: 5 additions & 0 deletions java/ql/integration-tests/java/javac-tool-custom-file/test.ql
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
import java

from Class c
where c.fromSource()
select c.getName()
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
---
category: minorAnalysis
---
* `JavacTool`-based compiler interception no longer requires an `--add-opens` directive when `FileObject.toUri` is accessible.
* `JavacTool`-based compiler interception no longer throws an exception visible to the program using `JavacTool` on failure to extract a file path from a passed `JavaFileObject`.
* `JavacTool`-based compiler interception now supports files that don't simply wrap a `file://` URL, such as a source file inside a JAR, or an in-memory file, but which do implement `getCharContent`.

0 comments on commit b2bb143

Please sign in to comment.