diff --git a/README.md b/README.md
index 8220b86d49..dc46076afa 100644
--- a/README.md
+++ b/README.md
@@ -191,20 +191,6 @@ or in offline mode:
./emissary config --place emissary.place.sample.ToLowerPlace --offline --detailed
```
-#### Run
-
-The Run command is a simple command to execute the main method of the given class. For example
-
-```
-./emissary run emissary.config.ConfigUtil
-```
-
-If you need to pass flags to the main method, use *--* to stop parsing flags and simply pass them along.
-
-```
-./emissary run emissary.config.ExtractResource -- -o outputdir somefile
-```
-
#### Server (Cluster)
Emissary is fun in standalone, but running cluster is more appropriate for real work. The way to run clustered
diff --git a/pom.xml b/pom.xml
index 9e05e52b4d..b8a3ec41a6 100644
--- a/pom.xml
+++ b/pom.xml
@@ -46,7 +46,6 @@
${project.version}
${project.basedir}/contrib/checkstyle.xml
- 1.5.0
1.15
4.4
1.21
@@ -145,11 +144,6 @@
guava
${dep.guava.version}
-
- commons-cli
- commons-cli
- ${dep.commons-cli.version}
-
commons-codec
commons-codec
@@ -350,10 +344,6 @@
com.google.guava
guava
-
- commons-cli
- commons-cli
-
commons-codec
commons-codec
diff --git a/src/main/java/emissary/Emissary.java b/src/main/java/emissary/Emissary.java
index 15824eae15..a70deec1cb 100644
--- a/src/main/java/emissary/Emissary.java
+++ b/src/main/java/emissary/Emissary.java
@@ -10,7 +10,6 @@
import emissary.command.HelpCommand;
import emissary.command.PeersCommand;
import emissary.command.PoolCommand;
-import emissary.command.RunCommand;
import emissary.command.ServerCommand;
import emissary.command.TopologyCommand;
import emissary.command.VersionCommand;
@@ -56,7 +55,7 @@ public class Emissary {
static {
List> cmds =
Arrays.asList(ServerCommand.class, HelpCommand.class, TopologyCommand.class, FeedCommand.class,
- AgentsCommand.class, PoolCommand.class, VersionCommand.class, RunCommand.class, EnvCommand.class,
+ AgentsCommand.class, PoolCommand.class, VersionCommand.class, EnvCommand.class,
PeersCommand.class, ConfigCommand.class, DirectoryCommand.class);
Map staticCopy = new HashMap<>();
for (Class extends EmissaryCommand> clz : cmds) {
diff --git a/src/main/java/emissary/command/RunCommand.java b/src/main/java/emissary/command/RunCommand.java
deleted file mode 100644
index 9a902cea0a..0000000000
--- a/src/main/java/emissary/command/RunCommand.java
+++ /dev/null
@@ -1,76 +0,0 @@
-package emissary.command;
-
-import org.slf4j.Logger;
-import org.slf4j.LoggerFactory;
-import picocli.CommandLine;
-import picocli.CommandLine.Command;
-import picocli.CommandLine.Parameters;
-
-import java.lang.reflect.InvocationTargetException;
-import java.lang.reflect.Method;
-import java.util.ArrayList;
-import java.util.Arrays;
-import java.util.List;
-
-@Deprecated
-@Command(description = "Run arbitrary class with optional args", subcommands = {HelpCommand.class})
-public class RunCommand extends BaseCommand {
-
- private static final Logger LOG = LoggerFactory.getLogger(RunCommand.class);
-
- @Parameters(
- arity = "1..*",
- description = "fully qualified class name to run with remaining arguments passed on as args to that classes main method. Use -- to stop processing strings as args and pass them along.")
- public List args = new ArrayList<>();
-
- @Override
- public String getCommandName() {
- return "run";
- }
-
- @Override
- public void run(CommandLine c) {
- setup();
- // make a class from whatever name
- String clazzName = args.get(0);
- String[] clazzArgs = null;
- LOG.info("Class to run is {}", clazzName);
- try {
- Class> clazz = Class.forName(clazzName);
- Method meth = clazz.getMethod("main", String[].class);
- if (args.size() > 1) {
- // run with rests of args
- clazzArgs = new String[args.size() - 1];
- for (int i = 1; i < args.size(); i++) {
- clazzArgs[i - 1] = args.get(i);
- }
- LOG.info("Running with {}", Arrays.toString(clazzArgs));
- meth.invoke(null, (Object) clazzArgs);
- } else {
- LOG.info("Running no args");
- // run class
- meth.invoke(null, (Object) new String[0]);
- }
- } catch (ClassNotFoundException e) {
- throw new RuntimeException("Could not find fully qualified class named " + clazzName);
- } catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException | NoSuchMethodException | SecurityException e) {
- String errorMsg = "Problem calling main from " + clazzName + " with " + Arrays.toString(clazzArgs);
- LOG.error(errorMsg, e);
- throw new RuntimeException(errorMsg + " : " + e.getMessage());
- }
- }
-
- @Override
- public void run() {
- run(null);
- }
-
- @Override
- public void setupCommand() {
- setupRun();
- }
-
- public void setupRun() {
- setupConfig();
- }
-}
diff --git a/src/main/java/emissary/id/SizeIdPlace.java b/src/main/java/emissary/id/SizeIdPlace.java
index d458883049..b2bbfc19ad 100644
--- a/src/main/java/emissary/id/SizeIdPlace.java
+++ b/src/main/java/emissary/id/SizeIdPlace.java
@@ -92,7 +92,4 @@ public String fileTypeBySize(int sz) {
return LABELS[LABELS.length - 1];
}
- public static void main(String[] args) {
- mainRunner(SizeIdPlace.class, args);
- }
}
diff --git a/src/main/java/emissary/id/UnixFilePlace.java b/src/main/java/emissary/id/UnixFilePlace.java
index 8af90a3fbe..9dc29243af 100755
--- a/src/main/java/emissary/id/UnixFilePlace.java
+++ b/src/main/java/emissary/id/UnixFilePlace.java
@@ -145,7 +145,4 @@ public void process(final IBaseDataObject d) {
}
}
- public static void main(final String[] args) {
- mainRunner(UnixFilePlace.class.getName(), args);
- }
}
diff --git a/src/main/java/emissary/output/DropOffPlace.java b/src/main/java/emissary/output/DropOffPlace.java
index 8095b93b86..057c954aa8 100755
--- a/src/main/java/emissary/output/DropOffPlace.java
+++ b/src/main/java/emissary/output/DropOffPlace.java
@@ -491,10 +491,4 @@ public void addFilter(final IDropOffFilter filter) {
this.outputFilters.add(filter);
}
- /**
- * Run the command line interface
- */
- public static void main(final String[] argv) throws Exception {
- mainRunner(DropOffPlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/pickup/file/FilePickUpClient.java b/src/main/java/emissary/pickup/file/FilePickUpClient.java
index 637691fe51..f6fdb276b1 100755
--- a/src/main/java/emissary/pickup/file/FilePickUpClient.java
+++ b/src/main/java/emissary/pickup/file/FilePickUpClient.java
@@ -473,8 +473,4 @@ protected void processDirectoryEntry(String root, String prefix, String caseid,
}
}
- public static void main(String[] args) {
- mainRunner(FilePickUpClient.class.getName(), args);
- }
-
}
diff --git a/src/main/java/emissary/pickup/file/FilePickUpPlace.java b/src/main/java/emissary/pickup/file/FilePickUpPlace.java
index 36860bb73c..920ed2b813 100755
--- a/src/main/java/emissary/pickup/file/FilePickUpPlace.java
+++ b/src/main/java/emissary/pickup/file/FilePickUpPlace.java
@@ -129,7 +129,4 @@ public void startDataServer() {
}
}
- public static void main(String[] args) {
- mainRunner(FilePickUpPlace.class.getName(), args);
- }
}
diff --git a/src/main/java/emissary/place/CoordinationPlace.java b/src/main/java/emissary/place/CoordinationPlace.java
index e205ae2293..cdf9f238cb 100755
--- a/src/main/java/emissary/place/CoordinationPlace.java
+++ b/src/main/java/emissary/place/CoordinationPlace.java
@@ -278,10 +278,4 @@ public List processHeavyDuty(IBaseDataObject d) throws Resource
return coordinate(d, true);
}
- /**
- * Test standalone main
- */
- public static void main(String[] argv) {
- mainRunner(CoordinationPlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/place/Main.java b/src/main/java/emissary/place/Main.java
deleted file mode 100644
index 7ae6b5e07c..0000000000
--- a/src/main/java/emissary/place/Main.java
+++ /dev/null
@@ -1,1169 +0,0 @@
-package emissary.place;
-
-import emissary.config.ConfigUtil;
-import emissary.core.Factory;
-import emissary.core.Family;
-import emissary.core.Form;
-import emissary.core.IBaseDataObject;
-import emissary.directory.DirectoryPlace;
-import emissary.directory.EmissaryNode;
-import emissary.kff.KffDataObjectHandler;
-import emissary.log.MDCConstants;
-import emissary.parser.ParserEOFException;
-import emissary.parser.ParserException;
-import emissary.parser.SessionParser;
-import emissary.parser.SessionProducer;
-import emissary.parser.SimpleParser;
-import emissary.util.Version;
-import emissary.util.shell.Executrix;
-
-import org.apache.commons.cli.CommandLine;
-import org.apache.commons.cli.CommandLineParser;
-import org.apache.commons.cli.HelpFormatter;
-import org.apache.commons.cli.Options;
-import org.apache.commons.cli.ParseException;
-import org.apache.commons.cli.PosixParser;
-import org.apache.commons.lang3.StringUtils;
-import org.slf4j.Logger;
-import org.slf4j.LoggerFactory;
-import org.slf4j.MDC;
-
-import java.io.File;
-import java.io.IOException;
-import java.io.PrintStream;
-import java.nio.file.Path;
-import java.nio.file.Paths;
-import java.util.ArrayList;
-import java.util.Collections;
-import java.util.HashMap;
-import java.util.HashSet;
-import java.util.LinkedList;
-import java.util.List;
-import java.util.Map;
-import java.util.NoSuchElementException;
-import java.util.Set;
-import java.util.regex.Pattern;
-import javax.annotation.Nullable;
-
-/**
- * This class handles running a the main method from a ServiceProviderPlace instance in a well defined but extensible
- * way. We have common command line options from getStandardOptions that can be used as-is or can be retrieved
- * and augmented with additional options and passed into the command line parser and runner. The normal flow is to use
- * the scripts/run.sh script passing in the name of the class to run followed by options, followed by a file or set of
- * files to process through that place.
- */
-@Deprecated
-public class Main {
- /** Name of the ServiceProviderPlace class to run */
- protected String placeClass;
- /** The arguments from the command line */
- protected String[] args;
- /** File args from command line after options parsed */
- protected String[] fileArgs;
- /** The instantiated service provider place */
- protected IServiceProviderPlace isp;
- /** The location of the config file or stream resource */
- protected String configLocation;
- /** Indicate if the default config has been overridden */
- private boolean configLocIsDefault = true;
- /** Current form for input data */
- protected String currentForm = emissary.core.Form.UNKNOWN;
- /** Where to send output */
- protected PrintStream outStream = System.out;
- /** Output directory for split option */
- protected String baseOutputDir = ".";
- /** True if output should be split to multiple files */
- protected boolean splitOutput = false;
- /** Recurse on input directories */
- protected boolean recurseOnFileArgs = false;
- /** Be extra quiet */
- protected boolean silent = false;
- /** Be extra verbose */
- protected boolean verbose = false;
- /** Print a short one-liner per input, good for ID places */
- protected boolean oneLiner = false;
- /** Regex for which metadata to print out */
- protected String[] metaPatterns;
- /** hash of parameters/values to set before process */
- protected HashMap params = new HashMap<>();
- /** Set to turn off Emissary node context and directory setup */
- protected boolean runWithoutContext = false;
- /** The command line options */
- protected Options options = getStandardOptions();
- /** set of alt views to print after processing */
- protected Set viewsToPrint = new HashSet<>();
- /** Class name of parser to run on input data */
- protected String parserName = SimpleParser.class.getName();
- /** Number of threads to use to process input files */
- protected int numThreads = 1;
- /** The workers that call the place */
- List workers = new ArrayList<>();
- /** Queue of payload object for workers to pull from */
- protected final LinkedList workQueue = new LinkedList<>();
- /** hashing support */
- protected KffDataObjectHandler kff = null;
- /** loop on input */
- protected boolean loopOnInput = false;
-
-
- // My logger (nb. not used for messages from the place being run)
- protected static final Logger logger = LoggerFactory.getLogger(Main.class);
-
- /**
- * Create an instance to run a job
- *
- * @param placeClass class name of the place to run
- * @param args the commandline args
- */
- public Main(String placeClass, String[] args) {
- this.placeClass = placeClass;
- this.configLocation = placeClass + ConfigUtil.CONFIG_FILE_ENDING;
- this.args = args;
- initKff();
- }
-
- /**
- * Initialize the Kff Handler with our policy settings
- */
- protected synchronized void initKff() {
- kff =
- new KffDataObjectHandler(KffDataObjectHandler.TRUNCATE_KNOWN_DATA, KffDataObjectHandler.SET_FORM_WHEN_KNOWN,
- KffDataObjectHandler.SET_FILE_TYPE);
- }
-
- /**
- * Return the standard command line options for running a place
- *
- * @return a Jakarta Commons CLI Options object
- */
- public static Options getStandardOptions() {
- Options options = new Options();
- options.addOption("1", "short", false, "short one-line listing per input");
- options.addOption("i", "loop", false, "loop on input");
- options.addOption("a", "altview", true, "name of alternate view to print");
- options.addOption("c", "config", true, "specify config file, resource, or stream name");
- options.addOption("d", "directory", true, "specify output directory for split output");
- options.addOption("h", "help", false, "print usage information");
- options.addOption("l", "level", true, "level for class logger (DEBUG|WARN|INFO|ERROR|FATAL)");
- options.addOption("L", "LEVEL", true, "level for root logger (DEBUG|WARN|INFO|ERROR|FATAL)");
- options.addOption("m", "metadata", true, "print value of named metadata slot (regex ok)");
- options.addOption("o", "output", true, "specify output filename");
- options.addOption("P", "parser", true, "class name of parser to use when loading data");
- options.addOption("p", "parameter", true, "DataObject parameter to set (key=value)");
- options.addOption("R", "recursive", false, "recurse on input directories");
- options.addOption("s", "silent", false, "dont print anything to output");
- options.addOption("S", "split", false, "split the output into separate files");
- options.addOption("t", "type", true, "type of data (current form)");
- options.addOption("T", "threads", true, "number of threads to use (default is 1)");
- options.addOption("v", "verbose", false, "print verbose output");
- options.addOption("X", "nocontext", false, "run without initializing context");
- return options;
- }
-
- /**
- * Get the currently configured options in order to change them
- *
- * @return the Jakarta Commons CLI Options object currently configured
- */
- public Options getOptions() {
- return options;
- }
-
- /**
- * Set a new options object. To start with the defaults use getStandardOptions, add your stuff, and set it here.
- *
- * @param options the new options that will be accepted
- */
- public void setOptions(Options options) {
- this.options = options;
- }
-
- /**
- * Print help and usage info based on the options
- */
- public void printUsage() {
- // automatically generate the help statement
- System.out.println("Emissary Version: " + new Version().toString());
- HelpFormatter formatter = new HelpFormatter();
- String header = null;
- String footer = " [file1 [ file2 ... fileN]]";
- boolean autoUsage = false;
- formatter.printHelp(placeClass, header, options, footer, autoUsage);
- }
-
- /**
- * Set the place config location to the specified file, resource, or stream
- *
- * @param s the string config location
- */
- public void setConfigLocation(String s) {
- configLocation = s;
- configLocIsDefault = false;
- }
-
- /**
- * Get config location value
- */
- public String getConfigLocation() {
- return configLocation;
- }
-
- /**
- * Set the parser class name
- */
- public void setParserName(String s) {
- parserName = s;
- }
-
- /**
- * Get the parser class name
- */
- public String getParserName() {
- return parserName;
- }
-
- /**
- * Set the current form for payloads to process
- *
- * @param s the new current form value
- */
- public void setCurrentForm(String s) {
- currentForm = s;
- }
-
- /**
- * Get current form value in use
- */
- public String getCurrentForm() {
- return currentForm;
- }
-
- /**
- * Set number of threads
- */
- public void setThreads(String s) {
- setThreads(Integer.parseInt(s));
- }
-
- /**
- * Set number of threads
- */
- public void setThreads(int i) {
- numThreads = i;
- }
-
- /**
- * Get the number of threads to use
- */
- public int getThreads() {
- return numThreads;
- }
-
- /**
- * Set to recurse on file args specified on command line
- *
- * @param value the new value for the recurse flag
- */
- public void setRecursive(boolean value) {
- logger.debug("Set recursive " + value);
- recurseOnFileArgs = value;
- }
-
- /**
- * Get value of recursion field
- */
- public boolean isRecursive() {
- return recurseOnFileArgs;
- }
-
- /**
- * Set to loop on input
- *
- * @param value the new value for the loop flag
- */
- public void setLoopOnInput(boolean value) {
- logger.debug("Set loop " + value);
- loopOnInput = value;
- }
-
- /**
- * Get value of loop on input field
- */
- public boolean isLoopOnInput() {
- return loopOnInput;
- }
-
- /**
- * Set value of verbose flag
- *
- * @param value the new value for verbose flag
- */
- public void setVerbose(boolean value) {
- logger.debug("Setting verbose " + value);
- verbose = value;
- }
-
- /**
- * Get value of verbose flag
- */
- public boolean isVerbose() {
- return verbose;
- }
-
- /**
- * Set the value of the oneLiner flag
- *
- * @param value the new value for the oneLiner flag
- */
- public void setOneLiner(boolean value) {
- logger.debug("Setting one liner " + value);
- oneLiner = value;
- }
-
- /**
- * Get the value of the oneLiner flag
- */
- public boolean isOneLiner() {
- return oneLiner;
- }
-
- /**
- * Get the value of the no context flag
- */
- public boolean isRunWithoutContext() {
- return runWithoutContext;
- }
-
- /**
- * Set the value of the no context flag
- */
- public void setRunWithoutContext(boolean val) {
- logger.debug("Set runWithoutContext " + val);
- runWithoutContext = val;
- }
-
- /**
- * Get file args remaining after option processing
- *
- * @return the list of file arguments or an empty list
- */
- public List getFileArgs() {
- List l = new ArrayList<>();
- for (int i = 0; fileArgs != null && i < fileArgs.length; i++) {
- l.add(fileArgs[i]);
- }
- return l;
- }
-
- /**
- * Set value of silent
- *
- * @param value the new value of the silent flag
- */
- public void setSilent(boolean value) {
- logger.debug("Set silent " + value);
- silent = value;
- }
-
- /**
- * Get the value of the silent flag
- */
- public boolean isSilent() {
- return silent;
- }
-
- /**
- * Set the split output flag
- */
- public void setSplitOutput(boolean value) {
- logger.debug("Set split output " + value);
- splitOutput = value;
- }
-
- /**
- * Get the value of the splitOutput flag
- */
- public boolean isSplitOutput() {
- return splitOutput;
- }
-
- /**
- * Set the value of the base directory for split output
- */
- public void setBaseOutputDir(String value) {
- baseOutputDir = value;
- }
-
- /**
- * Get the value of the base output directory for split output
- */
- public String getBaseOutputDir() {
- return baseOutputDir;
- }
-
- /**
- * Instantiate the parser when one is specified for the bytes of data This could be expanded to use RAF parsers, NIO
- * parsers, etc.
- */
- protected SessionParser createParser(byte[] data) {
- if (getParserName() == null) {
- return null;
- }
- return (SessionParser) Factory.create(getParserName(), data);
- }
-
- /**
- * Instantiate the specified processing place
- *
- * @return true if it is created
- */
- protected boolean createPlace() {
- String classOnlyName = placeClass.substring(placeClass.lastIndexOf(".") + 1);
-
- // See if we need to alter the default config location to use the old
- // style class no-package config name
- if (configLocIsDefault) {
- String cf = ConfigUtil.getConfigFile(configLocation);
- File fcf = new File(cf);
- if (!fcf.exists()) {
- cf = ConfigUtil.getConfigFile(classOnlyName + ConfigUtil.CONFIG_FILE_ENDING);
- fcf = new File(cf);
- if (fcf.exists()) {
- // Save it if long name does not exist but short one does
- setConfigLocation(classOnlyName + ConfigUtil.CONFIG_FILE_ENDING);
- }
- }
- }
-
- boolean pseudoNodeCreated = false;
- try {
- EmissaryNode node = new EmissaryNode();
- if (!node.isValid()) {
- System.setProperty(EmissaryNode.NODE_NAME_PROPERTY, "localhost");
- System.setProperty(EmissaryNode.NODE_PORT_PROPERTY, "0000");
- node = new EmissaryNode();
- pseudoNodeCreated = true;
-
- }
- String placeLoc = "http://" + node.getNodeName() + ":" + node.getNodePort() + "/" + classOnlyName;
- isp =
- (IServiceProviderPlace) Factory.create(placeClass, configLocation, "EMISSARY_DIRECTORY_SERVICES.STUDY.DIRECTORY_PLACE.http://"
- + node.getNodeName() + ":" + node.getNodePort() + "/DirectoryPlace", placeLoc);
-
- } catch (Throwable ex) {
- logger.error("Cannot create " + placeClass + ": " + ex.getMessage());
- return false;
- } finally {
- if (pseudoNodeCreated) {
- System.clearProperty(EmissaryNode.NODE_NAME_PROPERTY);
- System.clearProperty(EmissaryNode.NODE_PORT_PROPERTY);
- }
- }
-
- return true;
- }
-
- /**
- * Run the main method in normal testing mode using the standard or configured options and taking the remaining
- * arguments as data files or directories of data files to be processed
- */
- public void run() {
- parseArguments();
-
- // Create the pool of workers
- logger.debug("Setting up " + getThreads() + " worker threads");
- for (int i = 0; i < getThreads(); i++) {
- Worker w = new Worker();
- new Thread(w, "MainWorker-" + i).start();
- workers.add(w);
- }
-
- if (isp != null) {
- do {
- // Process the data, loading it into the work queue
- processMainData(fileArgs, null);
-
- // Wait for the work queue to be empty
- int size = -1;
- synchronized (workQueue) {
- size = workQueue.size();
- }
- while (size > 0) {
- int workQueueSize = -1;
- synchronized (workQueue) {
- workQueueSize = workQueue.size();
- }
- if (workQueueSize < size) {
- size = workQueueSize;
- logger.debug("Waiting for work queue " + size + " remaining");
- }
- try {
- Thread.sleep(100);
- } catch (InterruptedException ignore) {
- Thread.currentThread().interrupt();
- }
- }
- } while (loopOnInput);
- }
-
- // Shutdown the workers
- logger.debug("Stopping " + workers.size() + " workers");
- for (int i = 0; i < getThreads(); i++) {
- workers.get(i).stop();
- }
-
- synchronized (workQueue) {
- workQueue.notifyAll();
- }
-
- // Wait for all workers to actually stop
- for (int i = 0; i < getThreads(); i++) {
- Worker w = workers.get(i);
- while (!w.isIdle()) {
- try {
- Thread.sleep(10);
- } catch (InterruptedException ignore) {
- Thread.currentThread().interrupt();
- }
- }
- }
- }
-
- /**
- * Allow extenders to determine whether args are parsed here or not
- *
- * @param cmd the parsed command line
- * @return true if normal argument processing should continue
- */
- protected boolean preArgumentsHook(CommandLine cmd) {
- // Do nothing in this impl but allow the normal stuff to go on
- return true;
- }
-
- /**
- * Allow extenders to handle any additional arguments
- *
- * @param cmd the parsed command line
- */
- protected void postArgumentsHook(CommandLine cmd) {
- // Do nothing in this impl.
- }
-
- /**
- * Parse the command line arguments
- */
- public void parseArguments() {
- CommandLineParser parser = new PosixParser();
- CommandLine cmd = null;
- try {
- cmd = parser.parse(options, args);
- fileArgs = cmd.getArgs();
- if (fileArgs != null) {
- logger.debug("Parsed arguments and have " + fileArgs.length + " file args remaining");
- }
- } catch (ParseException ex) {
- // oops, something went wrong
- logger.error(placeClass + ": Parsing failed. Reason: " + ex.getMessage());
- printUsage();
- return;
- }
-
- boolean shouldContinue = preArgumentsHook(cmd);
-
- if (shouldContinue) {
- processParsedArguments(cmd);
- postArgumentsHook(cmd);
- }
-
- }
-
- protected void processParsedArguments(CommandLine cmd) {
- // Take care of the help option
- if (cmd.hasOption("h")) {
- printUsage();
- return;
- }
-
- // Set boolean flags
- setSilent(cmd.hasOption("s"));
- setVerbose(cmd.hasOption("v"));
- setRecursive(cmd.hasOption("R"));
- setLoopOnInput(cmd.hasOption("i"));
- setOneLiner(cmd.hasOption("1"));
- setRunWithoutContext(cmd.hasOption("X"));
-
- // Override config if specified
- if (cmd.hasOption("c")) {
- setConfigLocation(cmd.getOptionValue("c"));
- }
-
- // Set up a parser if one is specified
- if (cmd.hasOption("P")) {
- setParserName(cmd.getOptionValue("P"));
- }
-
- // Override current form if specified
- if (cmd.hasOption("t")) {
- setCurrentForm(cmd.getOptionValue("t"));
- }
-
- // Set up number of threads to use
- if (cmd.hasOption("T")) {
- logger.debug("Thread option is " + cmd.getOptionValue("T"));
- setThreads(cmd.getOptionValue("T"));
- }
-
- // save parameter key=value pairs to set
- if (cmd.hasOption("p")) {
- setParameters(cmd.getOptionValues("p"));
- }
-
- // Save regex's for looking at metadata
- setMetaPatterns(cmd.getOptionValues("m"));
-
- // Save alt view names to print in a set
- String[] altViews = cmd.getOptionValues("a");
- for (int i = 0; altViews != null && i < altViews.length; i++) {
- viewsToPrint.add(altViews[i]);
- }
-
- // Do some of the things that the normal context initializer does
- boolean pseudoNodeCreated = false;
- if (!isRunWithoutContext()) {
- try {
- EmissaryNode node = new EmissaryNode();
- if (!node.isValid()) {
- System.setProperty(EmissaryNode.NODE_NAME_PROPERTY, "localhost");
- System.setProperty(EmissaryNode.NODE_PORT_PROPERTY, "0000");
- node = new EmissaryNode();
- pseudoNodeCreated = true;
- }
- new DirectoryPlace("EMISSARY_DIRECTORY_SERVICES.STUDY.DIRECTORY_PLACE.http://" + node.getNodeName() + ":" + node.getNodePort()
- + "/DirectoryPlace", node);
- } catch (IOException iox) {
- logger.debug("Could not create standalone directory", iox);
- } finally {
- if (pseudoNodeCreated) {
- System.clearProperty(EmissaryNode.NODE_NAME_PROPERTY);
- System.clearProperty(EmissaryNode.NODE_PORT_PROPERTY);
- }
- }
- }
-
- // Create the place
- if (!createPlace()) {
- logger.warn("Unable to run main processing, no place created");
- return;
- }
-
- // Grab the base output directory for split
- if (cmd.hasOption("d")) {
- setBaseOutputDir(cmd.getOptionValue("d"));
- }
-
- // Turn on splitting of output if requested
- if (cmd.hasOption("S")) {
- setSplitOutput(true);
- }
-
-
- // Redirect the output stream if desired
- if (cmd.hasOption("o")) {
- try {
- setOutputStream(cmd.getOptionValue("o"));
- } catch (IOException iox) {
- logger.error("Cannot redirect output to " + cmd.getOptionValue("o") + ": " + iox.getMessage());
- }
-
- }
- }
-
- /**
- * Set the output stream to the new value
- *
- * @param stream the PrintStream to use further output
- */
- public void setOutputStream(PrintStream stream) {
- outStream = stream;
- }
-
- /**
- * Set the output stream using the name provided
- *
- * @param name name to use in creating a new PrintStream
- */
- public void setOutputStream(String name) throws IOException {
- outStream = new PrintStream(name);
- }
-
- /**
- * Set parameters on the new DataObject. Format of pattern is key=value.
- *
- * @param keyval the params to use
- */
- public void setParameters(String[] keyval) {
- for (String p : keyval) {
- if (p.indexOf("=") > 0) {
- // limit to two, so key=val1=val2 just turns into
- // "key" = "val1=val2"
- String[] kv = p.split("=", 2);
- params.put(kv[0], kv[1]);
- } else {
- logger.debug("param format must be key=value, skipping " + p);
- }
- }
- }
-
- /**
- * return parameters on the new DataObject.
- */
- public HashMap getParameters() {
- return params;
- }
-
- /**
- * Set the metadata print patterns
- *
- * @param patterns the patterns to use
- */
- public void setMetaPatterns(String[] patterns) {
- metaPatterns = patterns;
- }
-
- /**
- * Get a ref to the processing place being used
- */
- public IServiceProviderPlace getProcessingPlace() {
- return isp;
- }
-
- /**
- * Hook called before processHeavyDuty with every payload
- *
- * @param payload the just created payload
- * @param attachments empty list that can be populated
- * @return false to stop call to processHeavyDuty from continuing
- */
- protected boolean preProcessHook(IBaseDataObject payload, List attachments) {
- // Do nothing in this impl but allow processing to continue
- return true;
- }
-
- /**
- * Hook called after every return from processHeavyDuty
- *
- * @param payload the completed payload
- * @param att the list of attachments returned
- */
- protected void postProcessHook(IBaseDataObject payload, List att) {
- // Do nothing in this impl.
- }
-
- /**
- * Hook called after any exception thrown from the call to processHeavyDuty
- *
- * @param payload the completed payload
- */
- protected void postProcessErrorHook(IBaseDataObject payload) {
- // Do nothing in this impl.
- }
-
- /**
- * Hook called before printing the payload and attachments
- *
- * @param payload the processed payload
- * @param att the list of attachments
- * @return false to stop printing
- */
- protected boolean prePrintHook(IBaseDataObject payload, List att) {
- // Do nothing here but allow printing to continue
- logger.debug("Preprint hook called");
- return true;
- }
-
- /**
- * Hook called after printing the payload and attachments
- *
- * @param payload the processed payload
- * @param att the list of attachments
- */
- protected void postPrintHook(IBaseDataObject payload, List att) {
- // Do nothing in this impl
- logger.debug("Postprint hook called");
- }
-
- /**
- * Hook called before splitting the output to files
- *
- * @param payload the processed payload
- * @param att the list of attachments
- * @return false to stop the splitting
- */
- protected boolean preSplitHook(IBaseDataObject payload, List att) {
- // Do nothing here but allow the splitting to continue
- return true;
- }
-
- /**
- * Hook called after splitting the output
- *
- * @param payload the processed payload
- * @param att the list of attachments
- */
- protected void postSplitHook(IBaseDataObject payload, List att) {
- // Do nothing in this impl
- }
-
- /**
- * Process the file arguments from the command-line or recursed directory
- *
- * @param args the remaining (non-option) arguments
- * @param path the path we are operating in, for appending when doing file recursion
- */
- public void processMainData(String[] args, @Nullable String path) {
- for (int i = 0; i < args.length; i++) {
- String spath = (path != null ? (path + "/") : "") + args[i];
- File f = new File(spath);
- if (f.isDirectory()) {
- if (isRecursive()) {
- String[] files = f.list();
- // recursion alert
- logger.debug("Recursing into directory " + spath);
- processMainData(files, spath);
- logger.debug("Leaving recursed directory" + spath);
- // end recursive call
- } else {
- outStream.println(spath + ": DIRECTORY");
- }
- continue;
- }
-
- if (!f.canRead()) {
- outStream.println(spath + ": UNREADABLE");
- continue;
- }
-
- // Process the file
- processFile(spath);
- }
- }
-
- /**
- * Process the single named file
- */
- public void processFile(String spath) {
- // This could be expanded to allow configuration to choose
- // to use a RAF or NIO parser instead of forcing a byte array
- // parser
- byte[] content = Executrix.readDataFromFile(spath);
- if (content == null) {
- outStream.println(spath + ": UNREADABLE");
- return;
- }
-
- SessionParser sp = null;
- try {
- sp = createParser(content);
- } catch (Throwable t) {
- logger.error("Cannot create parser " + getParserName() + ": " + t);
- sp = new SimpleParser(content);
- }
-
- try {
- SessionProducer source = new SessionProducer(sp, currentForm);
- int count = 1;
- while (true) {
- IBaseDataObject payload = source.getNextSession(spath + (count > 1 ? ("-" + count) : ""));
- kff.hash(payload);
-
- // Set up filetype if non-default
- if (!emissary.core.Form.UNKNOWN.equals(currentForm)) {
- payload.setFileType(currentForm);
- }
-
- // add command line metadata before processing
- payload.setParameters(params);
-
- // Push this payload onto a queue for the Worker consumers to pull from
- queuePayload(payload);
-
- if (count % numThreads == 0) {
- Thread.yield();
- }
-
- count++;
- }
- } catch (ParserEOFException eof) {
- // expected at end of file
- } catch (ParserException ex) {
- logger.error("File " + spath + " cannot be parsed by " + getParserName() + ": " + ex);
- }
- }
-
- /**
- * Stuff the payload onto the queue for processing and notify workers
- */
- public void queuePayload(IBaseDataObject payload) {
- synchronized (workQueue) {
- workQueue.addLast(payload);
- workQueue.notifyAll();
- }
- }
-
- /**
- * Called by the Worker consumer to process the given payload
- */
- public void processPayload(IBaseDataObject payload) {
- List attachments = new ArrayList<>();
- MDC.put(MDCConstants.SHORT_NAME, payload.shortName());
- try {
- boolean shouldContinue = preProcessHook(payload, attachments);
- if (shouldContinue) {
- attachments = isp.agentProcessHeavyDuty(payload);
- }
- postProcessHook(payload, attachments);
- } catch (Throwable pex) {
- payload.replaceCurrentForm(Form.ERROR);
- attachments = Collections.emptyList();
- postProcessErrorHook(payload);
- } finally {
- MDC.remove(MDCConstants.SHORT_NAME);
- }
-
- if (!isSilent()) {
- boolean shouldPrint = prePrintHook(payload, attachments);
-
- if (shouldPrint) {
- if (isOneLiner()) {
- printPayloadOneLiner(payload, attachments.size() + payload.getExtractedRecordCount());
- } else {
- printPayload(payload);
-
- for (IBaseDataObject att : attachments) {
- printPayload(att);
- }
-
- if (payload.hasExtractedRecords()) {
- for (IBaseDataObject rec : payload.getExtractedRecords()) {
- printPayload(rec);
- }
- }
- }
- }
-
- postPrintHook(payload, attachments);
- }
-
- // Split output into files if specified
- // This is independent of printStream output
- // printing, isSlient, shouldPrint, etc.
- if (isSplitOutput()) {
- boolean shoudSplit = preSplitHook(payload, attachments);
- if (shoudSplit) {
- handleSplitOutput(payload, attachments);
- }
- postSplitHook(payload, attachments);
- }
- }
-
- /**
- * Print a one-liner for the payload
- *
- * @param payload the processed object
- * @param attachmentCount number of attachments
- */
- public void printPayloadOneLiner(IBaseDataObject payload, int attachmentCount) {
- String ft = payload.getFileType();
- String cf = payload.currentForm();
- outStream.println(payload.getFilename() + ": " + (payload.currentFormSize() > 1 ? payload.getAllCurrentForms() : payload.currentForm())
- + ((ft != null && !ft.equals(cf)) ? (" > " + payload.getFileType()) : "") + " "
- + (attachmentCount > 0 ? ("(" + attachmentCount + ")") : ""));
- }
-
- /**
- * Print out stuff related to a command-line processed payload
- *
- * @param payload the processed object
- */
- public void printPayload(@Nullable IBaseDataObject payload) {
- if (payload == null) {
- return;
- }
- outStream.println(payload.getFilename());
- outStream.println("Current form: " + payload.getAllCurrentForms());
- outStream.println("File type: " + payload.getFileType());
- outStream.println("Encoding: " + payload.getFontEncoding());
- outStream.println("Length: " + payload.dataLength());
- if (payload.getNumAlternateViews() > 0) {
- outStream.println("Alt views: " + payload.getAlternateViewNames());
- }
- if (payload.getNumChildren() > 0) {
- outStream.println("Attachments: " + payload.getNumChildren());
- }
-
- if (metaPatterns != null && metaPatterns.length > 0) {
- for (Map.Entry entry : payload.getCookedParameters().entrySet()) {
- for (int i = 0; i < metaPatterns.length; i++) {
- if (Pattern.matches(metaPatterns[i], entry.getKey())) {
- outStream.println("Metadata " + entry.getKey() + ": " + entry.getValue());
- }
- }
- }
- }
-
- boolean needTrailingCr = true;
-
- if (payload.dataLength() > 0 && (isVerbose() || (viewsToPrint.contains("MAIN") && payload.getFilename().indexOf(Family.SEP) == -1))) {
- outStream.println("Data: <
- * Note: Output will be suppressed if an IBDO form or filename would yield an output path outside the configured
- * {@link #getBaseOutputDir() base output directory} path
- *
- *
- * @param payload the processed payload
- * @param att the list of attachments
- */
- public void handleSplitOutput(IBaseDataObject payload, List att) {
- String fn = getBaseOutputDir() + "/" + payload.shortName() + "." + payload.currentForm();
-
- String safePath;
- try {
- safePath = filePathIsWithinBaseDirectory(getBaseOutputDir(), fn);
- if (Executrix.writeDataToFile(payload.data(), safePath)) {
- logger.debug("Wrote output to {}", safePath);
- } else {
- logger.error("Could not write output to {}", safePath);
- }
- } catch (IllegalArgumentException e) {
- logger.error("Could not write output to {}", fn, e);
- }
- for (IBaseDataObject part : att) {
- fn = getBaseOutputDir() + "/" + part.shortName() + "." + part.currentForm();
- try {
- safePath = filePathIsWithinBaseDirectory(getBaseOutputDir(), fn);
- if (Executrix.writeDataToFile(payload.data(), safePath)) {
- logger.debug("Wrote attachment output to {}", safePath);
- } else {
- logger.error("Could not write output to {}", safePath);
- }
- } catch (IllegalArgumentException e) {
- logger.error("Could not write output to {}", fn, e);
- }
- }
- }
-
- /**
- * Normalizes the provided paths and tests to see whether the file path is within the required base directory.
- *
- * @param requiredBase required base directory
- * @param filePath file path to be tested
- * @return the normalized file path
- * @throws IllegalArgumentException if the filePath contains illegal characters or is outside the required base
- * directory
- */
- static String filePathIsWithinBaseDirectory(final String requiredBase, final String filePath) throws IllegalArgumentException {
-
- if (StringUtils.isBlank(requiredBase)) {
- throw new IllegalArgumentException("requiredBase must not be blank");
- }
-
- if (StringUtils.isBlank(filePath)) {
- throw new IllegalArgumentException("filePath must not be blank");
- }
-
- // probably an overly simplistic test
- if (filePath.contains("..")) {
- throw new IllegalArgumentException("filePath contains illegal character sequence \"..\"");
- }
- Path normalizedBasePath = Paths.get(requiredBase).normalize().toAbsolutePath();
- Path normalizedFilePath = Paths.get(filePath).normalize().toAbsolutePath();
-
- // append path separator to defeat traversal via a sibling directory with a similar name
- if (!normalizedFilePath.startsWith(normalizedBasePath.toString() + "/")) {
- throw new IllegalArgumentException("Normalized file path (\"" + filePath + "\") is outside the required base path (\""
- + requiredBase + "\")");
- }
- return normalizedFilePath.toString();
- }
-
- /**
- * Process an IBaseDataObject from teh work queue
- */
- protected class Worker implements Runnable {
- boolean idle = true;
- boolean timeToStop = false;
-
- public boolean isIdle() {
- return idle;
- }
-
- public void stop() {
- timeToStop = true;
- }
-
- /**
- * From the Runnable interface
- */
- @Override
- public void run() {
- IBaseDataObject payload = null;
-
- while (!timeToStop) {
- // Try to get a payload
- synchronized (workQueue) {
- try {
- payload = workQueue.removeFirst();
- } catch (NoSuchElementException ignore) {
- // empty catch block
- }
- }
-
- // Process a payload if there is one
- if (payload != null) {
- idle = false;
- processPayload(payload);
- payload = null;
- idle = true;
- }
-
- // Wait for a payload
- else if (!timeToStop) {
- synchronized (workQueue) {
- try {
- workQueue.wait(1000);
- } catch (InterruptedException ignore) {
- Thread.currentThread().interrupt();
- }
- }
- }
- }
- }
- }
-}
diff --git a/src/main/java/emissary/place/MultiFileUnixCommandPlace.java b/src/main/java/emissary/place/MultiFileUnixCommandPlace.java
index f0b8a0e73e..dc1e2de776 100755
--- a/src/main/java/emissary/place/MultiFileUnixCommandPlace.java
+++ b/src/main/java/emissary/place/MultiFileUnixCommandPlace.java
@@ -715,10 +715,4 @@ protected void cleanupDirectory(File dir) {
}
}
- /**
- * Test main
- */
- public static void main(String[] argv) {
- mainRunner(MultiFileUnixCommandPlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/place/ServiceProviderPlace.java b/src/main/java/emissary/place/ServiceProviderPlace.java
index 021fecf8d3..24c2ed03e3 100755
--- a/src/main/java/emissary/place/ServiceProviderPlace.java
+++ b/src/main/java/emissary/place/ServiceProviderPlace.java
@@ -1028,31 +1028,6 @@ public MobileAgent getAgent() throws NamespaceException {
return (MobileAgent) Namespace.lookup(Thread.currentThread().getName());
}
- /**
- * Convenience to access the Main runner from a subclass If more flexibility is desired, the subclass can instantiate
- * and use the Main class directly.
- *
- * @param placeClass the class to instantiate
- * @param args from command line
- */
- public static void mainRunner(Class> placeClass, String[] args) {
- Main main = new Main(placeClass.getName(), args);
- main.run();
- }
-
- /**
- * Convenience to access the Main runner from a subclass If more flexibility is desired, the subclass can instantiate
- * and use the Main class directly.
- *
- * @param placeClass the class name to instantiate
- * @param args from command line
- */
- public static void mainRunner(String placeClass, String[] args) {
- Main main = new Main(placeClass, args);
- main.run();
- }
-
-
@Override
public List getRunningConfig() {
List runningConfigList = new ArrayList<>();
diff --git a/src/main/java/emissary/place/UnixCommandPlace.java b/src/main/java/emissary/place/UnixCommandPlace.java
index 7eb717175a..b086d5341e 100755
--- a/src/main/java/emissary/place/UnixCommandPlace.java
+++ b/src/main/java/emissary/place/UnixCommandPlace.java
@@ -428,10 +428,4 @@ protected void processData(IBaseDataObject tData) throws ResourceException {
return;
}
- /**
- * Run the class
- */
- public static void main(String[] argv) {
- mainRunner(UnixCommandPlace.class, argv);
- }
}
diff --git a/src/main/java/emissary/place/sample/DelayPlace.java b/src/main/java/emissary/place/sample/DelayPlace.java
index 92b3e07491..00235a12be 100644
--- a/src/main/java/emissary/place/sample/DelayPlace.java
+++ b/src/main/java/emissary/place/sample/DelayPlace.java
@@ -51,10 +51,4 @@ public void process(IBaseDataObject tData) throws ResourceException {
}
}
- /**
- * Test run
- */
- public static void main(String[] argv) {
- mainRunner(DelayPlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/place/sample/DevNullPlace.java b/src/main/java/emissary/place/sample/DevNullPlace.java
index 19e062e036..94f75178b1 100755
--- a/src/main/java/emissary/place/sample/DevNullPlace.java
+++ b/src/main/java/emissary/place/sample/DevNullPlace.java
@@ -39,10 +39,4 @@ public void process(IBaseDataObject tData) {
logger.debug("Nuked {} of {} current form values leaving {}", (after - before), before, tData.getAllCurrentForms());
}
- /**
- * Test run
- */
- public static void main(String[] argv) {
- mainRunner(DevNullPlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/place/sample/TemplatePlace.java b/src/main/java/emissary/place/sample/TemplatePlace.java
index afd68394da..ffb7165e56 100755
--- a/src/main/java/emissary/place/sample/TemplatePlace.java
+++ b/src/main/java/emissary/place/sample/TemplatePlace.java
@@ -105,10 +105,4 @@ public List processHeavyDuty(IBaseDataObject d) {
return Collections.emptyList();
}
- /**
- * Test standalone main
- */
- public static void main(String[] argv) {
- mainRunner(TemplatePlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/place/sample/ToLowerPlace.java b/src/main/java/emissary/place/sample/ToLowerPlace.java
index a1f6c14209..287abe0ca5 100755
--- a/src/main/java/emissary/place/sample/ToLowerPlace.java
+++ b/src/main/java/emissary/place/sample/ToLowerPlace.java
@@ -67,10 +67,4 @@ public void process(IBaseDataObject d) {
}
}
- /**
- * Test standalone main
- */
- public static void main(String[] argv) {
- mainRunner(ToLowerPlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/place/sample/ToUpperPlace.java b/src/main/java/emissary/place/sample/ToUpperPlace.java
index c8e480ab47..9cd61b8e53 100755
--- a/src/main/java/emissary/place/sample/ToUpperPlace.java
+++ b/src/main/java/emissary/place/sample/ToUpperPlace.java
@@ -67,10 +67,4 @@ public void process(IBaseDataObject d) {
}
}
- /**
- * Test standalone main
- */
- public static void main(String[] argv) {
- mainRunner(ToUpperPlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/transform/HtmlEscapePlace.java b/src/main/java/emissary/transform/HtmlEscapePlace.java
index 6e6231aa82..8c73e8202d 100755
--- a/src/main/java/emissary/transform/HtmlEscapePlace.java
+++ b/src/main/java/emissary/transform/HtmlEscapePlace.java
@@ -185,10 +185,4 @@ public static String makeString(byte[] s) {
}
- /**
- * Test standalone main
- */
- public static void main(String[] argv) {
- mainRunner(HtmlEscapePlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/transform/JavascriptEscapePlace.java b/src/main/java/emissary/transform/JavascriptEscapePlace.java
index 6469449e0b..395b86387a 100644
--- a/src/main/java/emissary/transform/JavascriptEscapePlace.java
+++ b/src/main/java/emissary/transform/JavascriptEscapePlace.java
@@ -89,10 +89,4 @@ public void process(IBaseDataObject d) {
}
- /**
- * Test standalone main
- */
- public static void main(String[] argv) {
- mainRunner(JavascriptEscapePlace.class.getName(), argv);
- }
}
diff --git a/src/main/java/emissary/transform/JsonEscapePlace.java b/src/main/java/emissary/transform/JsonEscapePlace.java
index 9c3a4bc084..43715ead1f 100755
--- a/src/main/java/emissary/transform/JsonEscapePlace.java
+++ b/src/main/java/emissary/transform/JsonEscapePlace.java
@@ -89,11 +89,4 @@ public void process(IBaseDataObject d) {
}
}
-
- /**
- * Test standalone main
- */
- public static void main(String[] argv) {
- mainRunner(JsonEscapePlace.class.getName(), argv);
- }
}
diff --git a/src/test/java/emissary/command/RunCommandIT.java b/src/test/java/emissary/command/RunCommandIT.java
deleted file mode 100644
index 3416bf397c..0000000000
--- a/src/test/java/emissary/command/RunCommandIT.java
+++ /dev/null
@@ -1,101 +0,0 @@
-package emissary.command;
-
-import emissary.test.core.junit5.UnitTest;
-
-import org.junit.jupiter.api.AfterEach;
-import org.junit.jupiter.api.BeforeEach;
-import org.junit.jupiter.api.Test;
-
-import java.io.ByteArrayOutputStream;
-import java.io.PrintStream;
-import java.util.Arrays;
-
-import static org.junit.jupiter.api.Assertions.assertThrows;
-import static org.junit.jupiter.api.Assertions.assertTrue;
-
-class RunCommandIT extends UnitTest {
-
- private ByteArrayOutputStream outContent;
- private ByteArrayOutputStream errContent;
-
- @Override
- @BeforeEach
- public void setUp() throws Exception {
- super.setUp();
- outContent = new ByteArrayOutputStream();
- errContent = new ByteArrayOutputStream();
- }
-
- @Override
- @AfterEach
- public void tearDown() throws Exception {
- super.tearDown();
- outContent = null;
- errContent = null;
- }
-
- @Test
- void testClassMustExist() {
- String clazzName = "com.junk.Who";
- Exception e = assertThrows(Exception.class, () -> {
- RunCommand cmd = RunCommand.parse(RunCommand.class, clazzName);
- cmd.run();
- });
- assertTrue(e.getMessage().contains("Could not find fully qualified class named " + clazzName));
- }
-
- @Test
- void testClassIsRunWhenFound() throws Exception {
- String clazzName = "emissary.command.RunCommandIT";
- RunCommand cmd = RunCommand.parse(RunCommand.class, clazzName);
-
- captureStdOutAndStdErrAndRunCommand(cmd);
-
- assertTrue(outContent.toString().contains("I am a test that runs myself. My args are []"));
- }
-
- @Test
- void testClassIsRunWhenFoundWithArgs() throws Exception {
- String clazzName = "emissary.command.RunCommandIT";
- String arg1 = "asdf";
- String arg2 = "dkdke";
- String arg3 = "k3k23k";
- RunCommand cmd = RunCommand.parse(RunCommand.class, clazzName, arg1, arg2, arg3);
-
- captureStdOutAndStdErrAndRunCommand(cmd);
-
- assertTrue(outContent.toString().contains("I am a test that runs myself. My args are [" + arg1 + ", " + arg2 + ", " + arg3 + "]"));
- }
-
- @Test
- void testFlagArgsPassedThrough() throws Exception {
- String clazzName = "emissary.command.RunCommandIT";
- String stopPicocliProcessing = "--";
- String arg1 = "-f";
- String arg2 = "somefile";
- String arg3 = "--greatestArg";
- String arg4 = "ever";
- RunCommand cmd = RunCommand.parse(RunCommand.class, clazzName, stopPicocliProcessing, arg1, arg2, arg3, arg4);
-
- captureStdOutAndStdErrAndRunCommand(cmd);
-
- assertTrue(outContent.toString().contains("I am a test that runs myself. My args are [" + arg1 + ", " + arg2 + ", " + arg3 + ", "
- + arg4 + "]"));
-
- }
-
- private void captureStdOutAndStdErrAndRunCommand(RunCommand cmd) {
- PrintStream origOut = System.out;
- PrintStream origErr = System.err;
- System.setOut(new PrintStream(outContent));
- System.setErr(new PrintStream(errContent));
- cmd.run();
- System.setOut(origOut);
- System.setErr(origErr);
- }
-
- public static void main(String[] args) {
- System.out.println("I am a test that runs myself. My args are " + Arrays.toString(args));
- }
-
-}
diff --git a/src/test/java/emissary/place/MainTest.java b/src/test/java/emissary/place/MainTest.java
deleted file mode 100644
index 08ed6eba34..0000000000
--- a/src/test/java/emissary/place/MainTest.java
+++ /dev/null
@@ -1,368 +0,0 @@
-package emissary.place;
-
-import emissary.core.DataObjectFactory;
-import emissary.core.Form;
-import emissary.core.IBaseDataObject;
-import emissary.core.Namespace;
-import emissary.core.ResourceException;
-import emissary.test.core.junit5.UnitTest;
-
-import org.apache.commons.cli.CommandLine;
-import org.apache.commons.cli.Options;
-import org.junit.jupiter.api.AfterEach;
-import org.junit.jupiter.api.BeforeEach;
-import org.junit.jupiter.api.Test;
-import org.junit.jupiter.api.io.TempDir;
-
-import java.io.ByteArrayOutputStream;
-import java.io.IOException;
-import java.io.OutputStream;
-import java.io.PrintStream;
-import java.nio.file.Files;
-import java.nio.file.Path;
-import java.nio.file.Paths;
-import java.util.ArrayList;
-import java.util.List;
-
-import static emissary.place.Main.filePathIsWithinBaseDirectory;
-import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;
-import static org.junit.jupiter.api.Assertions.assertEquals;
-import static org.junit.jupiter.api.Assertions.assertFalse;
-import static org.junit.jupiter.api.Assertions.assertNotNull;
-import static org.junit.jupiter.api.Assertions.assertThrows;
-import static org.junit.jupiter.api.Assertions.assertTrue;
-import static org.junit.jupiter.api.Assertions.fail;
-
-final class MainTest extends UnitTest {
- private final String className = "emissary.place.sample.DevNullPlace";
- private final String[] defaultArgs = {"-s"};
-
- @TempDir
- public Path testOutputFolder;
-
- @Override
- @BeforeEach
- public void setUp() throws Exception {
- Path dataFile = Paths.get(TMPDIR, "testmain.dat");
- try (OutputStream ros = Files.newOutputStream(dataFile)) {
- ros.write("abcdefghijklmnopqrstuvwxyz".getBytes());
- } catch (IOException ex) {
- fail("Unable to create test file", ex);
- }
- }
-
- @Override
- @AfterEach
- public void tearDown() throws Exception {
- super.tearDown();
- // Clear out namespace
- for (String key : Namespace.keySet()) {
- Namespace.unbind(key);
- }
- Path dataFile = Paths.get(TMPDIR, "testmain.dat");
- Files.deleteIfExists(dataFile);
- }
-
- @Test
- void testStandardOptions() {
- Options o = Main.getStandardOptions();
- assertNotNull(o, "Standard options should be produced");
-
- Main m = new Main(className, defaultArgs);
- Options mo = m.getOptions();
- assertNotNull(mo, "Standard options from instance should be produced");
-
- assertEquals(o.getOptions().size(), mo.getOptions().size(), "Standard options are used by default");
-
- }
-
- @Test
- void testParseDefaultValues() {
- Main m = new Main(className, defaultArgs);
- m.parseArguments();
- assertTrue(m.isSilent(), "Default args sets silent");
- assertFalse(m.isRecursive(), "Default args are not recursive");
- assertFalse(m.isVerbose(), "Default args are not verbose");
- assertEquals(className + ".cfg", m.getConfigLocation(), "Default config location should stick");
- assertEquals(emissary.core.Form.UNKNOWN, m.getCurrentForm(), "Default current form should be there");
- assertEquals(0, m.getFileArgs().size(), "No files should be left");
- }
-
- @Test
- void testParseReverseValues() {
- String[] newArgs = {"-v", "-R"};
- Main m = new Main(className, newArgs);
- m.parseArguments();
- assertFalse(m.isSilent(), "These args are not silent");
- assertTrue(m.isVerbose(), "These args are verbose");
- assertTrue(m.isRecursive(), "These args are recursive");
- assertEquals(0, m.getFileArgs().size(), "No files should be left");
- }
-
- @Test
- void testParseConfigLoc() {
- String[] newArgs = {"-c", "emissary.place.sample.ToLowerPlace.cfg"};
- Main m = new Main(className, newArgs);
- m.parseArguments();
- assertEquals(newArgs[1], m.getConfigLocation(), "Specified config location should stick");
- assertEquals(0, m.getFileArgs().size(), "No files should be left");
- }
-
- @Test
- void testCurrentForm() {
- String[] newArgs = {"-t", "FOOBAR"};
- Main m = new Main(className, newArgs);
- m.parseArguments();
- assertEquals(newArgs[1], m.getCurrentForm(), "Specified current form should stick");
- assertEquals(0, m.getFileArgs().size(), "No files should be left");
- }
-
- @Test
- void testSetParams() {
- String[] newArgs = {"-p", "FOO=BAR"};
- Main m = new Main(className, newArgs);
- m.parseArguments();
- IBaseDataObject payload = DataObjectFactory.getInstance("aaa".getBytes(), "test", "UNKNOWN");
- payload.setParameters(m.getParameters());
- assertEquals("BAR", payload.getStringParameter("FOO"), "Specified parameter should stick");
- }
-
- @Test
- void testLoggerLevels() {
- // Careful, DevNull and root category will not be reset after this
- String[] newArgs = {"-l", "FATAL", "-L", "INFO"};
- Main m = new Main(className, newArgs);
- m.parseArguments();
- assertEquals(0, m.getFileArgs().size(), "No files should be left");
- }
-
- @Test
- void testFilesRemaining() {
- String[] newArgs = {"-s", "foo.txt", "bar.txt"};
- Main m = new Main(className, newArgs);
- m.parseArguments();
- assertEquals(2, m.getFileArgs().size(), "Two files should be left");
- }
-
- @Test
- void testPrintHooks() {
- String[] args = {"-X", TMPDIR + "/testmain.dat"};
- MainWithHooks m = new MainWithHooks(className, args);
- m.run();
- assertTrue(m.printhook[0], "PrePrintHook must be called");
- assertTrue(m.printhook[1], "PostPrintHook must be called");
- }
-
- @Test
- void testProcHooks() {
- String[] args = {"-X", TMPDIR + "/testmain.dat"};
- MainWithHooks m = new MainWithHooks(className, args);
- m.run();
- assertTrue(m.processhook[0], "PreProcessHook must be called");
- assertTrue(m.processhook[1], "PostProcessHook must be called");
- }
-
- @Test
- void testArgumentHooks() {
- String[] args = {"-X", "foo.txt"};
- MainWithHooks m = new MainWithHooks(className, args);
- m.parseArguments();
- assertTrue(m.arghook[0], "PreArgumentsHook must be called");
- assertTrue(m.arghook[1], "PostArgumentsHook must be called");
- }
-
- @Test
- void testSplitHook() throws IOException {
- String[] args = {"-s", "-d", TMPDIR, "-S", TMPDIR + "/testmain.dat"};
- MainWithHooks m = new MainWithHooks(className, args);
- m.run();
- assertEquals(1, m.getFileArgs().size(), "Must process args and leave file");
- assertTrue(m.splithook[0], "PreSplitHook must be called");
- assertTrue(m.splithook[1], "PostSplitHook must be called");
- Path expectedSplitFile = Paths.get(TMPDIR, "testmain.dat." + Form.UNKNOWN);
- assertTrue(Files.exists(expectedSplitFile), "File must exist after split");
- Files.deleteIfExists(expectedSplitFile);
- }
-
- @Test
- void testExceptionHandlingInPlaceProcessing() {
- String[] args = {"-s", "-X", TMPDIR + "/testmain.dat"};
- Main m = new Main(OOMPlace.class.getName(), args);
- m.parseArguments();
- assertDoesNotThrow(m::run);
- }
-
- @Test
- void testMetaDataFromChildrenAndExtractedRecordsIsCaptured() {
- ByteArrayOutputStream baos = new ByteArrayOutputStream();
- Main m = new Main("emissary.place.MainTest$PlaceTest", new String[] {"-m", ".*"});
- m.setOutputStream(new PrintStream(baos));
- m.run();
-
- IBaseDataObject payload = DataObjectFactory.getInstance("aaa".getBytes(), "test", "UNKNOWN");
- payload.putParameter("PARENT_KEY", "parent value");
- m.processPayload(payload);
-
- String s = baos.toString();
- assertTrue(s.contains("PARENT_KEY"), "Should have recorded parent metadata - " + s);
- assertTrue(s.contains("CHILD_KEY"), "Should have recorded child metadata - " + s);
- assertTrue(s.contains("RECORD_KEY"), "Should have recorded record metadata - " + s);
- }
-
- @Test
- void testHandleSplitOutput() throws IOException {
- Path outputFolder = Files.createDirectories(testOutputFolder.resolve("testmain-split.dat"));
- String[] args = {"-S", "-d", outputFolder.toRealPath().toString()};
- Main m = new Main(this.className, args);
- m.parseArguments();
- try {
- m.run();
- } catch (Throwable t) {
- fail("Main runner allowed exception to escape", t);
- }
- IBaseDataObject payload = DataObjectFactory.getInstance("aaa".getBytes(), "test", "UNKNOWN");
- List atts = new ArrayList<>();
- IBaseDataObject att1 = DataObjectFactory.getInstance("bbb".getBytes(), "safe_attempt", "SAFE_ATTEMPT");
- atts.add(att1);
- IBaseDataObject att2 = DataObjectFactory.getInstance("ccc".getBytes(), "escape_attempt", "./../../pwned/ESCAPE_ATTEMPT");
- atts.add(att2);
-
- IBaseDataObject att3 = DataObjectFactory.getInstance("ccc".getBytes(), "attempt", "./../../testmain.dat_sibling/ESCAPE_ATTEMPT");
- atts.add(att3);
-
- m.handleSplitOutput(payload, atts);
-
- // validate that output files "test.UNKNOWN" and "safe_attempt.SAFE_ATTEMPT" were created, but file
- // "escape_attempt../../../pwned/ESCAPE_ATTEMPT" was not
- String normalizedPath = outputFolder + "/" + payload.shortName() + "." + payload.currentForm();
- assertTrue(Files.exists(Paths.get(normalizedPath).normalize()), "File \"" + normalizedPath + "\" should have been created");
-
- normalizedPath = outputFolder + "/" + att1.shortName() + "." + att1.currentForm();
- assertTrue(Files.exists(Paths.get(normalizedPath).normalize()), "File \"" + normalizedPath + "\" should have been created");
-
- normalizedPath = outputFolder + "/" + att2.shortName() + "." + att2.currentForm();
- assertFalse(Files.exists(Paths.get(normalizedPath).normalize()), "File \"" + normalizedPath + "\" should have NOT been created");
-
- normalizedPath = outputFolder + "/" + att3.shortName() + "." + att3.currentForm();
- assertFalse(Files.exists(Paths.get(normalizedPath).normalize()), "File \"" + normalizedPath + "\" should have NOT been created");
- }
-
- @Test
- void testFilePathIsWithinBaseDirectory() {
-
- String basePath = testOutputFolder.resolve("foo").toString();
- assertEquals(basePath + "/somefile", filePathIsWithinBaseDirectory(basePath, basePath + "/somefile"));
- assertEquals(basePath + "/otherfile", filePathIsWithinBaseDirectory(basePath, basePath + "//otherfile"));
- assertEquals(basePath + "/foo/otherfile", filePathIsWithinBaseDirectory(basePath, basePath + "/./foo/otherfile"));
- assertEquals(basePath + "/sub/otherfile", filePathIsWithinBaseDirectory(basePath, basePath + "/sub/././otherfile"));
-
- // Each of these should thrown an Exception
- assertThrows(IllegalArgumentException.class, () -> filePathIsWithinBaseDirectory(basePath, "/var/log/somelog"));
-
- assertThrows(IllegalArgumentException.class,
- () -> filePathIsWithinBaseDirectory(basePath, basePath + "/../foo2/otherfile"),
- "Expected an IllegalArgumentException from input " + basePath + "/../foo2/otherfile");
-
- assertThrows(IllegalArgumentException.class,
- () -> filePathIsWithinBaseDirectory(basePath, basePath + "/../../somefile"),
- "Expected an IllegalArgumentException from input " + basePath + "/../../somefile");
-
- assertThrows(IllegalArgumentException.class,
- () -> filePathIsWithinBaseDirectory(basePath, basePath + "/path/../../../otherpath"),
- "Expected an IllegalArgumentException from input " + basePath + "/path/../../../otherpath");
- }
-
- // Override the hooks. The calls to super in these
- // hooks are not normal, but help us mark the super class
- // hooks as having been tested.
- private static final class MainWithHooks extends Main {
- public boolean[] printhook = {false, false};
- public boolean[] arghook = {false, false};
- public boolean[] processhook = {false, false};
- public boolean[] splithook = {false, false};
-
- public MainWithHooks(String className, String[] args) {
- super(className, args);
- }
-
- @Override
- protected boolean prePrintHook(IBaseDataObject payload, List att) {
- super.prePrintHook(payload, att);
- printhook[0] = true;
- return false;
- }
-
- @Override
- protected void postPrintHook(IBaseDataObject payload, List att) {
- super.postPrintHook(payload, att);
- printhook[1] = true;
- }
-
- @Override
- protected boolean preArgumentsHook(CommandLine cmd) {
- super.preArgumentsHook(cmd);
- arghook[0] = true;
- return true;
- }
-
- @Override
- protected void postArgumentsHook(CommandLine cmd) {
- super.postArgumentsHook(cmd);
- arghook[1] = true;
- }
-
- @Override
- protected boolean preProcessHook(IBaseDataObject payload, List attachments) {
- super.preProcessHook(payload, attachments);
- processhook[0] = true;
- // Form checked in split hook test for file
- if (payload.currentForm() == null) {
- payload.setCurrentForm(Form.UNKNOWN);
- }
- return true;
- }
-
- @Override
- protected void postProcessHook(IBaseDataObject payload, List att) {
- super.postProcessHook(payload, att);
- processhook[1] = true;
- }
-
- @Override
- protected boolean preSplitHook(IBaseDataObject payload, List att) {
- super.preSplitHook(payload, att);
- splithook[0] = true;
- return true;
- }
-
- @Override
- protected void postSplitHook(IBaseDataObject payload, List att) {
- super.postSplitHook(payload, att);
- splithook[1] = true;
- }
- }
-
- public static class PlaceTest extends ServiceProviderPlace {
- public PlaceTest(String configInfo, String dir, String placeLoc) throws IOException {
- super();
- }
-
- @Override
- public List processHeavyDuty(IBaseDataObject payload) throws ResourceException {
- IBaseDataObject extr = DataObjectFactory.getInstance("cde".getBytes(), "test-att-2", "UNKNOWN");
- extr.putParameter("RECORD_KEY", "record value");
- payload.addExtractedRecord(extr);
-
- IBaseDataObject child = DataObjectFactory.getInstance("cde".getBytes(), "test-att-1", "UNKNOWN");
- payload.putParameter("CHILD_KEY", "child value");
-
- List children = new ArrayList<>();
- children.add(child);
-
- return children;
- }
-
- @Override
- public void process(IBaseDataObject payload) throws ResourceException {}
- }
-}