Skip to content

Commit adb0ae5

Browse files
D-D-Hy1yang0
authored andcommittedAug 6, 2021
8261441: JFR: Filename expansion
Reviewed-by: jbachorik, egahlin
1 parent e38e365 commit adb0ae5

File tree

9 files changed

+135
-21
lines changed

9 files changed

+135
-21
lines changed
 

‎src/java.base/share/man/java.1

+3
Original file line numberDiff line numberDiff line change
@@ -1747,6 +1747,9 @@ written when the recording is stopped, for example:
17471747
\f[CB]/home/user/recordings/recording.jfr\f[R]
17481748
.IP \[bu] 2
17491749
\f[CB]c:\\recordings\\recording.jfr\f[R]
1750+
.PP
1751+
If \f[CB]%p\f[R] and/or \f[CB]%t\f[R] is specified in the filename, it expands to the JVM\[aq]s
1752+
PID and the current timestamp, respectively.
17501753
.RE
17511754
.TP
17521755
.B \f[CB]name=\f[R]\f[I]identifier\f[R]

‎src/jdk.jcmd/share/man/jcmd.1

+6
Original file line numberDiff line numberDiff line change
@@ -441,6 +441,8 @@ If no filename is given, a filename is generated from the PID and the
441441
current date.
442442
The filename may also be a directory in which case, the filename is
443443
generated from the PID and the current date in the specified directory.
444+
If \f[CB]%p\f[R] and/or \f[CB]%t\f[R] is specified in the filename, it
445+
expands to the JVM\[aq]s PID and the current timestamp, respectively.
444446
(STRING, no default value)
445447
.IP \[bu] 2
446448
\f[CB]maxage\f[R]: (Optional) Length of time for dumping the flight
@@ -509,6 +511,8 @@ current date and is placed in the directory where the process was
509511
started.
510512
The filename may also be a directory in which case, the filename is
511513
generated from the PID and the current date in the specified directory.
514+
If \f[CB]%p\f[R] and/or \f[CB]%t\f[R] is specified in the filename, it
515+
expands to the JVM\[aq]s PID and the current timestamp, respectively.
512516
(STRING, no default value)
513517
.IP \[bu] 2
514518
\f[CB]maxage\f[R]: (Optional) Maximum time to keep the recorded data on
@@ -600,6 +604,8 @@ If no parameters are entered, then no recording is stopped.
600604
.IP \[bu] 2
601605
\f[CB]filename\f[R]: (Optional) Name of the file to which the recording is
602606
written when the recording is stopped.
607+
If \f[CB]%p\f[R] and/or \f[CB]%t\f[R] is specified in the filename, it
608+
expands to the JVM\[aq]s PID and the current timestamp, respectively.
603609
If no path is provided, the data from the recording is discarded.
604610
(STRING, no default value)
605611
.IP \[bu] 2

‎src/jdk.jfr/share/classes/jdk/jfr/internal/dcmd/AbstractDCmd.java

+39-1
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@
3030
import java.nio.file.Path;
3131
import java.nio.file.Paths;
3232
import java.time.Duration;
33+
import java.time.LocalDateTime;
3334
import java.util.ArrayList;
3435
import java.util.Collections;
3536
import java.util.Comparator;
@@ -191,7 +192,7 @@ protected final void print(String s) {
191192
}
192193

193194
protected final void print(String s, Object... args) {
194-
currentLine.append(String.format(s, args));
195+
currentLine.append(args.length > 0 ? String.format(s, args) : s);
195196
}
196197

197198
protected final void println(String s, Object... args) {
@@ -269,4 +270,41 @@ protected final String exampleDirectory() {
269270
return "/directory/recordings";
270271
}
271272
}
273+
274+
static String expandFilename(String filename) {
275+
if (filename == null || filename.indexOf('%') == -1) {
276+
return filename;
277+
}
278+
279+
String pid = null;
280+
String time = null;
281+
StringBuilder sb = new StringBuilder();
282+
for (int i = 0; i < filename.length(); i++) {
283+
char c = filename.charAt(i);
284+
if (c == '%' && i < filename.length() - 1) {
285+
char nc = filename.charAt(i + 1);
286+
if (nc == '%') { // %% ==> %
287+
sb.append('%');
288+
i++;
289+
} else if (nc == 'p') {
290+
if (pid == null) {
291+
pid = JVM.getJVM().getPid();
292+
}
293+
sb.append(pid);
294+
i++;
295+
} else if (nc == 't') {
296+
if (time == null) {
297+
time = Utils.formatDateTime(LocalDateTime.now());
298+
}
299+
sb.append(time);
300+
i++;
301+
} else {
302+
sb.append('%');
303+
}
304+
} else {
305+
sb.append(c);
306+
}
307+
}
308+
return sb.toString();
309+
}
272310
}

