$darkmode
DENOPTIM
Main.java
Go to the documentation of this file.
1/*
2 * DENOPTIM
3 * Copyright (C) 2022 Marco Foscato <marco.foscato@uib.no>
4 *
5 * This program is free software: you can redistribute it and/or modify
6 * it under the terms of the GNU Affero General Public License as published
7 * by the Free Software Foundation, either version 3 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU Affero General Public License for more details.
14 *
15 * You should have received a copy of the GNU Affero General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
17 */
18
19package denoptim.main;
20
21import java.awt.EventQueue;
22import java.io.File;
23import java.io.PrintWriter;
24import java.io.StringWriter;
25import java.lang.reflect.Constructor;
26import java.util.List;
27import java.util.concurrent.ExecutionException;
28
29import org.apache.commons.cli.CommandLine;
30import org.apache.commons.cli.CommandLineParser;
31import org.apache.commons.cli.DefaultParser;
32import org.apache.commons.cli.HelpFormatter;
33import org.apache.commons.cli.ParseException;
34
35import denoptim.constants.DENOPTIMConstants;
36import denoptim.exception.ExceptionUtils;
37import denoptim.files.FileFormat;
38import denoptim.files.FileUtils;
39import denoptim.gui.GUI;
40import denoptim.integration.python.Py4JGetawayServer;
41import denoptim.logging.Version;
42import denoptim.programs.combinatorial.FragSpaceExplorer;
43import denoptim.programs.denovo.GARunner;
44import denoptim.programs.fitnessevaluator.FitnessRunner;
45import denoptim.programs.fragmenter.Fragmenter;
46import denoptim.programs.genetweeker.GeneOpsRunner;
47import denoptim.programs.grapheditor.GraphEditor;
48import denoptim.programs.graphlisthandler.GraphListsHandler;
49import denoptim.programs.isomorphism.Isomorphism;
50import denoptim.programs.mol2graph.Mol2Graph;
51import denoptim.programs.moldecularmodelbuilder.MolecularModelBuilder;
52import denoptim.task.ProgramTask;
53import denoptim.task.StaticTaskManager;
54
58public class Main
59{
63 public static enum RunType {
68
73
79
84
90
95
100
105
110
115
120
125
130
131 // NB: to define a new run type:
132 // 1) add the enum alternative. The order is somewhat related to the
133 // importance (i.e., common use) of a run type.
134 // 2) set the value of "description" in the static block below
135 // 3) set the value of "isCLIEnabled" in the static block below
136 // 4) set the value of "needsInputFile" in the static block below
137 // 5) set the value of "programTaskImpl" in the static block below
138 // 6) consider whether a new parameter file format should be added
139 // in FileFormats.
140
144 private Class<?> programTaskImpl;
145
149 private String description = "";
150
154 private boolean isCLIEnabled = true;
155
160 private boolean needsInputFile = false;
161
162 static {
163 DRY.description = "Dry run";
164 FSE.description = "Combinatorial Fragment Space Exploration";
165 FIT.description = "Stand-alone FITness evaluation";
166 GA.description = "Genetic Algorithm";
167 GE.description = "Stand-alone Graph Editing task";
168 GI.description = "Stand-alone Graph Isomorphism analysis";
169 GO.description = "Stand-alone Genetic Operation";
170 GUI.description = "Graphycal User Interface";
171 CLG.description = "Comparison of Lists of Graphs";
172 B3D.description = "Stand-alone build a 3D molecular model from a "
173 + "graph";
174 FRG.description = "Fragmentation and managment of fragments";
175 M2G.description = "Convert Molecules to Graphs.";
176 PY4J.description = "Starts a server listening to Python";
177
178 DRY.isCLIEnabled = false;
179 FSE.isCLIEnabled = true;
180 FIT.isCLIEnabled = true;
181 GA.isCLIEnabled = true;
182 GE.isCLIEnabled = true;
183 GI.isCLIEnabled = true;
184 GO.isCLIEnabled = true;
185 GUI.isCLIEnabled = false;
186 CLG.isCLIEnabled = true;
187 B3D.isCLIEnabled = true;
188 FRG.isCLIEnabled = true;
189 M2G.isCLIEnabled = true;
190 PY4J.isCLIEnabled = true;
191
192 DRY.needsInputFile = false;
193 FSE.needsInputFile = true;
194 FIT.needsInputFile = true;
195 GA.needsInputFile = true;
196 GE.needsInputFile = true;
197 GI.needsInputFile = true;
198 GO.needsInputFile = true;
199 GUI.needsInputFile = false;
200 CLG.needsInputFile = true;
201 B3D.needsInputFile = true;
202 FRG.needsInputFile = true;
203 M2G.needsInputFile = true;
204 PY4J.needsInputFile = false;
205
206 DRY.programTaskImpl = null;
207 FSE.programTaskImpl = FragSpaceExplorer.class;
208 FIT.programTaskImpl = FitnessRunner.class;
209 GA.programTaskImpl = GARunner.class;
210 GE.programTaskImpl = GraphEditor.class;
211 GI.programTaskImpl = Isomorphism.class;
212 GO.programTaskImpl = GeneOpsRunner.class;
213 GUI.programTaskImpl = GUI.class;
214 CLG.programTaskImpl = GraphListsHandler.class;
215 B3D.programTaskImpl = MolecularModelBuilder.class;
216 FRG.programTaskImpl = Fragmenter.class;
217 M2G.programTaskImpl = Mol2Graph.class;
218 PY4J.programTaskImpl = null;
219 }
220
225 public Class<?> getProgramTaskImpl()
226 {
227 return programTaskImpl;
228 }
229
235 public static String getRunTypesForUser()
236 {
237 String s = "";
238 String separator = DENOPTIMConstants.EOL;
239 String last = ""; //in case the last entry should use " and "
240 for (int i=0; i<RunType.values().length; i++)
241 {
242 RunType rt = RunType.values()[i];
243 if (!rt.isCLIEnabled)
244 continue; //This run type should not be visible by the user
245 if (i>0 && i < RunType.values().length-2)
246 s = s + " '" + rt.toString() + "' for "+rt.description+ ",";
247 else if (i==0)
248 s = "'" + rt.toString() + "' for "+rt.description+",";
249 else
250 s = s + last+"'"+rt.toString()+"' for "+rt.description+".";
251 s = s + separator;
252 }
253 return s;
254 }
255
262 boolean isCLIEnabled()
263 {
264 return isCLIEnabled;
265 }
266
274 {
275 return needsInputFile;
276 }
277 };
278
279//------------------------------------------------------------------------------
280
287 public static void main(String[] args)
288 {
289 // First, we try to understand what the user wants to do.
290 Behavior behavior = defineProgramBehavior(args);
291
292 // In case of inconsistent requests, we report the error and stop
293 if (behavior.exitStatus!=0)
294 reportError(behavior);
295
296 // Deal with simple requests for information
297 if (RunType.DRY.equals(behavior.runType))
298 {
299 System.out.println(behavior.helpMsg);
300 System.exit(0);
301 }
302
303 // Deal with deamon/server launches
304 if (RunType.PY4J.equals(behavior.runType))
305 {
306 try
307 {
309 } catch (Throwable e)
310 {
311 e.printStackTrace();
312 reportError("Could not start J2PyServer. Hint on problem: "
313 + e.getMessage(), 1);
314 }
315 return;
316 }
317
318 List<String> inputFiles = behavior.cmd.getArgList();
319
320 // We instantiate also the task manager, even if it might not be used.
321 // This is to pre-start the tasks and get a more reliable queue status
322 // at any given time after this point.
324
325 // Now, we deal with proper program runs
326 if (ProgramTask.class.isAssignableFrom(
327 behavior.runType.getProgramTaskImpl()))
328 {
329 File inpFile = null;
330 File wDir = null;
331 if (inputFiles.size()>1)
332 {
333 reportError("Only one input file allowed when requesting run "
334 + behavior.runType + ". Found " + inputFiles.size()
335 + " files: " + inputFiles, 1);
336 }
337 if (behavior.runType.needsInputFile())
338 {
339 if (inputFiles.size()<1)
340 {
341 reportError("Need an input file when requesting run "
342 + behavior.runType + ". Found " + inputFiles.size(), 1);
343 } else if (inputFiles.size()==1) {
344 inpFile = new File(inputFiles.get(0));
345 wDir = inpFile.getParentFile();
346 if (wDir==null)
347 {
348 wDir = new File(System.getProperty("user.dir"));
349 }
350 }
351 }
352 runProgramTask(behavior.runType.getProgramTaskImpl(), inpFile, wDir);
353 terminate();
354 } else if (RunType.GUI.equals(behavior.runType)) {
355 EventQueue.invokeLater(new GUI(behavior.cmd));
356 } else {
357 reportError("You requested a run of type '" + behavior.runType
358 + "', but I found no such implementation. "
359 + "Please, report this to the authors.", 1);
360 }
361 }
362
363//------------------------------------------------------------------------------
364
373 private static void runProgramTask(Class<?> taskClass, File inputFile,
374 File workDir)
375 {
376 if (!ProgramTask.class.isAssignableFrom(taskClass))
377 {
378 reportError("Attempt to create a program task for class '"
379 + taskClass.getSimpleName() + "', but such class is not a "
380 + "extension of '" + ProgramTask.class.getSimpleName()
381 + "'.", 1);
382 }
383
384 ProgramTask task = null;
385 try
386 {
387 Constructor<?> taskBuilder = taskClass.getConstructor(File.class,
388 File.class);
389 task = (ProgramTask) taskBuilder.newInstance(inputFile,workDir);
390 } catch (Exception e)
391 {
392 reportError("Could not create a program task for "
393 + taskClass.getSimpleName() + DENOPTIMConstants.EOL
394 + ". Details: " + DENOPTIMConstants.EOL
396 }
397
398 try
399 {
401 } catch (InterruptedException e)
402 {
403 e.printStackTrace();
404 } catch (ExecutionException e)
405 {
406 e.printStackTrace();
407 }
408 }
409
410//------------------------------------------------------------------------------
411
417 private static Behavior ensureFileExistsAndIsReadable(File file,
418 Behavior behavior)
419 {
420 if (!file.exists())
421 {
422 return new Behavior(behavior.runType, behavior.cmd, 1, null,
423 "File '"+ file.getAbsolutePath() +"' not found.");
424 }
426 try {
427 format = FileUtils.detectFileFormat(file);
428 } catch (Throwable t) {
429 // Ignore: if anything goes wrong, will still get UNRECOGNIZED which
430 // triggers the error.
431 }
432 if (FileFormat.UNRECOGNIZED.equals(format))
433 {
434 return new Behavior(behavior.runType, behavior.cmd, 1, null,
435 "Could not open file '" + file.getAbsolutePath()
436 + "' because its format is not recognized.");
437 }
438 return behavior;
439 }
440
441//------------------------------------------------------------------------------
442
450 protected static Behavior defineProgramBehavior(String[] args)
451 {
452 CommandLineParser parser = new DefaultParser();
453 CommandLine cmd = null;
454
455 try {
456 cmd = parser.parse(CLIOptions.getInstance(), args);
457 } catch (ParseException e) {
458 String helpMsg = getHelpString();
459 String errMsg = "Unable to parse command-line arguments. "
460 + "Please, check your input! " + DENOPTIMConstants.EOL
461 + "Details: " + e.getMessage();
462 return new Behavior(null, null, 1, helpMsg, errMsg);
463 }
464
465 if (cmd.hasOption(CLIOptions.help))
466 {
467 String helpMsg = getHelpString();
468 return new Behavior(RunType.DRY, null, 0, helpMsg, null);
469 }
470
471 if (cmd.hasOption(CLIOptions.version))
472 {
473 return new Behavior(RunType.DRY, null, 0, Version.VERSION, null);
474 }
475
476 Behavior result = null;
477 if (cmd.getOptions().length==0)
478 result = new Behavior(RunType.GUI, cmd, 0, null, null);
479 else {
480 if (cmd.hasOption(CLIOptions.run))
481 {
482 String rts = cmd.getOptionValue(CLIOptions.run).toString();
483 RunType rt = null;
484 try
485 {
486 rt = RunType.valueOf(rts.toUpperCase());
487 } catch (Exception e)
488 {
489 String errMsg = "Unacceptable value for "
490 + CLIOptions.run.getLongOpt() + " option: "
491 + "'" + rts + "'. Please, modify "
492 + "your command. ";
493 return new Behavior(null, null, 1, null, errMsg);
494 }
495
496 if (rt.isCLIEnabled())
497 {
498 result = new Behavior(rt, cmd);
499 } else {
500 result = new Behavior(null, null, 1, null,
501 "RunType '"+rt+"' is not enabled from CLI. "
502 + "Please, contacts the developers.");
503 }
504 } else {
505 reportError("Command line has no " + CLIOptions.run + " option, "
506 + "but more than zero options. Please, contact the "
507 + "developers.", 1);
508 }
509 }
510
511 List<String> inputFiles = cmd.getArgList();
512 for (String pathname : inputFiles)
513 result = ensureFileExistsAndIsReadable(new File(pathname), result);
514
515 return result;
516 }
517
518//------------------------------------------------------------------------------
519
524 private static String getHelpString()
525 {
526 HelpFormatter formatter = new HelpFormatter();
527 StringWriter out = new StringWriter();
528 PrintWriter pw = new PrintWriter(out);
529
530 formatter.printHelp(pw, 80, "denoptim", DENOPTIMConstants.EOL,
532 formatter.getLeftPadding(), formatter.getDescPadding(),
534 "Run without arguments to launch the graphical user "
535 + "interface (GUI) without opening any file. ", true);
536 pw.flush();
537 return out.toString();
538 }
539
540//------------------------------------------------------------------------------
541
549 private static void reportError(String msg, int exitCode)
550 {
551 System.out.println(DENOPTIMConstants.EOL
552 + "ERROR! " + msg
554 System.exit(exitCode);
555 }
556
557//------------------------------------------------------------------------------
558
563 private static void reportError(Behavior behavior)
564 {
565 if (behavior.helpMsg!=null)
566 System.out.println(behavior.helpMsg);
567 reportError(behavior.errorMsg, behavior.exitStatus);
568 }
569
570//------------------------------------------------------------------------------
571
575 private static void terminate()
576 {
577 try {
579 } catch (Exception e) {
580 e.printStackTrace();
581 System.out.println("StaticTaskManager had problems stopping. "
582 + "Forcing termination.");
583 }
584 Runtime.getRuntime().halt(0);
585 }
586
587//------------------------------------------------------------------------------
588
589}
General set of constants used in DENOPTIM.
static final String EOL
new line character
static String getStackTraceAsString(Throwable t)
Prints the stack trace of an exception into a string.
static FileFormat detectFileFormat(File inFile)
Inspects a file/folder and tries to detect if there is one of the data sources that is recognized by ...
Definition: FileUtils.java:399
Graphical User Interface of the DENOPTIM package.
Definition: GUI.java:57
A tool that start a Py4J gateway server that can listens to calls from Python and translate JAVA obje...
static void launch()
Starts a gateway server using this class as entry point, which then becomes the interpreter of any da...
Class handling DENOPTIM's version identifier for headers.
Definition: Version.java:36
static final String VERSION
Version identifier (from pom.xml via Maven properties)
Definition: Version.java:48
Represents the behavior of the program at start-up.
Definition: Behavior.java:31
RunType runType
The type of run that is requested.
Definition: Behavior.java:51
String errorMsg
The error message.
Definition: Behavior.java:40
CommandLine cmd
The parsed command line arguments.
Definition: Behavior.java:56
int exitStatus
A non-zero value means some error has occurred and the program will terminate.
Definition: Behavior.java:46
String helpMsg
The help message.
Definition: Behavior.java:35
static Option help
Option requesting the printing of the help message.
Definition: CLIOptions.java:37
static CLIOptions getInstance()
Gets the singleton instance of this class.
Definition: CLIOptions.java:79
static Option version
Option requesting only the printing of the version.
Definition: CLIOptions.java:42
static Option run
Option controlling the type of run.
Definition: CLIOptions.java:47
Entry point of any kind of run of the DENOPTIM program.
Definition: Main.java:59
static void reportError(Behavior behavior)
Prints an error message on standard output, possibly after the help message, and then stops the Virtu...
Definition: Main.java:563
static Behavior ensureFileExistsAndIsReadable(File file, Behavior behavior)
Checks the file exists and format is recognized.
Definition: Main.java:417
static String getHelpString()
Prints the help message on a string.
Definition: Main.java:524
static void main(String[] args)
Launches the appropriate program according to the arguments given.
Definition: Main.java:287
static void terminate()
Stops services and halts the Virtual Machine.
Definition: Main.java:575
static void runProgramTask(Class<?> taskClass, File inputFile, File workDir)
Creates a task for the given class.
Definition: Main.java:373
static void reportError(String msg, int exitCode)
Prints an error message on standard output and then stops the Virtual Machine with an exit code.
Definition: Main.java:549
static Behavior defineProgramBehavior(String[] args)
Does the processing of the application arguments and decides what the program should do.
Definition: Main.java:450
Combinatorial exploration of the fragment space.
Programs that runs de novo design by a genetic algorithm.
Definition: GARunner.java:40
Tool to create fragments by chopping 2D/3D chemical structures.
Definition: Fragmenter.java:39
Tool to run genetic operations in a stand-alone fashion, i.e., outside of a genetic algorithm run.
Tool to perform isomorphism analysis on DGraphs.
Tool for creating DGraphs from molecules.
Definition: Mol2Graph.java:48
Task structure for any of the main programs in the denoptim project, such as genetic algorithm and co...
Manager for tasks submitted by the GUI.
static void submitAndWait(Task task)
Submits a task and waits for completion.
static StaticTaskManager getInstance()
Gets the singleton instance of this class.
File formats identified by DENOPTIM.
Definition: FileFormat.java:32
Types of runs that can be requested to the DENOPTIM Main class.
Definition: Main.java:63
String description
A short description of the run type.
Definition: Main.java:149
Class<?> programTaskImpl
The implementation of ProgramTask capable of this run type.
Definition: Main.java:144
CLG
Run a comparison of lists of graphs.
Definition: Main.java:114
GI
Run a stand-alone test for graph isomorphism.
Definition: Main.java:109
B3D
Run the stand-alone conversion of graph into a three-dimensional molecular model.
Definition: Main.java:89
GO
Run a stand-alone genetic operation.
Definition: Main.java:99
GE
Run a stand-alone graph editing task.
Definition: Main.java:104
GA
Run the genetic algorithm with DenoptimGA.
Definition: Main.java:72
GUI
Launch the graphical user interface denoptim.gui.GUI.
Definition: Main.java:67
FIT
stand-alone fitness evaluation
Definition: Main.java:83
boolean needsInputFile
Flag indicating if this run type needs any input file when called from CLI.
Definition: Main.java:160
boolean isCLIEnabled()
Returns true if this run type can be requested from the CLI.
Definition: Main.java:262
PY4J
Starts a listener to Python;.
Definition: Main.java:129
Class<?> getProgramTaskImpl()
Returns the class that implements the program with this type of run.
Definition: Main.java:225
FRG
Run a fragmentation task.
Definition: Main.java:119
M2G
Run a conversion of molecules to graphs.
Definition: Main.java:124
static String getRunTypesForUser()
Returns a string that contains a textual list (e.g., "A, B, and C") of the possible types that can be...
Definition: Main.java:235
DRY
Only prints help or version and close program.
Definition: Main.java:94
FSE
Run a combinatorial generation of candidates with FragSpaceExplorer.
Definition: Main.java:78
boolean needsInputFile()
Returns true if this run type needs an input file when requested from CLI.
Definition: Main.java:273
boolean isCLIEnabled
Flag indicating if this run type can be called from CLI.
Definition: Main.java:154