Skip to contentMethod: forExecutable(String)
1: /*
2: * *************************************************************************************************************************************************************
3: *
4: * TheseFoolishThings: Miscellaneous utilities
5: * http://tidalwave.it/projects/thesefoolishthings
6: *
7: * Copyright (C) 2009 - 2025 by Tidalwave s.a.s. (http://tidalwave.it)
8: *
9: * *************************************************************************************************************************************************************
10: *
11: * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License.
12: * You may obtain a copy of the License at
13: *
14: * http://www.apache.org/licenses/LICENSE-2.0
15: *
16: * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
17: * CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
18: *
19: * *************************************************************************************************************************************************************
20: *
21: * git clone https://bitbucket.org/tidalwave/thesefoolishthings-src
22: * git clone https://github.com/tidalwave-it/thesefoolishthings-src
23: *
24: * *************************************************************************************************************************************************************
25: */
26: package it.tidalwave.util.spi;
27:
28: import javax.annotation.concurrent.ThreadSafe;
29: import jakarta.annotation.Nonnull;
30: import jakarta.annotation.Nullable;
31: import java.util.ArrayList;
32: import java.util.Collections;
33: import java.util.List;
34: import java.util.Scanner;
35: import java.util.concurrent.ExecutorService;
36: import java.util.concurrent.Executors;
37: import java.util.concurrent.atomic.AtomicBoolean;
38: import java.util.concurrent.atomic.AtomicInteger;
39: import java.util.regex.Pattern;
40: import java.io.File;
41: import java.io.IOException;
42: import java.io.InputStream;
43: import java.io.InputStreamReader;
44: import java.io.PrintWriter;
45: import it.tidalwave.util.ProcessExecutor;
46: import lombok.AccessLevel;
47: import lombok.Getter;
48: import lombok.NoArgsConstructor;
49: import lombok.RequiredArgsConstructor;
50: import lombok.Setter;
51: import lombok.extern.slf4j.Slf4j;
52:
53: /***************************************************************************************************************************************************************
54: *
55: * @author Fabrizio Giudici
56: * @since 1.39
57: *
58: **************************************************************************************************************************************************************/
59: @ThreadSafe @NoArgsConstructor(access=AccessLevel.PRIVATE) @Slf4j
60: public class DefaultProcessExecutor implements ProcessExecutor
61: {
62: private final ExecutorService executorService = Executors.newFixedThreadPool(10);
63:
64: /***********************************************************************************************************************************************************
65: **********************************************************************************************************************************************************/
66: @RequiredArgsConstructor(access=AccessLevel.PACKAGE)
67: public class DefaultConsoleOutput implements ConsoleOutput
68: {
69: @Nonnull
70: private final String name;
71:
72: @Nonnull
73: private final InputStream input;
74:
75: @Getter
76: private final List<String> content = Collections.synchronizedList(new ArrayList<>());
77:
78: private volatile String latestLine;
79:
80: private final AtomicInteger li = new AtomicInteger(0);
81:
82: private final AtomicBoolean started = new AtomicBoolean();
83:
84: @Nullable @Setter @Getter
85: private Listener listener;
86:
87: /***************************************************************************************************************
88: *
89: *
90: ***************************************************************************************************************/
91: private final Runnable reader = new Runnable()
92: {
93: @Override
94: public void run()
95: {
96: try
97: {
98: read();
99: }
100: catch (IOException e)
101: {
102: log.warn("while reading from " + name, e);
103: }
104: }
105: };
106:
107: /***************************************************************************************************************
108: *
109: *
110: ***************************************************************************************************************/
111: private final Runnable logger = new Runnable()
112: {
113: @Override
114: public void run()
115: {
116: var l = 0;
117:
118: for (;;)
119: {
120: try
121: {
122: if ((l != li.get()) && (latestLine != null))
123: {
124: log.trace(">>>>>>>> {} {}", name, latestLine);
125: }
126:
127: l = li.get();
128: Thread.sleep(500);
129: }
130: catch (Throwable e)
131: {
132: return;
133: }
134: }
135: }
136: };
137:
138: /***************************************************************************************************************
139: *
140: * Should not be used by the programmer.
141: *
142: * @return -
143: *
144: ***************************************************************************************************************/
145: @Nonnull
146: public ConsoleOutput start()
147: {
148: if (started.getAndSet(true))
149: {
150: throw new IllegalStateException("Already started");
151: }
152:
153: log.info("{} - started", name);
154: executorService.submit(reader);
155: executorService.submit(logger);
156: return this;
157: }
158:
159: /***************************************************************************************************************
160: *
161: * {@inheritDoc}
162: *
163: ***************************************************************************************************************/
164: @Override
165: public boolean latestLineMatches (@Nonnull final String regexp)
166: {
167: String s = null;
168:
169: if (latestLine != null)
170: {
171: s = latestLine;
172: }
173: else if (!content.isEmpty())
174: {
175: s = content.get(content.size() - 1);
176: }
177:
178: log.trace(">>>> testing '{}' for '{}'", s, regexp);
179: return (s != null) && Pattern.compile(regexp).matcher(s).matches();
180: // FIXME: sync
181: }
182:
183: /***************************************************************************************************************
184: *
185: * {@inheritDoc}
186: *
187: ***************************************************************************************************************/
188: @Override @Nonnull
189: public Scanner filteredAndSplitBy (@Nonnull final String filterRegexp, @Nonnull final String delimiterRegexp)
190: {
191: final var string = filteredBy(filterRegexp).get(0);
192: return new Scanner(string).useDelimiter(Pattern.compile(delimiterRegexp));
193: }
194:
195: /***************************************************************************************************************
196: *
197: * {@inheritDoc}
198: *
199: ***************************************************************************************************************/
200: @Override @Nonnull
201: public List<String> filteredBy (@Nonnull final String regexp)
202: {
203: final var pattern = Pattern.compile(regexp);
204: final List<String> result = new ArrayList<>();
205: final var strings = new ArrayList<>(content);
206:
207: // TODO: sync
208: if (latestLine != null)
209: {
210: strings.add(latestLine);
211: }
212:
213: for (final var s : strings)
214: {
215: // log.trace(">>>>>>>> matching '{}' with '{}'...", s, filter);
216: final var m = pattern.matcher(s);
217:
218: if (m.matches())
219: {
220: result.add(m.group(1));
221: }
222: }
223:
224: return result;
225: }
226:
227: /***************************************************************************************************************
228: *
229: * {@inheritDoc}
230: *
231: ***************************************************************************************************************/
232: @Override @Nonnull
233: public ConsoleOutput waitFor (@Nonnull final String regexp)
234: throws InterruptedException, IOException
235: {
236: log.debug("{} - waitFor({})", name, regexp);
237:
238: while (filteredBy(regexp).isEmpty())
239: {
240: try
241: {
242: final var exitValue = process.exitValue();
243: throw new IOException("Process exited with " + exitValue);
244: }
245: catch (IllegalThreadStateException e) // ok, process not terminated yet
246: {
247: synchronized (this)
248: {
249: wait(50); // FIXME: polls because it doesn't get notified
250: }
251: }
252: }
253:
254: return this;
255: }
256:
257: /***************************************************************************************************************
258: *
259: * {@inheritDoc}
260: *
261: ***************************************************************************************************************/
262: @Override
263: public void clear()
264: {
265: content.clear();
266: latestLine = null;
267: }
268:
269: /***************************************************************************************************************
270: *
271: *
272: ***************************************************************************************************************/
273: private void read()
274: throws IOException
275: {
276: try (final var is = new InputStreamReader(input))
277: {
278: var l = new StringBuilder();
279:
280: for (;;)
281: {
282: final var c = is.read();
283:
284: if (c < 0)
285: {
286: break;
287: }
288:
289: // if (c == 10)
290: // {
291: // continue;
292: // }
293:
294: if ((c == 13) || (c == 10))
295: {
296: latestLine = l.toString();
297: li.incrementAndGet();
298: content.add(latestLine);
299: l = new StringBuilder();
300: log.trace(">>>>>>>> {} {}", name, latestLine);
301:
302: if (listener != null)
303: {
304: listener.onReceived(latestLine);
305: }
306: }
307: else
308: {
309: l.append((char)c);
310: latestLine = l.toString();
311: li.incrementAndGet();
312: }
313:
314: synchronized (this)
315: {
316: notifyAll();
317: }
318: }
319:
320: log.debug(">>>>>> {} closed", name);
321: }
322: }
323: }
324:
325: private final List<String> arguments = new ArrayList<>();
326:
327: private Process process;
328:
329: @Getter
330: private ConsoleOutput stdout;
331:
332: @Getter
333: private ConsoleOutput stderr;
334:
335: private PrintWriter stdin;
336:
337: /***********************************************************************************************************************************************************
338: * Factory method for associating an executable. It returns an intermediate executor that must be configured and
339: * later started. Under Windows, the '.exe' suffix is automatically appended to the name of the executable.
340: *
341: * @see #start()
342: *
343: * @param executable the executable (with the full path)
344: * @return the executor
345: **********************************************************************************************************************************************************/
346: @Nonnull
347: public static DefaultProcessExecutor forExecutable (@Nonnull final String executable)
348: {
349: final var executor = new DefaultProcessExecutor();
350:• executor.arguments.add(new File(executable + (isWindows() ? ".exe" : "")).getAbsolutePath());
351: return executor;
352: }
353:
354: // /*******************************************************************************************************************
355: // *
356: // *
357: // ******************************************************************************************************************/
358: // @Nonnull
359: // private static String findPath (final @Nonnull String executable)
360: // throws NotFoundException
361: // {
362: // for (final String path : System.getenv("PATH").split(File.pathSeparator))
363: // {
364: // final File file = new File(new File(path), executable);
365: //
366: // if (file.canExecute())
367: // {
368: // return file.getAbsolutePath();
369: // }
370: // }
371: //
372: // throw new NotFoundException("Can't find " + executable + " in PATH");
373: // }
374:
375: /***********************************************************************************************************************************************************
376: * {@inheritDoc}
377: **********************************************************************************************************************************************************/
378: @Override @Nonnull
379: public DefaultProcessExecutor withArgument (@Nonnull final String argument)
380: {
381: arguments.add(argument);
382: return this;
383: }
384:
385: /***********************************************************************************************************************************************************
386: * {@inheritDoc}
387: **********************************************************************************************************************************************************/
388: @Override @Nonnull
389: public DefaultProcessExecutor withArguments (@Nonnull final String ... arguments)
390: {
391: this.arguments.addAll(List.of(arguments));
392: return this;
393: }
394:
395: /***********************************************************************************************************************************************************
396: * {@inheritDoc}
397: **********************************************************************************************************************************************************/
398: @Override @Nonnull
399: public DefaultProcessExecutor start()
400: throws IOException
401: {
402: log.info(">>>> executing {} ...", arguments);
403:
404: final List<String> environment = new ArrayList<>();
405:
406: for (final var e : System.getenv().entrySet())
407: {
408: environment.add(String.format("%s=%s", e.getKey(), e.getValue()));
409: }
410:
411: log.info(">>>> environment: {}", environment);
412: process = Runtime.getRuntime().exec(arguments.toArray(new String[0]),
413: environment.toArray(new String[0]));
414:
415: stdout = new DefaultConsoleOutput("out", process.getInputStream()).start();
416: stderr = new DefaultConsoleOutput("err", process.getErrorStream()).start();
417: stdin = new PrintWriter(process.getOutputStream(), true);
418:
419: return this;
420: }
421:
422: /***********************************************************************************************************************************************************
423: * {@inheritDoc}
424: **********************************************************************************************************************************************************/
425: @Override
426: public void stop()
427: {
428: log.info("stop()");
429: process.destroy();
430: executorService.shutdownNow();
431: }
432:
433: /***********************************************************************************************************************************************************
434: * {@inheritDoc}
435: **********************************************************************************************************************************************************/
436: @Override @Nonnull
437: public DefaultProcessExecutor waitForCompletion()
438: throws InterruptedException
439: {
440: if (process.waitFor() != 0)
441: {
442: // throw new IOException("Process exited with " + process.exitValue()); FIXME
443: }
444:
445: return this;
446: }
447:
448: /***********************************************************************************************************************************************************
449: * {@inheritDoc}
450: **********************************************************************************************************************************************************/
451: @Override @Nonnull
452: public DefaultProcessExecutor send (@Nonnull final String string)
453: {
454: log.debug(">>>> sending '{}'...", string.replaceAll("\n", "<CR>"));
455: stdin.print(string);
456: stdin.flush();
457: return this;
458: }
459:
460: /***********************************************************************************************************************************************************
461: **********************************************************************************************************************************************************/
462: private static boolean isWindows()
463: {
464: return System.getProperty ("os.name").toLowerCase().startsWith("windows");
465: }
466: }