|
1 /* |
|
2 * Copyright (c) 2013, Oracle and/or its affiliates. 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 import java.io.IOException; |
|
25 import java.io.File; |
|
26 import java.nio.file.Files; |
|
27 import java.nio.file.Path; |
|
28 import java.util.Arrays; |
|
29 import java.util.regex.Pattern; |
|
30 import java.util.regex.Matcher; |
|
31 import jdk.testlibrary.OutputAnalyzer; |
|
32 import jdk.testlibrary.JDKToolLauncher; |
|
33 import jdk.testlibrary.ProcessTools; |
|
34 import jdk.testlibrary.Utils; |
|
35 import jdk.testlibrary.ProcessThread; |
|
36 |
|
37 /* |
|
38 * Utility functions for test runners. |
|
39 * (Test runner = class that launch a test) |
|
40 */ |
|
41 public class RunnerUtil { |
|
42 /** |
|
43 * The Application process must be run concurrently with our tests since |
|
44 * the tests will attach to the Application. |
|
45 * We will run the Application process in a separate thread. |
|
46 * |
|
47 * The Application must be started with flag "-Xshare:off" for the Retransform |
|
48 * test in TestBasics to pass on all platforms. |
|
49 * |
|
50 * The Application will write its pid and shutdownPort in the given outFile. |
|
51 */ |
|
52 public static ProcessThread startApplication(String outFile) throws Throwable { |
|
53 String classpath = System.getProperty("test.class.path", "."); |
|
54 String[] args = Utils.addTestJavaOpts( |
|
55 "-Dattach.test=true", "-classpath", classpath, "Application", outFile); |
|
56 ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(args); |
|
57 ProcessThread pt = new ProcessThread("runApplication", pb); |
|
58 pt.start(); |
|
59 return pt; |
|
60 } |
|
61 |
|
62 /** |
|
63 * Will stop the running Application. |
|
64 * First tries to shutdown nicely by connecting to the shut down port. |
|
65 * If that fails, the process will be killed hard with stopProcess(). |
|
66 * |
|
67 * If the nice shutdown fails, then an Exception is thrown and the test should fail. |
|
68 * |
|
69 * @param port The shut down port. |
|
70 * @param processThread The process to stop. |
|
71 */ |
|
72 public static void stopApplication(int port, ProcessThread processThread) throws Throwable { |
|
73 if (processThread == null) { |
|
74 System.out.println("RunnerUtil.stopApplication ignored since proc is null"); |
|
75 return; |
|
76 } |
|
77 try { |
|
78 System.out.println("RunnerUtil.stopApplication waiting to for shutdown"); |
|
79 OutputAnalyzer output = ProcessTools.executeTestJvm( |
|
80 "-classpath", |
|
81 System.getProperty("test.class.path", "."), |
|
82 "Shutdown", |
|
83 Integer.toString(port)); |
|
84 // Verify that both the Shutdown command and the Application finished ok. |
|
85 output.shouldHaveExitValue(0); |
|
86 processThread.joinAndThrow(); |
|
87 processThread.getOutput().shouldHaveExitValue(0); |
|
88 } catch (Throwable t) { |
|
89 System.out.println("RunnerUtil.stopApplication failed. Will kill it hard: " + t); |
|
90 processThread.stopProcess(); |
|
91 throw t; |
|
92 } |
|
93 } |
|
94 |
|
95 /** |
|
96 * Creates a jar file. |
|
97 * @param args Command to the jar tool. |
|
98 */ |
|
99 public static void createJar(String... args) { |
|
100 System.out.println("Running: jar " + Arrays.toString(args)); |
|
101 sun.tools.jar.Main jar = new sun.tools.jar.Main(System.out, System.err, "jar"); |
|
102 if (!jar.run(args)) { |
|
103 throw new RuntimeException("jar failed: args=" + Arrays.toString(args)); |
|
104 } |
|
105 } |
|
106 |
|
107 /** |
|
108 * Read process info for the running Application. |
|
109 * The Application writes its info to a file with this format: |
|
110 * shutdownPort=42994 |
|
111 * pid=19597 |
|
112 * done |
|
113 * |
|
114 * The final "done" is used to make sure the complete file has been written |
|
115 * before we try to read it. |
|
116 * This function will wait until the file is available. |
|
117 * |
|
118 * @param filename Path to file to read. |
|
119 * @return The ProcessInfo containing pid and shutdownPort. |
|
120 */ |
|
121 public static ProcessInfo readProcessInfo(String filename) throws Throwable { |
|
122 System.out.println("Reading port and pid from file: " + filename); |
|
123 File file = new File(filename); |
|
124 String content = null; |
|
125 |
|
126 // Read file or wait for it to be created. |
|
127 while (true) { |
|
128 content = readFile(file); |
|
129 if (content != null && content.indexOf("done") >= 0) { |
|
130 break; |
|
131 } |
|
132 Thread.sleep(100); |
|
133 } |
|
134 |
|
135 ProcessInfo info = new ProcessInfo(); |
|
136 // search for a line with format: key=nnn |
|
137 Pattern pattern = Pattern.compile("(\\w*)=([0-9]+)\\r?\\n"); |
|
138 Matcher matcher = pattern.matcher(content); |
|
139 while (matcher.find()) { |
|
140 String key = matcher.group(1); |
|
141 int value = Integer.parseInt(matcher.group(2)); |
|
142 if ("pid".equals(key)) { |
|
143 info.pid = value; |
|
144 } else if ("shutdownPort".equals(key)) { |
|
145 info.shutdownPort = value; |
|
146 } |
|
147 } |
|
148 System.out.println("processInfo.pid:" + info.pid); |
|
149 System.out.println("processInfo.shutdownPort:" + info.shutdownPort); |
|
150 return info; |
|
151 } |
|
152 |
|
153 /** |
|
154 * Read the content of a file. |
|
155 * @param file The file to read. |
|
156 * @return The file content or null if file does not exists. |
|
157 */ |
|
158 public static String readFile(File file) throws IOException { |
|
159 if (!file.exists()) { |
|
160 return null; |
|
161 } |
|
162 try { |
|
163 byte[] bytes = Files.readAllBytes(file.toPath()); |
|
164 String content = new String(bytes); |
|
165 return content; |
|
166 } catch (IOException e) { |
|
167 e.printStackTrace(); |
|
168 throw e; |
|
169 } |
|
170 } |
|
171 |
|
172 /** |
|
173 * Helper class with info of the running Application. |
|
174 */ |
|
175 public static class ProcessInfo { |
|
176 public int pid = -1; |
|
177 public int shutdownPort = -1; |
|
178 } |
|
179 |
|
180 } |