‎src/jdk.jfr/share/classes/jdk/jfr/internal/dcmd/DCmdDump.java

+5-1
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,7 @@ final class DCmdDump extends AbstractDCmd {
5555
public void execute(ArgumentParser parser) throws DCmdException {
5656
parser.checkUnknownArguments();
5757
String name = parser.getOption("name");
58-
String filename = parser.getOption("filename");
58+
String filename = expandFilename(parser.getOption("filename"));
5959
Long maxAge = parser.getOption("maxage");
6060
Long maxSize = parser.getOption("maxsize");
6161
String begin = parser.getOption("begin");
@@ -232,6 +232,10 @@ public String[] printHelp() {
232232
case, the filename is generated from the PID and the current date in
233233
the specified directory. (STRING, no default value)
234234
235+
Note: If a filename is given, '%%p' in the filename will be
236+
replaced by the PID, and '%%t' will be replaced by the time in
237+
'yyyy_MM_dd_HH_mm_ss' format.
238+
235239
maxage (Optional) Length of time for dumping the flight recording data to a
236240
file. (INTEGER followed by 's' for seconds 'm' for minutes or 'h' for
237241
hours, no default value)

‎src/jdk.jfr/share/classes/jdk/jfr/internal/dcmd/DCmdStart.java

+5-1
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,7 @@ public void execute(ArgumentParser parser) throws DCmdException {
7676
Long delay = parser.getOption("delay");
7777
Long duration = parser.getOption("duration");
7878
Boolean disk = parser.getOption("disk");
79-
String path = parser.getOption("filename");
79+
String path = expandFilename(parser.getOption("filename"));
8080
Long maxAge = parser.getOption("maxage");
8181
Long maxSize = parser.getOption("maxsize");
8282
Long flush = parser.getOption("flush-interval");
@@ -339,6 +339,10 @@ Virtual Machine (JVM) shuts down. If set to 'true' and no value
339339
generated from the PID and the current date in the specified
340340
directory. (STRING, no default value)
341341
342+
Note: If a filename is given, '%%p' in the filename will be
343+
replaced by the PID, and '%%t' will be replaced by the time in
344+
'yyyy_MM_dd_HH_mm_ss' format.
345+
342346
maxage (Optional) Maximum time to keep the recorded data on disk. This
343347
parameter is valid only when the disk parameter is set to true.
344348
Note 0s means forever. (INTEGER followed by 's' for seconds 'm'

‎src/jdk.jfr/share/classes/jdk/jfr/internal/dcmd/DCmdStop.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ final class DCmdStop extends AbstractDCmd {
4444
protected void execute(ArgumentParser parser) throws DCmdException {
4545
parser.checkUnknownArguments();
4646
String name = parser.getOption("name");
47-
String filename = parser.getOption("filename");
47+
String filename = expandFilename(parser.getOption("filename"));
4848
try {
4949
Recording recording = findRecording(name);
5050
WriteableUserPath path = PrivateAccess.getInstance().getPlatformRecording(recording).getDestination();
@@ -82,6 +82,9 @@ public String[] printHelp() {
8282
recording is stopped. If no path is provided, the data from the recording
8383
is discarded. (STRING, no default value)
8484
85+
Note: If a path is given, '%%p' in the path will be replaced by the PID,
86+
and '%%t' will be replaced by the time in 'yyyy_MM_dd_HH_mm_ss' format.
87+
8588
name Name of the recording (STRING, no default value)
8689
8790
Options must be specified using the <key> or <key>=<value> syntax.

‎test/jdk/jdk/jfr/jcmd/JcmdHelper.java

+13
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525

2626
import java.io.File;
2727
import java.util.Arrays;
28+
import java.util.Iterator;
2829
import java.util.stream.Collectors;
2930

3031
import jdk.test.lib.Asserts;
@@ -118,4 +119,16 @@ public static OutputAnalyzer jcmd(String... args) {
118119
public static OutputAnalyzer jcmdCheck(String recordingName, boolean verbose) {
119120
return jcmd("JFR.check", "name=" + recordingName, "verbose=" + verbose);
120121
}
122+
123+
public static String readFilename(OutputAnalyzer output) throws Exception {
124+
Iterator<String> it = output.asLines().iterator();
125+
while (it.hasNext()) {
126+
String line = it.next();
127+
if (line.contains("written to")) {
128+
line = it.next(); // blank line
129+
return it.next();
130+
}
131+
}
132+
throw new Exception("Could not find filename of dumped recording.");
133+
}
121134
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
/*
2+
* Copyright (c) 2021, Alibaba Group Holding Limited. All Rights Reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
package jdk.jfr.jcmd;
25+
26+
import java.io.File;
27+
import java.util.regex.Matcher;
28+
import java.util.regex.Pattern;
29+
30+
import jdk.test.lib.Asserts;
31+
import jdk.test.lib.jfr.FileHelper;
32+
import jdk.test.lib.process.OutputAnalyzer;
33+
34+
/**
35+
* @test
36+
* @summary The test verifies JFR.start/dump/stop commands
37+
* @key jfr
38+
* @requires vm.hasJFR
39+
* @library /test/lib /test/jdk
40+
* @run main/othervm jdk.jfr.jcmd.TestFilenameExpansion
41+
*/
42+
public class TestFilenameExpansion {
43+
44+
public static void main(String[] args) throws Exception {
45+
String pid = Long.toString(ProcessHandle.current().pid());
46+
String name = "output_%p_%t_%%.jfr";
47+
String pattern = "output_" + pid + "_" + "\\d{4}_\\d{2}_\\d{2}_\\d{2}_\\d{2}_\\d{2}" + "_%\\.jfr";
48+
49+
JcmdHelper.jcmd("JFR.start name=test");
50+
String filename = JcmdHelper.readFilename(JcmdHelper.jcmd("JFR.dump name=test filename=" + name));
51+
File file = new File(filename);
52+
Asserts.assertTrue(file.exists(), file.getAbsolutePath() + " does not exist");
53+
Asserts.assertTrue(file.isFile(), file.getAbsolutePath() + " is not a file");
54+
Asserts.assertTrue(Pattern.compile(pattern).matcher(filename).find());
55+
}
56+
}

‎test/jdk/jdk/jfr/jcmd/TestJcmdDumpGeneratedFilename.java

+4-17
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,6 @@
2626
import java.io.File;
2727
import java.nio.file.Path;
2828
import java.nio.file.Paths;
29-
import java.util.Iterator;
3029

3130
import jdk.jfr.Configuration;
3231
import jdk.jfr.Recording;
@@ -60,26 +59,26 @@ public static void main(String[] args) throws Exception {
6059

6160
private static void testDumpFilename() throws Exception {
6261
OutputAnalyzer output = JcmdHelper.jcmd("JFR.dump");
63-
verifyFile(readFilename(output), null);
62+
verifyFile(JcmdHelper.readFilename(output), null);
6463
}
6564

6665
private static void testDumpFilename(Recording r) throws Exception {
6766
OutputAnalyzer output = JcmdHelper.jcmd("JFR.dump", "name=" + r.getId());
68-
verifyFile(readFilename(output), r.getId());
67+
verifyFile(JcmdHelper.readFilename(output), r.getId());
6968
}
7069

7170
private static void testDumpDiectory() throws Exception {
7271
Path directory = Paths.get(".").toAbsolutePath().normalize();
7372
OutputAnalyzer output = JcmdHelper.jcmd("JFR.dump", "filename=" + directory);
74-
String filename = readFilename(output);
73+
String filename = JcmdHelper.readFilename(output);
7574
verifyFile(filename, null);
7675
verifyDirectory(filename, directory);
7776
}
7877

7978
private static void testDumpDiectory(Recording r) throws Exception {
8079
Path directory = Paths.get(".").toAbsolutePath().normalize();
8180
OutputAnalyzer output = JcmdHelper.jcmd("JFR.dump", "name=" + r.getId(), "filename=" + directory);
82-
String filename = readFilename(output);
81+
String filename = JcmdHelper.readFilename(output);
8382
verifyFile(filename, r.getId());
8483
verifyDirectory(filename, directory);
8584
}
@@ -98,16 +97,4 @@ private static void verifyFile(String filename, Long id) throws Exception {
9897
}
9998
FileHelper.verifyRecording(new File(filename));
10099
}
101-
102-
private static String readFilename(OutputAnalyzer output) throws Exception {
103-
Iterator<String> it = output.asLines().iterator();
104-
while (it.hasNext()) {
105-
String line = it.next();
106-
if (line.contains("written to")) {
107-
line = it.next(); // blank line
108-
return it.next();
109-
}
110-
}
111-
throw new Exception("Could not find filename of dumped recording.");
112-
}
113100
}

0 commit comments

Comments
 (0)
Please sign in to comment.