1a21c1a783
Reviewed-by: amenkov
219 lines
7.6 KiB
Java
219 lines
7.6 KiB
Java
/*
|
|
* Copyright (c) 2018, 2023, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
package lib.jdb;
|
|
|
|
import jdk.test.lib.JDWP;
|
|
import jdk.test.lib.Utils;
|
|
import jdk.test.lib.process.ProcessTools;
|
|
|
|
import java.io.Closeable;
|
|
import java.io.IOException;
|
|
import java.util.LinkedList;
|
|
import java.util.List;
|
|
import java.util.concurrent.TimeUnit;
|
|
import java.util.concurrent.TimeoutException;
|
|
import java.util.function.Function;
|
|
import java.util.regex.Matcher;
|
|
import java.util.regex.Pattern;
|
|
import java.util.stream.Collectors;
|
|
|
|
/**
|
|
* Helper class to run java debuggee and parse agent listening transport/address.
|
|
* Usage:
|
|
* 1)
|
|
* Debugee debugee = Debuggee.launcher("MyClass").setTransport("dt_shmem").launch();
|
|
* try {
|
|
* String transport = debuggee.getTransport();
|
|
* String addr = debuggee.getAddress();
|
|
* } finally {
|
|
* debuggee.shutdown();
|
|
* }
|
|
* 2) (using try-with-resource)
|
|
* try (Debugee debugee = Debuggee.launcher("MyClass").launch()) {
|
|
* String transport = debuggee.getTransport();
|
|
* String addr = debuggee.getAddress();
|
|
* }
|
|
* 3)
|
|
* ProcessBuilder pb = Debuggee.launcher("MyClass").setSuspended(false).prepare();
|
|
* ProcessTools.executeProcess(pb);
|
|
*/
|
|
public class Debuggee implements Closeable {
|
|
|
|
public static Launcher launcher(String mainClass) {
|
|
return new Launcher(mainClass);
|
|
}
|
|
|
|
public static class Launcher {
|
|
private final String mainClass;
|
|
private final List<String> options = new LinkedList<>();
|
|
private String vmOptions = null;
|
|
private String transport = "dt_socket";
|
|
private String address = null;
|
|
private boolean suspended = true;
|
|
private String onthrow = "";
|
|
private static final String LAUNCH_ECHO_STRING = "Listen Args:";
|
|
|
|
private Launcher(String mainClass) {
|
|
this.mainClass = mainClass;
|
|
}
|
|
public Launcher addOption(String option) {
|
|
options.add(option);
|
|
return this;
|
|
}
|
|
public Launcher addOptions(List<String> options) {
|
|
this.options.addAll(options);
|
|
return this;
|
|
}
|
|
public Launcher addVMOptions(String vmOptions) {
|
|
this.vmOptions = vmOptions;
|
|
return this;
|
|
}
|
|
// default is "dt_socket"
|
|
public Launcher setTransport(String value) {
|
|
transport = value;
|
|
return this;
|
|
}
|
|
// default is "null" (auto-generate)
|
|
public Launcher setAddress(String value) {
|
|
address = value;
|
|
return this;
|
|
}
|
|
// default is "true"
|
|
public Launcher setSuspended(boolean value) {
|
|
suspended = value;
|
|
return this;
|
|
}
|
|
|
|
public Launcher enableOnThrow(String exceptionClassName) {
|
|
this.onthrow = exceptionClassName;
|
|
return this;
|
|
}
|
|
|
|
public ProcessBuilder prepare() {
|
|
List<String> debuggeeArgs = new LinkedList<>();
|
|
if (vmOptions != null) {
|
|
debuggeeArgs.add(vmOptions);
|
|
}
|
|
String onthrowArgs = onthrow.isEmpty() ? "" : ",onthrow=" + onthrow + ",launch=echo " + LAUNCH_ECHO_STRING;
|
|
debuggeeArgs.add("-agentlib:jdwp=transport=" + transport
|
|
+ (address == null ? "" : ",address=" + address)
|
|
+ ",server=y,suspend=" + (suspended ? "y" : "n")
|
|
+ onthrowArgs);
|
|
debuggeeArgs.addAll(options);
|
|
debuggeeArgs.add(mainClass);
|
|
return ProcessTools.createTestJavaProcessBuilder(debuggeeArgs);
|
|
}
|
|
|
|
public Debuggee launch(String name) {
|
|
return new Debuggee(prepare(), name,
|
|
onthrow.isEmpty() ?
|
|
JDWP::parseListenAddress :
|
|
Launcher::parseLaunchEchoListenAddress
|
|
);
|
|
}
|
|
public Debuggee launch() {
|
|
return launch("debuggee");
|
|
}
|
|
|
|
/**
|
|
* Parses debuggee output to get listening transport and address, printed by `launch=echo`.
|
|
* Returns null if the string specified does not contain required info.
|
|
*/
|
|
private static JDWP.ListenAddress parseLaunchEchoListenAddress(String debuggeeOutput) {
|
|
Pattern listenRegexp = Pattern.compile(LAUNCH_ECHO_STRING + " \\b(.+)\\b \\b(.+)\\b");
|
|
Matcher m = listenRegexp.matcher(debuggeeOutput);
|
|
if (m.find()) {
|
|
return new JDWP.ListenAddress(m.group(1), m.group(2));
|
|
}
|
|
return null;
|
|
}
|
|
}
|
|
|
|
// starts the process, waits until the provided addressDetector detects transport/address from the process output
|
|
private Debuggee(ProcessBuilder pb, String name, Function<String, JDWP.ListenAddress> addressDetector) {
|
|
JDWP.ListenAddress[] listenAddress = new JDWP.ListenAddress[1];
|
|
try {
|
|
p = ProcessTools.startProcess(name, pb,
|
|
s -> output.add(s), // output consumer
|
|
s -> {
|
|
listenAddress[0] = addressDetector.apply(s);
|
|
return listenAddress[0] != null;
|
|
},
|
|
30, TimeUnit.SECONDS);
|
|
transport = listenAddress[0].transport();
|
|
address = listenAddress[0].address();
|
|
} catch (IOException | InterruptedException | TimeoutException ex) {
|
|
throw new RuntimeException("failed to launch debuggee", ex);
|
|
}
|
|
}
|
|
|
|
private final Process p;
|
|
private final List<String> output = new LinkedList<>();
|
|
private final String transport;
|
|
private final String address;
|
|
|
|
public void shutdown() {
|
|
try {
|
|
close();
|
|
} catch (IOException ex) {
|
|
// ignore
|
|
}
|
|
}
|
|
|
|
// waits until the process shutdown or crash
|
|
public boolean waitFor(long timeout, TimeUnit unit) {
|
|
try {
|
|
return p.waitFor(Utils.adjustTimeout(timeout), unit);
|
|
} catch (InterruptedException e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// returns the whole debuggee output as a string
|
|
public String getOutput() {
|
|
return output.stream().collect(Collectors.joining(Utils.NEW_LINE));
|
|
}
|
|
|
|
String getTransport() {
|
|
if (transport == null) {
|
|
throw new IllegalStateException("transport is not available");
|
|
}
|
|
return transport;
|
|
}
|
|
|
|
public String getAddress() {
|
|
if (address == null) {
|
|
throw new IllegalStateException("address is not available");
|
|
}
|
|
return address;
|
|
}
|
|
|
|
@Override
|
|
public void close() throws IOException {
|
|
if (p.isAlive()) {
|
|
p.destroy();
|
|
}
|
|
}
|
|
}
|