JaStaCry.java
package org.jastacry;
import java.util.Locale;
import java.util.ResourceBundle;
import org.apache.commons.cli.CommandLine;
import org.apache.commons.cli.CommandLineParser;
import org.apache.commons.cli.DefaultParser;
import org.apache.commons.cli.HelpFormatter;
import org.apache.commons.cli.MissingOptionException;
import org.apache.commons.cli.Option;
import org.apache.commons.cli.OptionGroup;
import org.apache.commons.cli.Options;
import org.apache.commons.cli.ParseException;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.jastacry.GlobalData.Action;
import org.jastacry.GlobalData.Returncode;
import net.sourceforge.cobertura.CoverageIgnore;
/**
* Main JaStaCry class to start.
*
* <p>SPDX-License-Identifier: MIT
*
* @author Kai Kretschmann
*/
public final class JaStaCry
{
/**
* Holder of i18n translations.
*/
private static ResourceBundle bundle;
/**
* Parameter, short version, for "help".
*/
private static final String P_SHORT_HELP = "h";
/**
* Parameter, long version, for "help".
*/
private static final String P_LONG_HELP = "help";
/**
* Parameter, short version, for "verbose".
*/
private static final String P_SHORT_VERBOSE = "v";
/**
* Parameter, short version, for "ASCII".
*/
private static final String P_SHORT_ASCII = "t";
/**
* Parameter, long version, for "ASCII".
*/
private static final String P_LONG_ASCII = "text";
/**
* Parameter, short version, for "encode".
*/
private static final String P_SHORT_ENCODE = "e";
/**
* Parameter, long version, for "encode".
*/
private static final String P_LONG_ENCODE = "encode";
/**
* Parameter, short version, for "decode".
*/
private static final String P_SHORT_DECODE = "d";
/**
* Parameter, long version, for "decode".
*/
private static final String P_LONG_DECODE = "decode";
/**
* Parameter, short version, for "config".
*/
private static final String P_SHORT_CONFFILE = "c";
/**
* Parameter, long version, for "config".
*/
private static final String P_LONG_CONFFILE = "conffile";
/**
* Parameter, short version, for "infile".
*/
private static final String P_SHORT_INFILE = "i";
/**
* Parameter, long version, for "infile".
*/
private static final String P_LONG_INFILE = "infile";
/**
* Parameter, short version, for "outfile".
*/
private static final String P_SHORT_OUTFILE = "o";
/**
* Parameter, long version, for "outfile".
*/
private static final String P_LONG_OUTFILE = "outfile";
/**
* log4j logger object.
*/
private static final Logger LOGGER = LogManager.getLogger();
/**
* boolean status: do we encode to plain text transport format.
*/
private static boolean doASCIItransport;
/**
* Filename of configuration file.
*/
private static String confFilename;
/**
* Some input filename.
*/
private static String inputFilename;
/**
* Some output filename.
*/
private static String outputFilename;
/**
* Be verbose about every step.
*/
private static boolean isVerbose;
/**
* action variable.
*/
private static Action action;
/**
* Hidden constructor.
*/
@CoverageIgnore
private JaStaCry()
{
// not called
}
/**
* Main class for running a command line interface.
*
* @param args parsed by Apache commons CLI package
*/
@CoverageIgnore
@SuppressWarnings("squid:S4823") // Using command line arguments is security-sensitive
public static void main(final String[] args)
{
final int returncode = mainMethod(args);
System.exit(returncode);
}
/**
* Main method.
*
* @param args for parsing
* @return int result code
*/
public static int mainMethod(final String... args)
{
LOGGER.traceEntry();
localizer();
int returncode = setup(args);
if (0 != returncode)
{
LOGGER.error(getText("error.setupfound"), returncode);
return returncode;
} // if
final Worker worker = new Worker();
worker.setAction(action);
worker.setConfFilename(confFilename);
worker.setDoAsciitransport(doASCIItransport);
worker.setInputFilename(inputFilename);
worker.setOutputFilename(outputFilename);
worker.setVerbose(isVerbose);
returncode = worker.mainWork();
return LOGGER.traceExit(returncode);
}
/**
* Create command line Options object.
*
* @return Options object
*/
private static Options createOptions()
{
LOGGER.traceEntry();
final Options options = new Options();
// optional parameters
options.addOption(P_SHORT_HELP, P_LONG_HELP, false, getText("help.help"));
options.addOption(P_SHORT_VERBOSE, false, getText("help.verbose"));
options.addOption(P_SHORT_ASCII, P_LONG_ASCII, false, getText("help.ascii"));
// either/or arguments, but mandatory as a set
final OptionGroup ogAction = new OptionGroup();
Option option;
option = Option.builder(P_SHORT_ENCODE).required(false).longOpt(P_LONG_ENCODE).desc(getText("help.encode")).build();
ogAction.addOption(option);
option = Option.builder(P_SHORT_DECODE).required(false).longOpt(P_LONG_DECODE).desc(getText("help.decode")).build();
ogAction.addOption(option);
ogAction.setRequired(true);
options.addOptionGroup(ogAction);
// mandatory parameters
option = Option.builder(P_SHORT_CONFFILE).required(true).hasArg().longOpt(P_LONG_CONFFILE).argName("FILE")
.desc("use FILE as stack configuration").build();
options.addOption(option);
option = Option.builder(P_SHORT_INFILE).required(true).hasArg().longOpt(P_LONG_INFILE).argName("FILE")
.desc("use FILE as input stream").build();
options.addOption(option);
option = Option.builder(P_SHORT_OUTFILE).required(true).hasArg().longOpt(P_LONG_OUTFILE).argName("FILE")
.desc("use FILE as output stream").build();
options.addOption(option);
return LOGGER.traceExit(options);
}
/**
* Read current locale and load i18n bundle.
*/
private static void localizer()
{
LOGGER.traceEntry();
final Locale currentLocale = Locale.getDefault();
LOGGER.info("Locale: {}", currentLocale);
// First read locale i18n stuff
bundle = ResourceBundle.getBundle("Bundle");
LOGGER.traceExit();
}
/**
* Helper function to get translated text from bundle.
* @param key String key for value
* @return String as translation
*/
public static String getText(final String key)
{
return bundle.getString(key);
}
/**
* Setup environment via command line arguments.
*
* @param args array of Strings from command line
* @return int error value
*/
private static int setup(final String... args)
{
LOGGER.traceEntry();
// Command line parameters
final Options options = createOptions();
// Manual check for help, ignoring otherwise mandatory arguments
final HelpFormatter formatter = new HelpFormatter();
if (args.length > 0 && "-h".equalsIgnoreCase(args[0]))
{
formatter.printHelp(GlobalData.HELP, options);
return Returncode.RC_HELP.getNumVal();
} // if
final CommandLineParser parser = new DefaultParser();
CommandLine cmdLine;
try
{
cmdLine = parser.parse(options, args);
}
catch (final MissingOptionException exOpt)
{
formatter.printHelp(GlobalData.HELP, options);
return Returncode.RC_ERROR.getNumVal();
}
catch (final ParseException e2)
{
LOGGER.catching(e2);
return LOGGER.traceExit(Returncode.RC_ERROR.getNumVal());
}
if (null == cmdLine)
{
LOGGER.error("cmdLine null");
return LOGGER.traceExit(Returncode.RC_ERROR.getNumVal());
}
// No need to check for P_SHORT_HELP here anymore.
// Is verbose?
isVerbose = cmdLine.hasOption(P_SHORT_VERBOSE);
if (isVerbose)
{
LOGGER.info("JaStaCry starting");
}
action = Action.UNKOWN;
// is it called with all needed parameters?
if (cmdLine.hasOption(P_SHORT_ENCODE) || cmdLine.hasOption(P_LONG_ENCODE))
{
action = Action.ENCODE;
} // if
if (cmdLine.hasOption(P_SHORT_DECODE) || cmdLine.hasOption(P_LONG_DECODE))
{
action = Action.DECODE;
} // if
// Use text format?
doASCIItransport = cmdLine.hasOption(P_SHORT_ASCII) || cmdLine.hasOption(P_LONG_ASCII);
// Get file names for config, input and output
confFilename = cmdLine.getOptionValue(P_LONG_CONFFILE);
inputFilename = cmdLine.getOptionValue(P_LONG_INFILE);
outputFilename = cmdLine.getOptionValue(P_LONG_OUTFILE);
return LOGGER.traceExit(0);
}
}