JAL-629 More docs. Usage statement. Adjust some logging output to respect --quiet.
[jalview.git] / src / jalview / bin / argparser / Arg.java
index d77c075..02584fb 100644 (file)
@@ -3,21 +3,190 @@ package jalview.bin.argparser;
 import java.util.ArrayList;
 import java.util.Arrays;
 import java.util.EnumSet;
+import java.util.Iterator;
 import java.util.List;
 import java.util.Locale;
 import java.util.stream.Collectors;
 
+import jalview.bin.Cache;
+import jalview.util.ChannelProperties;
+
 public enum Arg
 {
-  HELP("h"), CALCULATION, MENUBAR, STATUS, SHOWOVERVIEW, SHOWANNOTATIONS,
-  COLOUR("color"), FEATURES, ANNOTATIONS, GROOVY, GROUPS, HEADLESS, JABAWS,
-  DISPLAY, NEWS, SORTBYTREE, USAGESTATS, APPEND, OPEN, PROPS, QUESTIONNAIRE,
-  SETPROP, TREE, VDOC, VSESS, OUTPUT, SSANNOTATIONS, NOTEMPFAC, TEMPFAC,
-  TITLE, PAEMATRIX, WRAP, NOSTRUCTURE, STRUCTURE, STRUCTUREVIEWER, IMAGE,
-  TYPE, FORMAT, OVERWRITE, TEXTRENDERER, QUIT, CLOSE, DEBUG("d"), TRACE,
-  QUIET("q"), ARGFILE, NEW, NPP("n++"), SUBSTITUTIONS, INITSUBSTITUTIONS,
-  NIL, SPLASH, SETARGFILE, UNSETARGFILE, WEBSERVICEDISCOVERY, ALL, BACKUPS,
-  TESTOUTPUT, SEQID;
+
+  // Initialising arguments (BOOTSTRAP)
+  HELP("Display this help statement", Opt.UNARY, Opt.BOOTSTRAP),
+  HEADLESS(
+          "Run Jalview in headless mode. No GUI interface will be created and Jalview will quit after all arguments have been processed.",
+          Opt.UNARY, Opt.BOOTSTRAP),
+  JABAWS("Set a different URL to connect to a JABAWS server.", Opt.STRING,
+          Opt.BOOTSTRAP),
+  NEWS("Show (or don't show) the news feed.", true, Opt.BOOLEAN,
+          Opt.BOOTSTRAP),
+  SPLASH("Show (or don't show) the About Jalview splash screen.", true,
+          Opt.BOOLEAN, Opt.BOOTSTRAP),
+  QUESTIONNAIRE(
+          "Show (or don't show) the questionnaire if one is available.",
+          true, Opt.BOOLEAN, Opt.BOOTSTRAP),
+  USAGESTATS("Send (or don't send) initial launch usage stats.", true,
+          Opt.BOOLEAN, Opt.BOOTSTRAP),
+  WEBSERVICEDISCOVERY(
+          "Attempt (or don't attempt) to connect to JABAWS web services.",
+          true, Opt.BOOLEAN, Opt.BOOTSTRAP),
+  PROPS("Use file filename as the preferences file instead of the usual ~/.jalview_properties file.",
+          Opt.STRING, Opt.BOOTSTRAP),
+  DEBUG("Start Jalview in debug log level.", Opt.BOOLEAN, Opt.BOOTSTRAP),
+  TRACE("Start Jalview in trace log level.", Opt.BOOLEAN, Opt.BOOTSTRAP,
+          Opt.SECRET),
+  QUIET("Stop all output to STDOUT (after the Java Virtual Machine has started). Use ‑‑quiet a second time to stop all output to STDERR.",
+          Opt.UNARY, Opt.MULTI, Opt.BOOTSTRAP),
+  INITSUBSTITUTIONS(
+          "Assume that ‑‑substitutions are initially enabled (or initially disabled).",
+          true, Opt.BOOLEAN, Opt.BOOTSTRAP, Opt.NOACTION),
+
+  // Opening an alignment
+  OPEN("Opens one or more alignment files filename or URLs URL in new alignment windows.",
+          Opt.STRING, Opt.LINKED, Opt.INCREMENTDEFAULTCOUNTER, Opt.MULTI,
+          Opt.GLOB, Opt.ALLOWSUBSTITUTIONS, Opt.INPUT),
+  APPEND("Appends one or more alignment files filename or URLs URL to the open alignment window (or opens a new alignment if none already open).",
+          Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.GLOB,
+          Opt.ALLOWSUBSTITUTIONS, Opt.INPUT),
+  TITLE("Specifies the title for the open alignment window as string.",
+          Opt.STRING, Opt.LINKED),
+  COLOUR("Applies the colour scheme to the open alignment window. Valid values are:\n"
+          + "clustal,\n" + "blosum62,\n" + "pc-identity,\n" + "zappo,\n"
+          + "taylor,\n" + "gecos-flower,\n" + "gecos-blossom,\n"
+          + "gecos-sunset,\n" + "gecos-ocean,\n" + "hydrophobic,\n"
+          + "helix-propensity,\n" + "strand-propensity,\n"
+          + "turn-propensity,\n" + "buried-index,\n" + "nucleotide,\n"
+          + "nucleotide-ambiguity,\n" + "purine-pyrimidine,\n"
+          + "rna-helices,\n" + "t-coffee-scores,\n" + "sequence-id.",
+          Opt.STRING, Opt.LINKED),
+  FEATURES("Add a feature file filename or URL URL to the open alignment.",
+          Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.ALLOWSUBSTITUTIONS),
+  TREE("Add a tree file filename or URL URL to the open alignment.",
+          Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.ALLOWSUBSTITUTIONS),
+  SORTBYTREE(
+          "Enforces sorting (or not sorting) the alignment in the order of an attached phylogenetic tree.",
+          true, Opt.LINKED, Opt.BOOLEAN),
+  ANNOTATIONS(
+          "Add an annotations file filename or URL URL to the open alignment.",
+          Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.ALLOWSUBSTITUTIONS),
+  SHOWANNOTATIONS(
+          "Enforces showing (or not showing) alignment annotations.",
+          Opt.BOOLEAN, Opt.LINKED),
+  WRAP("Enforces wrapped (or not wrapped) alignment formatting.",
+          Opt.BOOLEAN, Opt.LINKED),
+  NOSTRUCTURE(
+          "Do not open or process any 3D structure in the ‑‑open or ‑‑append files.",
+          Opt.UNARY, Opt.LINKED),
+
+  // Adding a 3D structure
+  STRUCTURE(
+          "Load a structure file filename or URL URL associated with a sequence in the open alignment. The sequence to be associated with can be specified with a following --seqid argument, or the subval modifier seqid=ID can be used. A subval INDEX can also be used to specify the INDEX-th sequence in the open alignment.",
+          Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.ALLOWSUBSTITUTIONS),
+  SEQID("Specify the sequence name for the preceding --structure to be associated with.",
+          Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.ALLOWSUBSTITUTIONS),
+  PAEMATRIX(
+          "Add a PAE json matrix file filename to the preceding --structure.",
+          Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.ALLOWSUBSTITUTIONS),
+  TEMPFAC("Set the type of temperature factor. Possible values for name are default, plddt.",
+          Opt.STRING, Opt.LINKED),
+  STRUCTUREVIEWER(
+          "Set the structure viewer to use to open the 3d structure file specified in previous --structure to name. Possible values of name are:\n"
+                  + "none, jmol, chimera, chimerax, pymol.",
+          Opt.STRING, Opt.LINKED, Opt.MULTI),
+  NOTEMPFAC(
+          "Do not show the temperature factor annotation for the preceding --structure.",
+          Opt.UNARY, Opt.LINKED),
+  SHOWSSANNOTATIONS(null, Opt.BOOLEAN, Opt.LINKED),
+
+  // Outputting files
+  IMAGE("Output an image of the open alignment window. Format is specified by the subval modifier, a following --type argument or guessed from the file extension. Valid formats/extensions are: svg, png, eps, html, biojs.",
+          Opt.STRING, Opt.LINKED, Opt.ALLOWSUBSTITUTIONS, Opt.ALLOWALL,
+          Opt.REQUIREINPUT),
+  TYPE("Set the image format for the preceding --image to name. Valid values for name are: svg, png, eps, html, biojs.",
+          Opt.STRING, Opt.LINKED, Opt.ALLOWALL),
+  TEXTRENDERER(
+          "Sets whether text in a vector image format (SVG, HTML, EPS) should be rendered as text or vector line-art. Possible values for name are: text, lineart.",
+          Opt.STRING, Opt.LINKED, Opt.ALLOWALL),
+  OUTPUT("Export the open alignment to file filename. The format name is specified by the subval modifier format=name, a following --format name argument or guessed from the file extension. Valid format names (and file extensions) are:\n"
+          + "fasta (fa, fasta, mfa, fastq),\n" + "pfam (pfam),\n"
+          + "stockholm (sto, stk),\n" + "pir (pir),\n" + "blc (blc),\n"
+          + "amsa (amsa),\n" + "json (json),\n" + "pileup (pileup),\n"
+          + "msf (msf),\n" + "clustal (aln),\n" + "phylip (phy),\n"
+          + "jalview (jvp, jar).", Opt.STRING, Opt.LINKED,
+          Opt.ALLOWSUBSTITUTIONS, Opt.ALLOWALL, Opt.REQUIREINPUT),
+  FORMAT("Sets the format for the preceding --output file. Valid formats are:\n"
+          + "fasta,\n" + "pfam,\n" + "stockholm,\n" + "pir,\n" + "blc,\n"
+          + "amsa,\n" + "json,\n" + "pileup,\n" + "msf,\n" + "clustal,\n"
+          + "phylip,\n" + "jalview.", Opt.STRING, Opt.LINKED, Opt.ALLOWALL),
+  GROOVY("Process a groovy script in the file for the open alignment.",
+          Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.ALLOWSUBSTITUTIONS),
+  BACKUPS("Enable (or disable) writing backup files when saving an ‑‑output file. This applies to the current open alignment -- to apply to all ‑‑output and ‑‑image files, use after ‑‑all.",
+          true, Opt.BOOLEAN, Opt.LINKED, Opt.ALLOWALL),
+  OVERWRITE(
+          "Enable (or disable) overwriting of output files without backups enabled. This applies to the current open alignment -- to apply to all ‑‑output and ‑‑image files, use after ‑‑all.",
+          Opt.BOOLEAN, Opt.LINKED, Opt.ALLOWALL),
+  CLOSE("Close the current open alignment window. This occurs after other output arguments. This applies to the current open alignment -- to apply to all ‑‑output and ‑‑image files, use after ‑‑all.",
+          Opt.UNARY, Opt.LINKED, Opt.ALLOWALL),
+
+  // controlling flow of arguments
+  NEW("Move on to a new alignment window. This will ensure --append will start a new alignment window and other linked arguments will apply to the new alignment window.",
+          Opt.UNARY, Opt.MULTI, Opt.NOACTION, Opt.INCREMENTDEFAULTCOUNTER),
+  SUBSTITUTIONS(
+          "The following argument values allow (or don't allow) subsituting filename parts. This is initially true. Valid substitutions are {basename} - the filename-without-extension of the currently --opened file (or first --appended file),\n"
+                  + "{dirname}, - the directory (folder) name of the currently --opened file (or first --appended file),\n"
+                  + "{argfilebasename} - the filename-without-extension of the current --argfile,\n"
+                  + "{argfiledirname} - the directory (folder) name of the current --argfile,\n"
+                  + "{n} - the value of the index counter (starting at 0).\n"
+                  + "{++n} - increase and substitute the value of the index counter,\n"
+                  + "{} - the value of the current alignment window default index.",
+          true, Opt.BOOLEAN, Opt.MULTI, Opt.NOACTION),
+  ARGFILE("Open one or more files filename and read, line-by-line, as arguments to Jalview.\n"
+          + "Note that if you use one or more --argfile arguments then all other non-initialising arguments will be ignored.",
+          Opt.STRING, Opt.MULTI, Opt.BOOTSTRAP, Opt.GLOB,
+          Opt.ALLOWSUBSTITUTIONS),
+  NPP("Increase the index counter used in argument value substitutions.",
+          Opt.UNARY, Opt.MULTI, Opt.NOACTION),
+  ALL("Apply the following output arguments to all sets of linked arguments.",
+          Opt.BOOLEAN, Opt.MULTI, Opt.NOACTION),
+  QUIT("After all files have been opened, appended and output, quit Jalview. In ‑‑headless mode this already happens.",
+          Opt.UNARY),
+
+  // secret options
+  TESTOUTPUT(
+          "Allow specific stdout information.  For testing purposes only.",
+          Opt.UNARY, Opt.BOOTSTRAP, Opt.SECRET), // do not show this to the user
+  SETPROP("Set an individual Java System property.", Opt.STRING, Opt.MULTI,
+          Opt.BOOTSTRAP, Opt.SECRET), // not in use yet
+  NIL("This argument does nothing on its own, but can be used with linkedIds.",
+          Opt.UNARY, Opt.LINKED, Opt.MULTI, Opt.NOACTION, Opt.SECRET),
+
+  // private options (inserted during arg processing)
+  SETARGFILE(
+          "Sets the current value of the argfilename.  Inserted before argfilecontents.",
+          Opt.UNARY, Opt.LINKED, Opt.STRING, Opt.MULTI, Opt.PRIVATE,
+          Opt.NOACTION),
+  UNSETARGFILE(
+          "Unsets the current value of the argfilename.  Inserted after argfile contents.",
+          Opt.UNARY, Opt.LINKED, Opt.MULTI, Opt.PRIVATE, Opt.NOACTION),
+
+  // these last two have no purpose in the normal Jalview application but are
+  // used by jalview.bin.Launcher to set memory settings. They are not used by
+  // argparser but are here for Usage statement reasons.
+  JVMMEMPC(
+          "Only available with standalone executable jar or jalview.bin.Launcher.\n"
+                  + "Limit maximum heap size (memory) to PERCENT% of total physical memory detected. This defaults to 90 if total physical memory can be detected.\n"
+                  + "The equals sign (\"=\") separator must be used with no spaces.",
+          Opt.NOACTION, Opt.BOOTSTRAP, Opt.STRING),
+  JVMMEMMAX(
+          "Only available with standalone executable jar or jalview.bin.Launcher.\n"
+                  + "Limit maximum heap size (memory) to MAXMEMORY. MAXMEMORY can be specified in bytes, kilobytes(k), megabytes(m), gigabytes(g) or if you're lucky enough, terabytes(t). This defaults to 32g if total physical memory can be detected, or to 8g if total physical memory cannot be detected.\n"
+                  + "The equals sign (\"=\") separator must be used with no spaces.",
+          Opt.NOACTION, Opt.BOOTSTRAP, Opt.STRING),
+
+  ;
 
   public static enum Opt
   {
@@ -53,6 +222,8 @@ public enum Arg
                         // SubVals and values.
     PRIVATE, // This Arg is used internally, and cannot be specified by the
              // user.
+    SECRET, // This Arg is used by development processes and although it can be
+            // set by the user, it is not displayed to the user.
     ALLOWALL, // This Arg can use the '*' linkedId to apply to all known
               // linkedIds
     INCREMENTDEFAULTCOUNTER, // If an Arg has this option and the default
@@ -63,88 +234,6 @@ public enum Arg
                   // input (i.e. --open or --append)
   }
 
-  static
-  {
-    HELP.setOptions("Display this help message", Opt.UNARY, Opt.BOOTSTRAP);
-    CALCULATION.setOptions(true, Opt.BOOLEAN); // default "true" implies only
-                                               // expecting "--nocalculation"
-    MENUBAR.setOptions(true, Opt.BOOLEAN);
-    STATUS.setOptions(true, Opt.BOOLEAN);
-    SHOWOVERVIEW.setOptions(Opt.UNARY, Opt.LINKED);
-    SHOWANNOTATIONS.setOptions(Opt.BOOLEAN, Opt.LINKED);
-    COLOUR.setOptions(Opt.STRING, Opt.LINKED);
-    FEATURES.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI,
-            Opt.ALLOWSUBSTITUTIONS);
-    ANNOTATIONS.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI,
-            Opt.ALLOWSUBSTITUTIONS);
-    TREE.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI,
-            Opt.ALLOWSUBSTITUTIONS);
-    GROOVY.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI,
-            Opt.ALLOWSUBSTITUTIONS);
-    GROUPS.setOptions(Opt.STRING, Opt.LINKED);
-    HEADLESS.setOptions(Opt.UNARY, Opt.BOOTSTRAP);
-    TESTOUTPUT.setOptions(Opt.UNARY, Opt.BOOTSTRAP);
-    JABAWS.setOptions(Opt.STRING, Opt.BOOTSTRAP);
-    DISPLAY.setOptions(true, Opt.BOOLEAN);
-    NEWS.setOptions(true, Opt.BOOLEAN, Opt.BOOTSTRAP);
-    SPLASH.setOptions(true, Opt.BOOLEAN, Opt.BOOTSTRAP);
-    SORTBYTREE.setOptions(true, Opt.LINKED, Opt.BOOLEAN);
-    QUESTIONNAIRE.setOptions(true, Opt.BOOLEAN, Opt.BOOTSTRAP);
-    USAGESTATS.setOptions(true, Opt.BOOLEAN, Opt.BOOTSTRAP);
-    WEBSERVICEDISCOVERY.setOptions(true, Opt.BOOLEAN, Opt.BOOTSTRAP);
-    APPEND.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI, Opt.GLOB,
-            Opt.ALLOWSUBSTITUTIONS, Opt.INPUT);
-    OPEN.setOptions(Opt.STRING, Opt.LINKED, Opt.INCREMENTDEFAULTCOUNTER,
-            Opt.MULTI, Opt.GLOB, Opt.ALLOWSUBSTITUTIONS, Opt.INPUT);
-    PROPS.setOptions(Opt.STRING, Opt.BOOTSTRAP);
-    SETPROP.setOptions(Opt.STRING, Opt.MULTI, Opt.BOOTSTRAP);
-
-    VDOC.setOptions(Opt.UNARY);
-    VSESS.setOptions(Opt.UNARY);
-
-    OUTPUT.setOptions(Opt.STRING, Opt.LINKED, Opt.ALLOWSUBSTITUTIONS,
-            Opt.ALLOWALL, Opt.REQUIREINPUT);
-
-    SSANNOTATIONS.setOptions(Opt.BOOLEAN, Opt.LINKED);
-    NOTEMPFAC.setOptions(Opt.UNARY, Opt.LINKED);
-    TEMPFAC.setOptions(Opt.STRING, Opt.LINKED);
-    TITLE.setOptions(Opt.STRING, Opt.LINKED);
-    PAEMATRIX.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI,
-            Opt.ALLOWSUBSTITUTIONS);
-    SEQID.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI,
-            Opt.ALLOWSUBSTITUTIONS);
-    NOSTRUCTURE.setOptions(Opt.UNARY, Opt.LINKED);
-    STRUCTURE.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI,
-            Opt.ALLOWSUBSTITUTIONS);
-    STRUCTUREVIEWER.setOptions(Opt.STRING, Opt.LINKED, Opt.MULTI);
-    WRAP.setOptions(Opt.BOOLEAN, Opt.LINKED);
-    IMAGE.setOptions(Opt.STRING, Opt.LINKED, Opt.ALLOWSUBSTITUTIONS,
-            Opt.ALLOWALL, Opt.REQUIREINPUT);
-    TYPE.setOptions(Opt.STRING, Opt.LINKED, Opt.ALLOWALL);
-    FORMAT.setOptions(Opt.STRING, Opt.LINKED, Opt.ALLOWALL);
-    TEXTRENDERER.setOptions(Opt.STRING, Opt.LINKED, Opt.ALLOWALL);
-    QUIT.setOptions(Opt.UNARY);
-    OVERWRITE.setOptions(Opt.BOOLEAN, Opt.LINKED, Opt.ALLOWALL);
-    BACKUPS.setOptions(true, Opt.BOOLEAN, Opt.LINKED, Opt.ALLOWALL);
-    CLOSE.setOptions(Opt.UNARY, Opt.LINKED, Opt.ALLOWALL);
-    DEBUG.setOptions(Opt.BOOLEAN, Opt.BOOTSTRAP);
-    TRACE.setOptions(Opt.BOOLEAN, Opt.BOOTSTRAP);
-    QUIET.setOptions(Opt.UNARY, Opt.MULTI, Opt.BOOTSTRAP);
-    ARGFILE.setOptions(Opt.STRING, Opt.MULTI, Opt.BOOTSTRAP, Opt.GLOB,
-            Opt.ALLOWSUBSTITUTIONS);
-    NEW.setOptions(Opt.UNARY, Opt.MULTI, Opt.NOACTION,
-            Opt.INCREMENTDEFAULTCOUNTER);
-    NPP.setOptions(Opt.UNARY, Opt.MULTI, Opt.NOACTION);
-    SUBSTITUTIONS.setOptions(Opt.BOOLEAN, Opt.MULTI, Opt.NOACTION);
-    INITSUBSTITUTIONS.setOptions(true, Opt.BOOLEAN, Opt.BOOTSTRAP,
-            Opt.NOACTION); // defaulting substitutions to true
-    NIL.setOptions(Opt.UNARY, Opt.LINKED, Opt.MULTI, Opt.NOACTION);
-    SETARGFILE.setOptions(Opt.STRING, Opt.MULTI, Opt.PRIVATE, Opt.NOACTION);
-    UNSETARGFILE.setOptions(Opt.MULTI, Opt.PRIVATE, Opt.NOACTION);
-    ALL.setOptions(Opt.BOOLEAN, Opt.MULTI, Opt.NOACTION);
-
-  }
-
   private final String[] argNames;
 
   private Opt[] argOptions;
