I have a java_binary
target in my workspace that I'm later passing as an executable to ctx.actions.run
inside the rule. So far so good.
Now I want to debug this java_binary
while Bazel is executing the rule. In order to attach a debugger I need the java_binary
run in debug mode. So far, the only thing I came up with is setting jvm_flags
on the java_binary
. I was able to get that to work. But I was wondering if there is a way to achieve it from the command line instead of baking it into the java_binary
.
java_binary(
...
jvm_flags = [
"-agentlib:jdwp=transport=dt_socket,server=y,suspend=y,address=8000"
],
)
Is it possible to achieve this from the command line without hard coding jvm_flags
?
java_binary
outputs a wrapper script that includes a --debug[=<port>]
flag that will tell the JVM to wait for debugging. For example:
bazel run //:my-target -- --debug
Another strategy that gives more direct control and shows you how your Starlark rule is running the binary is to run the build with --subcommands
. With this Bazel will print out all the commands it runs during the build. Then find the command line corresponding to the invocation of the java_binary
you're interested in. Then you can copy/paste that command (including the cd
part) and modify it to include the debug flags, and debug it as you would any other process.
For example:
bazel build //:some-target-using-the-binary --subcommands=pretty_print
This should output something like:
INFO: Analyzed target //:some-target-using-the-binary (...).
INFO: Found 1 target...
SUBCOMMAND: # //:some-target-using-the-binary [action '...']
(cd /private/var/tmp/_bazel_username/hash/execroot/core && \
exec env - \
LC_CTYPE=en_US.UTF-8 \
LD_LIBRARY_PATH='' \
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin \
bazel-out/exec-config/bin/path/to/binary/package/TheBinaryToDebug \
@bazel-out/target-config/bin/some-target-using-the-binary-0.params)
--sandbox_debug
may be needed to prevent Bazel from deleting the execution root sandbox.
Note also that --subcommands
will only print the commands that are actually executed during the build, so a fully cached / fully incremental build will print nothing. You may need to do a clean
, or delete some of the outputs of the action you're interested in so that bazel runs that command.