@@ -153,20 +242,32 @@ public enum Arg
 
   private String description = null;
 
-  private Arg()
+  private Arg(String description, Opt... options)
   {
-    this(new String[0]);
+    this(null, description, false, options);
   }
 
-  private Arg(String... names)
+  private Arg(String description, boolean defaultBoolean, Opt... options)
   {
-    int length = (names == null || names.length == 0
-            || (names.length == 1 && names[0] == null)) ? 1
-                    : names.length + 1;
-    this.argNames = new String[length];
-    this.argNames[0] = this.getName();
-    if (length > 1)
-      System.arraycopy(names, 0, this.argNames, 1, names.length);
+    this(null, description, defaultBoolean, options);
+  }
+
+  private Arg(String alternativeName, String description, Opt... options)
+  {
+    this(alternativeName, description, false, options);
+  }
+
+  private Arg(String alternativeName, String description,
+          boolean defaultBoolean, Opt... options)
+  {
+    this.argNames = alternativeName != null
+            ? new String[]
+            { this.getName(), alternativeName }
+            : new String[]
+            { this.getName() };
+    this.description = description;
+    this.defaultBoolValue = defaultBoolean;
+    this.setOptions(options);
   }
 
   public String argString()
@@ -238,24 +339,6 @@ public enum Arg
 
   protected void setOptions(Opt... options)
   {
-    setOptions("", false, options);
-  }
-
-  protected void setOptions(String desc, Opt... options)
-  {
-    setOptions(desc, false, options);
-  }
-
-  protected void setOptions(boolean defaultBoolValue, Opt... options)
-  {
-    setOptions("", defaultBoolValue, options);
-  }
-
-  protected void setOptions(String desc, boolean defaultBoolValue,
-          Opt... options)
-  {
-    this.description = desc;
-    this.defaultBoolValue = defaultBoolValue;
     this.argOptions = options;
   }
 
@@ -264,11 +347,6 @@ public enum Arg
     return defaultBoolValue;
   }
 
-  private void setDescription(String d)
-  {
-    description = d;
-  }
-
   protected String getDescription()
   {
     return description;
@@ -289,14 +367,24 @@ public enum Arg
   {
     StringBuilder sb = new StringBuilder();
 
-    sb.append("Usage: jalview [args]");
+    sb.append(ChannelProperties.getProperty("app_name"));
+    String version = Cache.getDefault("VERSION", null);
+    if (version != null)
+    {
+      sb.append(" version ");
+      sb.append(Cache.getDefault("VERSION", "unknown"));
+    }
+    sb.append(System.lineSeparator());
+    sb.append("Usage: jalview [files...] [args]");
+    sb.append(System.lineSeparator());
     sb.append(System.lineSeparator());
 
     int maxArgLength = 0;
     for (Arg a : EnumSet.allOf(Arg.class))
     {
-      if (a.hasOption(Opt.PRIVATE))
+      if (a.hasOption(Opt.PRIVATE) || a.hasOption(Opt.SECRET))
         continue;
+
       StringBuilder argSb = new StringBuilder();
       argSb.append(a.hasOption(Opt.BOOLEAN) ? booleanArgString(a)
               : a.argString());
@@ -309,15 +397,34 @@ public enum Arg
     // might want to sort these
     for (Arg a : EnumSet.allOf(Arg.class))
     {
-      if (a.hasOption(Opt.PRIVATE))
+      if (a.hasOption(Opt.PRIVATE) || a.hasOption(Opt.SECRET))
         continue;
       StringBuilder argSb = new StringBuilder();
       argSb.append(a.hasOption(Opt.BOOLEAN) ? booleanArgString(a)
               : a.argString());
       if (a.hasOption(Opt.STRING))
         argSb.append("=value");
-      sb.append(String.format("%-" + maxArgLength + "s  - %s",
-              argSb.toString(), a.getDescription()));
+      Iterator<String> descLines = null;
+      if (a.getDescription() != null)
+      {
+        descLines = Arrays.stream(a.getDescription().split("\\n"))
+                .iterator();
+      }
+      sb.append(String.format("%-" + maxArgLength + "s", argSb.toString()));
+      boolean first = true;
+      if (descLines != null)
+      {
+        while (descLines.hasNext())
+        {
+          if (first)
+            sb.append(" - ");
+          else
+            sb.append(" ".repeat(maxArgLength + 3));
+          sb.append(descLines.next());
+          sb.append(System.lineSeparator());
+          first = false;
+        }
+      }
 
       List<String> options = new ArrayList<>();
 
@@ -347,11 +454,26 @@ public enum Arg
         options.add("allows substitutions");
       }
 
+      if (a.hasOption(Opt.PRIVATE))
+      {
+        options.add("for internal use only");
+      }
+
+      if (a.hasOption(Opt.SECRET))
+      {
+        options.add("for development use only");
+      }
+
       if (options.size() > 0)
       {
-        sb.append(" (");
+        if (first)
+          sb.append(" - ");
+        else
+          sb.append(" ".repeat(maxArgLength + 3));
+        sb.append("(");
         sb.append(String.join("; ", options));
         sb.append(')');
+        sb.append(System.lineSeparator());
       }
       sb.append(System.lineSeparator());
     }