Преглед на файлове

Switch from CXXOPTS to CLI11. (#148)

We want to move to auto-formatting all our code, and it just seemed impossible
to make cxxopts usage tidy under clang-format's dominion. While trying to work
out its quirks, I realised that CLI11 did everything I wanted much better, and
so we've switched.

We're also going to chuck the usage of ExternalProject_Add(), at least for the
simplest use cases such as single-header include files. We'll just commit them
directly; that's kind of the whole point.

The one discipline we'll maintain is that commits that involve third_party/
should be as self-contained as possible (without breaking the app).
Pär Winzell преди 6 години
родител
ревизия
be1b75431d
променени са 6 файла, в които са добавени 4571 реда и са изтрити 300 реда
  1. 87 0
      .clang-format
  2. 2 13
      CMakeLists.txt
  3. 361 279
      src/FBX2glTF.cpp
  4. 7 7
      src/FBX2glTF.h
  5. 1 1
      src/gltf/Raw2Gltf.cpp
  6. 4113 0
      third_party/CLI11/CLI11.hpp

+ 87 - 0
.clang-format

@@ -0,0 +1,87 @@
+---
+AccessModifierOffset: -1
+AlignAfterOpenBracket: AlwaysBreak
+AlignConsecutiveAssignments: false
+AlignConsecutiveDeclarations: false
+AlignEscapedNewlinesLeft: true
+AlignOperands:   false
+AlignTrailingComments: false
+AllowAllParametersOfDeclarationOnNextLine: false
+AllowShortBlocksOnASingleLine: false
+AllowShortCaseLabelsOnASingleLine: false
+AllowShortFunctionsOnASingleLine: Empty
+AllowShortIfStatementsOnASingleLine: false
+AllowShortLoopsOnASingleLine: false
+AlwaysBreakAfterReturnType: None
+AlwaysBreakBeforeMultilineStrings: true
+AlwaysBreakTemplateDeclarations: true
+BinPackArguments: false
+BinPackParameters: false
+BraceWrapping:
+  AfterClass:      false
+  AfterControlStatement: false
+  AfterEnum:       false
+  AfterFunction:   false
+  AfterNamespace:  false
+  AfterObjCDeclaration: false
+  AfterStruct:     false
+  AfterUnion:      false
+  BeforeCatch:     false
+  BeforeElse:      false
+  IndentBraces:    false
+BreakBeforeBinaryOperators: None
+BreakBeforeBraces: Attach
+BreakBeforeTernaryOperators: true
+BreakConstructorInitializersBeforeComma: false
+BreakAfterJavaFieldAnnotations: false
+BreakStringLiterals: false
+ColumnLimit:     80
+CommentPragmas:  '^ IWYU pragma:'
+ConstructorInitializerAllOnOneLineOrOnePerLine: true
+ConstructorInitializerIndentWidth: 4
+ContinuationIndentWidth: 4
+Cpp11BracedListStyle: true
+DerivePointerAlignment: false
+DisableFormat:   false
+ForEachMacros:   [ FOR_EACH, FOR_EACH_ENUMERATE, FOR_EACH_KV, FOR_EACH_R, FOR_EACH_RANGE, FOR_EACH_RANGE_R, ]
+IncludeCategories:
+  - Regex:           '^<.*\.h(pp)?>'
+    Priority:        1
+  - Regex:           '^<.*'
+    Priority:        2
+  - Regex:           '.*'
+    Priority:        3
+IndentCaseLabels: true
+IndentWidth:     2
+IndentWrappedFunctionNames: false
+KeepEmptyLinesAtTheStartOfBlocks: false
+MacroBlockBegin: ''
+MacroBlockEnd:   ''
+MaxEmptyLinesToKeep: 1
+NamespaceIndentation: None
+ObjCBlockIndentWidth: 2
+ObjCSpaceAfterProperty: false
+ObjCSpaceBeforeProtocolList: false
+PenaltyBreakBeforeFirstCallParameter: 1
+PenaltyBreakComment: 300
+PenaltyBreakFirstLessLess: 120
+PenaltyBreakString: 1000
+PenaltyExcessCharacter: 1000000
+PenaltyReturnTypeOnItsOwnLine: 200
+PointerAlignment: Left
+ReflowComments:  true
+SortIncludes:    true
+SpaceAfterCStyleCast: false
+SpaceBeforeAssignmentOperators: true
+SpaceBeforeParens: ControlStatements
+SpaceInEmptyParentheses: false
+SpacesBeforeTrailingComments: 1
+SpacesInAngles:  false
+SpacesInContainerLiterals: true
+SpacesInCStyleCastParentheses: false
+SpacesInParentheses: false
+SpacesInSquareBrackets: false
+Standard:        Cpp11
+TabWidth:        8
+UseTab:          Never
+...

+ 2 - 13
CMakeLists.txt

@@ -130,17 +130,6 @@ ExternalProject_Add(CPPCodec
 )
 set(CPPCODEC_INCLUDE_DIR "${CMAKE_BINARY_DIR}/cppcodec/src/CPPCodec")
 
-# CXXOPTS
-ExternalProject_Add(CxxOpts
-  GIT_REPOSITORY https://github.com/jarro2783/cxxopts
-  GIT_TAG v1.4.4
-  PREFIX cxxopts
-  CONFIGURE_COMMAND ${CMAKE_COMMAND} -E echo "Skipping cxxopts configure step."
-  BUILD_COMMAND ${CMAKE_COMMAND} -E echo "Skipping cxxopts build step."
-  INSTALL_COMMAND ${CMAKE_COMMAND} -E echo "Skipping cxxopts install step."
-)
-set(CXXOPTS_INCLUDE_DIR "${CMAKE_BINARY_DIR}/cxxopts/src/CxxOpts/include")
-
 # FMT
 ExternalProject_Add(Fmt
   PREFIX fmt
@@ -223,6 +212,7 @@ set(LIB_SOURCE_FILES
         src/utils/Image_Utils.hpp
         src/utils/String_Utils.cpp
         src/utils/String_Utils.hpp
+        third_party/CLI11/CLI11.hpp
 )
 
 add_library(libFBX2glTF STATIC ${LIB_SOURCE_FILES})
@@ -236,7 +226,6 @@ add_dependencies(libFBX2glTF
   FiFoMap
   Json
   STB
-  CxxOpts
   CPPCodec
   Fmt
 )
@@ -300,7 +289,7 @@ if (Iconv_FOUND)
 endif()
 
 target_include_directories(appFBX2glTF PUBLIC
-  ${CXXOPTS_INCLUDE_DIR}
+  "third_party/CLI11"
 )
 target_link_libraries(appFBX2glTF libFBX2glTF)
 

+ 361 - 279
src/FBX2glTF.cpp

@@ -7,310 +7,392 @@
  * of patent rights can be found in the PATENTS file in the same directory.
  */
 
-#include <vector>
-#include <unordered_map>
-#include <map>
-#include <iostream>
 #include <fstream>
+#include <iostream>
+#include <map>
+#include <unordered_map>
+#include <vector>
 
-#if defined( __unix__ ) || defined( __APPLE__ )
+#if defined(__unix__) || defined(__APPLE__)
 
 #include <sys/stat.h>
 
 #define _stricmp strcasecmp
 #endif
 
-#include <cxxopts.hpp>
+#include <CLI11.hpp>
 
 #include "FBX2glTF.h"
-#include "utils/String_Utils.hpp"
-#include "utils/File_Utils.hpp"
 #include "fbx/Fbx2Raw.hpp"
 #include "gltf/Raw2Gltf.hpp"
+#include "utils/File_Utils.hpp"
+#include "utils/String_Utils.hpp"
 
 bool verboseOutput = false;
 
-int main(int argc, char *argv[])
-{
-    cxxopts::Options options(
-        "FBX2glTF",
-        fmt::sprintf("FBX2glTF %s: Generate a glTF 2.0 representation of an FBX model.", FBX2GLTF_VERSION)
-    );
-
-    std::string inputPath;
-    std::string outputPath;
-
-    std::vector<std::function<Vec2f(Vec2f)>> texturesTransforms;
-
-    GltfOptions gltfOptions;
-
-    options.positional_help("[<FBX File>]");
-    options.add_options()
-               (
-                   "i,input", "The FBX model to convert.",
-                   cxxopts::value<std::string>(inputPath))
-               (
-                   "o,output", "Where to generate the output, without suffix.",
-                   cxxopts::value<std::string>(outputPath))
-               (
-                   "e,embed", "Inline buffers as data:// URIs within generated non-binary glTF.",
-                   cxxopts::value<bool>(gltfOptions.embedResources))
-               (
-                   "b,binary", "Output a single binary format .glb file.",
-                   cxxopts::value<bool>(gltfOptions.outputBinary))
-               (
-                   "long-indices", "Whether to use 32-bit indices (never|auto|always).",
-                   cxxopts::value<std::vector<std::string>>())
-               (
-                   "d,draco", "Apply Draco mesh compression to geometries.",
-                   cxxopts::value<bool>(gltfOptions.draco.enabled))
-               (
-                   "draco-compression-level", "The compression level to tune Draco to, from 0 to 10. (default: 7)",
-                   cxxopts::value<int>(gltfOptions.draco.compressionLevel))
-               (
-                   "draco-bits-for-positions", "How many bits to quantize position to. (default: 14)",
-                   cxxopts::value<int>(gltfOptions.draco.quantBitsPosition))
-               (
-                   "draco-bits-for-uv", "How many bits to quantize UV coordinates to. (default: 10)",
-                   cxxopts::value<int>(gltfOptions.draco.quantBitsTexCoord))
-               (
-                   "draco-bits-for-normals", "How many bits to quantize normals to. (default: 10)",
-                   cxxopts::value<int>(gltfOptions.draco.quantBitsNormal))
-               (
-                   "draco-bits-for-colors", "How many bits to quantize color to. (default: 8)",
-                   cxxopts::value<int>(gltfOptions.draco.quantBitsColor))
-               (
-                   "draco-bits-for-other", "How many bits to quantize other vertex attributes to to. (default: 8)",
-                   cxxopts::value<int>(gltfOptions.draco.quantBitsGeneric))
-               (
-                   "compute-normals", "When to compute normals for vertices (never|broken|missing|always).",
-                   cxxopts::value<std::vector<std::string>>())
-               ("flip-u", "Flip all U texture coordinates.")
-               ("flip-v", "Flip all V texture coordinates (default behaviour!)")
-               ("no-flip-v", "Suppress the default flipping of V texture coordinates")
-               (
-                   "pbr-metallic-roughness", "Try to glean glTF 2.0 native PBR attributes from the FBX.",
-                   cxxopts::value<bool>(gltfOptions.usePBRMetRough))
-               (
-                   "khr-materials-unlit", "Use KHR_materials_unlit extension to specify Unlit shader.",
-                   cxxopts::value<bool>(gltfOptions.useKHRMatUnlit))
-               (
-                   "no-khr-punctual-lights", "Don't use KHR_punctual_lights extension to export lights.",
-                   cxxopts::value<bool>(gltfOptions.useKHRPunctualLights))
-               (
-                   "user-properties", "Transcribe FBX User Properties into glTF node and material 'extras'.",
-                   cxxopts::value<bool>(gltfOptions.enableUserProperties))
-               (
-                   "blend-shape-normals", "Include blend shape normals, if reported present by the FBX SDK.",
-                   cxxopts::value<bool>(gltfOptions.useBlendShapeNormals))
-               (
-                   "blend-shape-tangents", "Include blend shape tangents, if reported present by the FBX SDK.",
-                   cxxopts::value<bool>(gltfOptions.useBlendShapeTangents))
-               (
-                   "k,keep-attribute", "Used repeatedly to build a limiting set of vertex attributes to keep.",
-                   cxxopts::value<std::vector<std::string>>())
-               ("v,verbose", "Enable verbose output.")
-               ("h,help", "Show this help.")
-               ("V,version", "Display the current program version.");
-
-    try {
-        options.parse_positional("input");
-        options.parse(argc, argv);
-
-    } catch (const cxxopts::OptionException &e) {
-        fmt::printf(options.help());
-        return 1;
-    }
-
-    if (options.count("version")) {
-        fmt::printf("FBX2glTF version %s\nCopyright (c) 2016-2018 Oculus VR, LLC.\n", FBX2GLTF_VERSION);
-        return 0;
-    }
-
-    if (options.count("help")) {
-        fmt::printf(options.help());
-        return 0;
-    }
-
-    if (!options.count("input")) {
-        fmt::printf("You must supply a FBX file to convert.\n");
-        fmt::printf(options.help());
-        return 1;
-    }
-
-    if (options.count("verbose")) {
-        verboseOutput = true;
-    }
-
-    if (!gltfOptions.useKHRMatUnlit && !gltfOptions.usePBRMetRough) {
-        if (verboseOutput) {
-            fmt::printf("Defaulting to --pbr-metallic-roughness material support.\n");
-        }
-        gltfOptions.usePBRMetRough = true;
-    }
+int main(int argc, char* argv[]) {
+  GltfOptions gltfOptions;
 
-    if (gltfOptions.draco.compressionLevel != -1 &&
-        (gltfOptions.draco.compressionLevel < 1 || gltfOptions.draco.compressionLevel > 10)) {
-        fmt::printf("Draco compression level must lie in [1, 10].\n");
-        return 0;
-    }
-
-    if (options.count("flip-u") > 0) {
-        texturesTransforms.emplace_back([](Vec2f uv) { return Vec2f(1.0f - uv[0], uv[1]); });
-    }
-    if (options.count("flip-v") > 0) {
-        fmt::printf("Note: The --flip-v command switch is now default behaviour.\n");
-    }
-    if (options.count("no-flip-v") == 0) {
-        texturesTransforms.emplace_back([](Vec2f uv) { return Vec2f(uv[0], 1.0f - uv[1]); });
-    } else if (verboseOutput) {
-        fmt::printf("Suppressing --flip-v transformation of texture coordinates.\n");
-    }
+  CLI::App app{
+      fmt::sprintf(
+          "FBX2glTF %s: Generate a glTF 2.0 representation of an FBX model.",
+          FBX2GLTF_VERSION),
+      "FBX2glTF"};
 
-    for (const std::string &choice : options["long-indices"].as<std::vector<std::string>>()) {
-        if (choice == "never") {
-            gltfOptions.useLongIndices = UseLongIndicesOptions::NEVER;
-        } else if (choice == "auto") {
-            gltfOptions.useLongIndices = UseLongIndicesOptions::AUTO;
-        } else if (choice == "always") {
-            gltfOptions.useLongIndices = UseLongIndicesOptions::ALWAYS;
-        } else {
-            fmt::printf("Unknown --long-indices: %s\n", choice);
-            fmt::printf(options.help());
-            return 1;
-        }
-    }
+  app.add_flag(
+      "-v,--verbose",
+      verboseOutput,
+      "Include blend shape tangents, if reported present by the FBX SDK.");
 
-    if (options.count("compute-normals") > 0) {
-        for (const std::string &choice : options["compute-normals"].as<std::vector<std::string>>()) {
-            if (choice == "never") {
-                gltfOptions.computeNormals = ComputeNormalsOption::NEVER;
-            } else if (choice == "broken") {
-                gltfOptions.computeNormals = ComputeNormalsOption::BROKEN;
-            } else if (choice == "missing") {
-                gltfOptions.computeNormals = ComputeNormalsOption::MISSING;
-            } else if (choice == "always") {
-                gltfOptions.computeNormals = ComputeNormalsOption::ALWAYS;
-            } else {
-                fmt::printf("Unknown --compute-normals: %s\n", choice);
-                fmt::printf(options.help());
-                return 1;
-            }
+  app.add_flag_function("-V,--version", [&](size_t count) {
+    fmt::printf(
+        "FBX2glTF version %s\nCopyright (c) 2016-2018 Oculus VR, LLC.\n",
+        FBX2GLTF_VERSION);
+    exit(0);
+  });
+
+  std::string inputPath;
+  app.add_option("FBX Model", inputPath, "The FBX model to convert.")
+      ->check(CLI::ExistingFile);
+  app.add_option("-i,--input", inputPath, "The FBX model to convert.")
+      ->check(CLI::ExistingFile);
+
+  std::string outputPath;
+  app.add_option(
+      "-o,--output",
+      outputPath,
+      "Where to generate the output, without suffix.");
+
+  app.add_flag(
+      "-e,--embed",
+      gltfOptions.embedResources,
+      "Inline buffers as data:// URIs within generated non-binary glTF.");
+  app.add_flag(
+      "-b,--binary",
+      gltfOptions.outputBinary,
+      "Output a single binary format .glb file.");
+
+  app.add_option(
+         "--long-indices",
+         [&](std::vector<std::string> choices) -> bool {
+           for (const std::string choice : choices) {
+             if (choice == "never") {
+               gltfOptions.useLongIndices = UseLongIndicesOptions::NEVER;
+             } else if (choice == "auto") {
+               gltfOptions.useLongIndices = UseLongIndicesOptions::AUTO;
+             } else if (choice == "always") {
+               gltfOptions.useLongIndices = UseLongIndicesOptions::ALWAYS;
+             } else {
+               fmt::printf("Unknown --long-indices: %s\n", choice);
+               throw CLI::RuntimeError(1);
+             }
+           }
+           return true;
+         },
+         "Whether to use 32-bit indices.")
+      ->type_name("(never|auto|always)");
+
+  app.add_option(
+         "--compute-normals",
+         [&](std::vector<std::string> choices) -> bool {
+           for (const std::string choice : choices) {
+             if (choice == "never") {
+               gltfOptions.computeNormals = ComputeNormalsOption::NEVER;
+             } else if (choice == "broken") {
+               gltfOptions.computeNormals = ComputeNormalsOption::BROKEN;
+             } else if (choice == "missing") {
+               gltfOptions.computeNormals = ComputeNormalsOption::MISSING;
+             } else if (choice == "always") {
+               gltfOptions.computeNormals = ComputeNormalsOption::ALWAYS;
+             } else {
+               fmt::printf("Unknown --compute-normals option: %s\n", choice);
+               throw CLI::RuntimeError(1);
+             }
+           }
+           return true;
+         },
+         "When to compute vertex normals from mesh geometry.")
+      ->type_name("(never|broken|missing|always)");
+
+  std::vector<std::function<Vec2f(Vec2f)>> texturesTransforms;
+  app.add_flag_function(
+      "--flip-u",
+      [&](size_t count) {
+        if (count > 0) {
+          texturesTransforms.emplace_back(
+              [](Vec2f uv) { return Vec2f(1.0f - uv[0], uv[1]); });
+          if (verboseOutput) {
+            fmt::printf("Flipping texture coordinates in the 'U' dimension.\n");
+          }
         }
-    }
-
-    if (options.count("keep-attribute") > 0) {
-        gltfOptions.keepAttribs = RAW_VERTEX_ATTRIBUTE_JOINT_INDICES | RAW_VERTEX_ATTRIBUTE_JOINT_WEIGHTS;
-        for (std::string attribute : options["keep-attribute"].as<std::vector<std::string>>()) {
-            if (attribute == "position") { gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_POSITION; }
-            else if (attribute == "normal") { gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_NORMAL; }
-            else if (attribute == "tangent") { gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_TANGENT; }
-            else if (attribute == "binormal") { gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_BINORMAL; }
-            else if (attribute == "color") { gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_COLOR; }
-            else if (attribute == "uv0") { gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_UV0; }
-            else if (attribute == "uv1") { gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_UV1; }
-            else if (attribute == "auto") { gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_AUTO; }
-            else {
-                fmt::printf("Unknown --keep-attribute: %s\n", attribute);
-                fmt::printf("Valid choices are: position, normal, tangent, binormial, color, uv0, uv1, auto,\n");
-                return 1;
-            }
+      },
+      "Flip all U texture coordinates.");
+
+  app.add_flag("--no-flip-u", "Don't flip U texture coordinates.")
+      ->excludes("--flip-u");
+
+  app.add_flag_function(
+      "--no-flip-v",
+      [&](size_t count) {
+        if (count > 0) {
+          texturesTransforms.emplace_back(
+              [](Vec2f uv) { return Vec2f(uv[0], 1.0f - uv[1]); });
+          if (verboseOutput) {
+            fmt::printf("NOT flipping texture coordinates in the 'V' dimension.\n");
+          }
         }
-    }
-
-    if (gltfOptions.embedResources && gltfOptions.outputBinary) {
-        fmt::printf("Note: Ignoring --embed; it's meaningless with --binary.\n");
-    }
-
-    if (options.count("output") == 0) {
-        // if -o is not given, default to the basename of the .fbx
-        outputPath = fmt::format(".{}{}", (const char)StringUtils::GetPathSeparator(), StringUtils::GetFileBaseString(inputPath));
-
-        fmt::printf("outputPath = %s\n", outputPath);
-    }
-    std::string outputFolder; // the output folder in .gltf mode, not used for .glb
-    std::string modelPath; // the path of the actual .glb or .gltf file
-    if (gltfOptions.outputBinary) {
-        // in binary mode, we write precisely where we're asked
-        modelPath = outputPath + ".glb";
-
-    } else {
-        // in gltf mode, we create a folder and write into that
-        outputFolder = fmt::format("{}_out{}", outputPath.c_str(), (const char)StringUtils::GetPathSeparator());
-        modelPath = outputFolder + StringUtils::GetFileNameString(outputPath) + ".gltf";
-    }
-    if (!FileUtils::CreatePath(modelPath.c_str())) {
-        fmt::fprintf(stderr, "ERROR: Failed to create folder: %s'\n", outputFolder.c_str());
-        return 1;
-    }
-
-    ModelData *data_render_model = nullptr;
-    RawModel  raw;
-
+      },
+      "Flip all V texture coordinates.");
+  app.add_flag("--flip-v", "Don't flip U texture coordinates.")
+      ->excludes("--no-flip-v");
+
+  app.add_flag(
+         "--pbr-metallic-rougnness",
+         gltfOptions.usePBRMetRough,
+         "Try to glean glTF 2.0 native PBR attributes from the FBX.")
+      ->group("Materials");
+
+  app.add_flag(
+         "--khr-materials-unlit",
+         gltfOptions.useKHRMatUnlit,
+         "Use KHR_materials_unlit extension to request an unlit shader.")
+      ->group("Materials");
+
+  app.add_flag(
+      "--khr-lights-punctual",
+      gltfOptions.useKHRLightsPunctual,
+      "Use KHR_lights_punctual extension to request an unlit shader.");
+
+  app.add_flag(
+      "--user-properties",
+      gltfOptions.enableUserProperties,
+      "Transcribe FBX User Properties into glTF node and material 'extras'.");
+
+  app.add_flag(
+      "--blend-shape-normals",
+      gltfOptions.useBlendShapeNormals,
+      "Include blend shape normals, if reported present by the FBX SDK.");
+
+  app.add_flag(
+      "--blend-shape-tangents",
+      gltfOptions.useBlendShapeTangents,
+      "Include blend shape tangents, if reported present by the FBX SDK.");
+
+  app.add_option(
+         "-k,--keep-attribute",
+         [&](std::vector<std::string> attributes) -> bool {
+           gltfOptions.keepAttribs = RAW_VERTEX_ATTRIBUTE_JOINT_INDICES |
+               RAW_VERTEX_ATTRIBUTE_JOINT_WEIGHTS;
+           for (std::string attribute : attributes) {
+             if (attribute == "position") {
+               gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_POSITION;
+             } else if (attribute == "normal") {
+               gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_NORMAL;
+             } else if (attribute == "tangent") {
+               gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_TANGENT;
+             } else if (attribute == "binormal") {
+               gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_BINORMAL;
+             } else if (attribute == "color") {
+               gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_COLOR;
+             } else if (attribute == "uv0") {
+               gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_UV0;
+             } else if (attribute == "uv1") {
+               gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_UV1;
+             } else if (attribute == "auto") {
+               gltfOptions.keepAttribs |= RAW_VERTEX_ATTRIBUTE_AUTO;
+             } else {
+               fmt::printf("Unknown --keep-attribute option: %s\n", attribute);
+               throw CLI::RuntimeError(1);
+             }
+           }
+           return true;
+         },
+         "Used repeatedly to build a limiting set of vertex attributes to keep.")
+      ->type_size(-1)
+      ->type_name("(position|normal|tangent|binormial|color|uv0|uv1|auto)");
+
+  app.add_flag(
+         "-d,--draco",
+         gltfOptions.draco.enabled,
+         "Apply Draco mesh compression to geometries.")
+      ->group("Draco");
+
+  app.add_option(
+         "--draco-compression-level",
+         gltfOptions.draco.compressionLevel,
+         "The compression level to tune Draco to.",
+         true)
+      ->check(CLI::Range(0, 10))
+      ->group("Draco");
+
+  app.add_option(
+         "--draco-bits-for-position",
+         gltfOptions.draco.quantBitsPosition,
+         "How many bits to quantize position to.",
+         true)
+      ->check(CLI::Range(1, 32))
+      ->group("Draco");
+
+  app.add_option(
+         "--draco-bits-for-uv",
+         gltfOptions.draco.quantBitsTexCoord,
+         "How many bits to quantize UV coordinates to.",
+         true)
+      ->check(CLI::Range(1, 32))
+      ->group("Draco");
+
+  app.add_option(
+         "--draco-bits-for-normals",
+         gltfOptions.draco.quantBitsNormal,
+         "How many bits to quantize nornals to.",
+         true)
+      ->check(CLI::Range(1, 32))
+      ->group("Draco");
+
+  app.add_option(
+         "--draco-bits-for-colors",
+         gltfOptions.draco.quantBitsColor,
+         "How many bits to quantize colors to.",
+         true)
+      ->check(CLI::Range(1, 32))
+      ->group("Draco");
+
+  app.add_option(
+         "--draco-bits-for-other",
+         gltfOptions.draco.quantBitsGeneric,
+         "How many bits to quantize all other vertex attributes to.",
+         true)
+      ->check(CLI::Range(1, 32))
+      ->group("Draco");
+
+  CLI11_PARSE(app, argc, argv);
+
+  if (inputPath.empty()) {
+    fmt::printf("You must supply a FBX file to convert.\n");
+    exit(1);
+  }
+
+  if (!gltfOptions.useKHRMatUnlit && !gltfOptions.usePBRMetRough) {
     if (verboseOutput) {
-        fmt::printf("Loading FBX File: %s\n", inputPath);
-    }
-    if (!LoadFBXFile(raw, inputPath.c_str(), "png;jpg;jpeg")) {
-        fmt::fprintf(stderr, "ERROR:: Failed to parse FBX: %s\n", inputPath);
-        return 1;
-    }
-
-    if (!texturesTransforms.empty()) {
-        raw.TransformTextures(texturesTransforms);
-    }
-    raw.Condense();
-    raw.TransformGeometry(gltfOptions.computeNormals);
-
-    std::ofstream outStream; // note: auto-flushes in destructor
-    const auto streamStart = outStream.tellp();
-
-    outStream.open(modelPath, std::ios::trunc | std::ios::ate | std::ios::out | std::ios::binary);
-    if (outStream.fail()) {
-        fmt::fprintf(stderr, "ERROR:: Couldn't open file for writing: %s\n", modelPath.c_str());
-        return 1;
+      fmt::printf("Defaulting to --pbr-metallic-roughness material support.\n");
     }
-    data_render_model = Raw2Gltf(outStream, outputFolder, raw, gltfOptions);
-
-    if (gltfOptions.outputBinary) {
-        fmt::printf(
-            "Wrote %lu bytes of binary glTF to %s.\n",
-            (unsigned long) (outStream.tellp() - streamStart), modelPath);
-        delete data_render_model;
-        return 0;
-    }
-
+    gltfOptions.usePBRMetRough = true;
+  }
+
+  if (gltfOptions.embedResources && gltfOptions.outputBinary) {
+    fmt::printf("Note: Ignoring --embed; it's meaningless with --binary.\n");
+  }
+
+  if (outputPath.empty()) {
+    // if -o is not given, default to the basename of the .fbx
+    outputPath = fmt::format(
+        ".{}{}",
+        (const char)StringUtils::GetPathSeparator(),
+        StringUtils::GetFileBaseString(inputPath));
+  }
+  // the output folder in .gltf mode, not used for .glb
+  std::string outputFolder;
+
+  // the path of the actual .glb or .gltf file
+  std::string modelPath;
+  if (gltfOptions.outputBinary) {
+    // in binary mode, we write precisely where we're asked
+    modelPath = outputPath + ".glb";
+
+  } else {
+    // in gltf mode, we create a folder and write into that
+    outputFolder = fmt::format(
+        "{}_out{}",
+        outputPath.c_str(),
+        (const char)StringUtils::GetPathSeparator());
+    modelPath =
+        outputFolder + StringUtils::GetFileNameString(outputPath) + ".gltf";
+  }
+  if (!FileUtils::CreatePath(modelPath.c_str())) {
+    fmt::fprintf(
+        stderr, "ERROR: Failed to create folder: %s'\n", outputFolder.c_str());
+    return 1;
+  }
+
+  ModelData* data_render_model = nullptr;
+  RawModel raw;
+
+  if (verboseOutput) {
+    fmt::printf("Loading FBX File: %s\n", inputPath);
+  }
+  if (!LoadFBXFile(raw, inputPath.c_str(), "png;jpg;jpeg")) {
+    fmt::fprintf(stderr, "ERROR:: Failed to parse FBX: %s\n", inputPath);
+    return 1;
+  }
+
+  if (!texturesTransforms.empty()) {
+    raw.TransformTextures(texturesTransforms);
+  }
+  raw.Condense();
+  raw.TransformGeometry(gltfOptions.computeNormals);
+
+  std::ofstream outStream; // note: auto-flushes in destructor
+  const auto streamStart = outStream.tellp();
+
+  outStream.open(
+      modelPath,
+      std::ios::trunc | std::ios::ate | std::ios::out | std::ios::binary);
+  if (outStream.fail()) {
+    fmt::fprintf(
+        stderr,
+        "ERROR:: Couldn't open file for writing: %s\n",
+        modelPath.c_str());
+    return 1;
+  }
+  data_render_model = Raw2Gltf(outStream, outputFolder, raw, gltfOptions);
+
+  if (gltfOptions.outputBinary) {
     fmt::printf(
-        "Wrote %lu bytes of glTF to %s.\n",
-        (unsigned long) (outStream.tellp() - streamStart), modelPath);
-
-    if (gltfOptions.embedResources) {
-        // we're done: everything was inlined into the glTF JSON
-        delete data_render_model;
-        return 0;
-    }
-
-    assert(!outputFolder.empty());
-
-    const std::string binaryPath = outputFolder + extBufferFilename;
-    FILE *fp = fopen(binaryPath.c_str(), "wb");
-    if (fp == nullptr) {
-        fmt::fprintf(stderr, "ERROR:: Couldn't open file '%s' for writing.\n", binaryPath);
-        return 1;
-    }
+        "Wrote %lu bytes of binary glTF to %s.\n",
+        (unsigned long)(outStream.tellp() - streamStart),
+        modelPath);
+    delete data_render_model;
+    return 0;
+  }
 
-    if (data_render_model->binary->empty() == false)
-    {
-        const unsigned char *binaryData = &(*data_render_model->binary)[0];
-        unsigned long       binarySize  = data_render_model->binary->size();
-        if (fwrite(binaryData, binarySize, 1, fp) != 1) {
-            fmt::fprintf(stderr, "ERROR: Failed to write %lu bytes to file '%s'.\n", binarySize, binaryPath);
-            fclose(fp);
-            return 1;
-        }
-        fclose(fp);
-        fmt::printf("Wrote %lu bytes of binary data to %s.\n", binarySize, binaryPath);
-    }
+  fmt::printf(
+      "Wrote %lu bytes of glTF to %s.\n",
+      (unsigned long)(outStream.tellp() - streamStart),
+      modelPath);
 
+  if (gltfOptions.embedResources) {
+    // we're done: everything was inlined into the glTF JSON
     delete data_render_model;
     return 0;
+  }
+
+  assert(!outputFolder.empty());
+
+  const std::string binaryPath = outputFolder + extBufferFilename;
+  FILE* fp = fopen(binaryPath.c_str(), "wb");
+  if (fp == nullptr) {
+    fmt::fprintf(
+        stderr, "ERROR:: Couldn't open file '%s' for writing.\n", binaryPath);
+    return 1;
+  }
+
+  if (data_render_model->binary->empty() == false) {
+    const unsigned char* binaryData = &(*data_render_model->binary)[0];
+    unsigned long binarySize = data_render_model->binary->size();
+    if (fwrite(binaryData, binarySize, 1, fp) != 1) {
+      fmt::fprintf(
+          stderr,
+          "ERROR: Failed to write %lu bytes to file '%s'.\n",
+          binarySize,
+          binaryPath);
+      fclose(fp);
+      return 1;
+    }
+    fclose(fp);
+    fmt::printf(
+        "Wrote %lu bytes of binary data to %s.\n", binarySize, binaryPath);
+  }
+
+  delete data_render_model;
+  return 0;
 }

+ 7 - 7
src/FBX2glTF.h

@@ -76,12 +76,12 @@ struct GltfOptions
     /** Whether and how to use KHR_draco_mesh_compression to minimize static geometry size. */
     struct {
         bool enabled = false;
-        int compressionLevel = -1;
-        int quantBitsPosition = -1;
-        int quantBitsTexCoord = -1;
-        int quantBitsNormal = -1;
-        int quantBitsColor = -1;
-        int quantBitsGeneric = -1;
+        int compressionLevel = 7;
+        int quantBitsPosition = 14;
+        int quantBitsTexCoord = 10;
+        int quantBitsNormal = 10;
+        int quantBitsColor = 8;
+        int quantBitsGeneric = 8;
     } draco;
 
     /** Whether to include FBX User Properties as 'extras' metadata in glTF nodes. */
@@ -93,7 +93,7 @@ struct GltfOptions
     bool usePBRMetRough { false };
 
     /** Whether to include lights through the KHR_punctual_lights extension. */
-    bool useKHRPunctualLights { true };
+    bool useKHRLightsPunctual { true };
 
     /** Whether to include blend shape normals, if present according to the SDK. */
     bool useBlendShapeNormals { false };

+ 1 - 1
src/gltf/Raw2Gltf.cpp

@@ -631,7 +631,7 @@ ModelData *Raw2Gltf(
         // lights
         //
         std::vector<json> khrPunctualLights;
-        if (options.useKHRPunctualLights) {
+        if (options.useKHRLightsPunctual) {
             for (int i = 0; i < raw.GetLightCount(); i ++) {
                 const RawLight &light = raw.GetLight(i);
                 LightData::Type type;

+ 4113 - 0
third_party/CLI11/CLI11.hpp

@@ -0,0 +1,4113 @@
+#pragma once
+
+// CLI11: Version 1.6.2
+// Originally designed by Henry Schreiner
+// https://github.com/CLIUtils/CLI11
+//
+// This is a standalone header file generated by MakeSingleHeader.py in CLI11/scripts
+// from: v1.6.2
+//
+// From LICENSE:
+//
+// CLI11 1.6 Copyright (c) 2017-2018 University of Cincinnati, developed by Henry
+// Schreiner under NSF AWARD 1414736. All rights reserved.
+// 
+// Redistribution and use in source and binary forms of CLI11, with or without
+// modification, are permitted provided that the following conditions are met:
+// 
+// 1. Redistributions of source code must retain the above copyright notice, this
+//    list of conditions and the following disclaimer. 
+// 2. Redistributions in binary form must reproduce the above copyright notice,
+//    this list of conditions and the following disclaimer in the documentation
+//    and/or other materials provided with the distribution.
+// 3. Neither the name of the copyright holder nor the names of its contributors
+//    may be used to endorse or promote products derived from this software without
+//    specific prior written permission.
+// 
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
+// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
+// ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
+// ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+// Standard combined includes:
+
+#include <algorithm>
+#include <deque>
+#include <exception>
+#include <fstream>
+#include <functional>
+#include <iomanip>
+#include <iostream>
+#include <istream>
+#include <iterator>
+#include <locale>
+#include <map>
+#include <memory>
+#include <numeric>
+#include <set>
+#include <sstream>
+#include <stdexcept>
+#include <string>
+#include <sys/stat.h>
+#include <sys/types.h>
+#include <tuple>
+#include <type_traits>
+#include <utility>
+#include <vector>
+
+
+// Verbatim copy from CLI/Version.hpp:
+
+
+#define CLI11_VERSION_MAJOR 1
+#define CLI11_VERSION_MINOR 6
+#define CLI11_VERSION_PATCH 2
+#define CLI11_VERSION "1.6.2"
+
+
+
+
+// Verbatim copy from CLI/Macros.hpp:
+
+
+// The following version macro is very similar to the one in PyBind11
+#if !(defined(_MSC_VER) && __cplusplus == 199711L) && !defined(__INTEL_COMPILER)
+#if __cplusplus >= 201402L
+#define CLI11_CPP14
+#if __cplusplus >= 201703L
+#define CLI11_CPP17
+#if __cplusplus > 201703L
+#define CLI11_CPP20
+#endif
+#endif
+#endif
+#elif defined(_MSC_VER) && __cplusplus == 199711L
+// MSVC sets _MSVC_LANG rather than __cplusplus (supposedly until the standard is fully implemented)
+// Unless you use the /Zc:__cplusplus flag on Visual Studio 2017 15.7 Preview 3 or newer
+#if _MSVC_LANG >= 201402L
+#define CLI11_CPP14
+#if _MSVC_LANG > 201402L && _MSC_VER >= 1910
+#define CLI11_CPP17
+#if __MSVC_LANG > 201703L && _MSC_VER >= 1910
+#define CLI11_CPP20
+#endif
+#endif
+#endif
+#endif
+
+#if defined(CLI11_CPP14)
+#define CLI11_DEPRECATED(reason) [[deprecated(reason)]]
+#elif defined(_MSC_VER)
+#define CLI11_DEPRECATED(reason) __declspec(deprecated(reason))
+#else
+#define CLI11_DEPRECATED(reason) __attribute__((deprecated(reason)))
+#endif
+
+
+
+
+// Verbatim copy from CLI/Optional.hpp:
+
+#ifdef __has_include
+
+// You can explicitly enable or disable support
+// by defining these to 1 or 0.
+#if defined(CLI11_CPP17) && __has_include(<optional>) && \
+     !defined(CLI11_STD_OPTIONAL)
+#define CLI11_STD_OPTIONAL 1
+#endif
+
+#if defined(CLI11_CPP14) && __has_include(<experimental/optional>) && \
+    !defined(CLI11_EXPERIMENTAL_OPTIONAL) \
+    && (!defined(CLI11_STD_OPTIONAL) || CLI11_STD_OPTIONAL == 0)
+#define CLI11_EXPERIMENTAL_OPTIONAL 1
+#endif
+
+#if __has_include(<boost/optional.hpp>) && !defined(CLI11_BOOST_OPTIONAL)
+#include <boost/version.hpp>
+#if BOOST_VERSION >= 105800
+#define CLI11_BOOST_OPTIONAL 1
+#endif
+#endif
+
+#endif
+
+#if CLI11_STD_OPTIONAL
+#include <optional>
+#endif
+#if CLI11_EXPERIMENTAL_OPTIONAL
+#include <experimental/optional>
+#endif
+#if CLI11_BOOST_OPTIONAL
+#include <boost/optional.hpp>
+#endif
+
+
+// From CLI/Version.hpp:
+
+
+
+// From CLI/Macros.hpp:
+
+
+
+// From CLI/Optional.hpp:
+
+namespace CLI {
+
+#if CLI11_STD_OPTIONAL
+template <typename T> std::istream &operator>>(std::istream &in, std::optional<T> &val) {
+    T v;
+    in >> v;
+    val = v;
+    return in;
+}
+#endif
+
+#if CLI11_EXPERIMENTAL_OPTIONAL
+template <typename T> std::istream &operator>>(std::istream &in, std::experimental::optional<T> &val) {
+    T v;
+    in >> v;
+    val = v;
+    return in;
+}
+#endif
+
+#if CLI11_BOOST_OPTIONAL
+template <typename T> std::istream &operator>>(std::istream &in, boost::optional<T> &val) {
+    T v;
+    in >> v;
+    val = v;
+    return in;
+}
+#endif
+
+// Export the best optional to the CLI namespace
+#if CLI11_STD_OPTIONAL
+using std::optional;
+#elif CLI11_EXPERIMENTAL_OPTIONAL
+using std::experimental::optional;
+#elif CLI11_BOOST_OPTIONAL
+using boost::optional;
+#endif
+
+// This is true if any optional is found
+#if CLI11_STD_OPTIONAL || CLI11_EXPERIMENTAL_OPTIONAL || CLI11_BOOST_OPTIONAL
+#define CLI11_OPTIONAL 1
+#endif
+
+} // namespace CLI
+
+// From CLI/StringTools.hpp:
+
+namespace CLI {
+namespace detail {
+
+// Based on http://stackoverflow.com/questions/236129/split-a-string-in-c
+/// Split a string by a delim
+inline std::vector<std::string> split(const std::string &s, char delim) {
+    std::vector<std::string> elems;
+    // Check to see if empty string, give consistent result
+    if(s.empty())
+        elems.emplace_back("");
+    else {
+        std::stringstream ss;
+        ss.str(s);
+        std::string item;
+        while(std::getline(ss, item, delim)) {
+            elems.push_back(item);
+        }
+    }
+    return elems;
+}
+
+/// Simple function to join a string
+template <typename T> std::string join(const T &v, std::string delim = ",") {
+    std::ostringstream s;
+    size_t start = 0;
+    for(const auto &i : v) {
+        if(start++ > 0)
+            s << delim;
+        s << i;
+    }
+    return s.str();
+}
+
+/// Join a string in reverse order
+template <typename T> std::string rjoin(const T &v, std::string delim = ",") {
+    std::ostringstream s;
+    for(size_t start = 0; start < v.size(); start++) {
+        if(start > 0)
+            s << delim;
+        s << v[v.size() - start - 1];
+    }
+    return s.str();
+}
+
+// Based roughly on http://stackoverflow.com/questions/25829143/c-trim-whitespace-from-a-string
+
+/// Trim whitespace from left of string
+inline std::string &ltrim(std::string &str) {
+    auto it = std::find_if(str.begin(), str.end(), [](char ch) { return !std::isspace<char>(ch, std::locale()); });
+    str.erase(str.begin(), it);
+    return str;
+}
+
+/// Trim anything from left of string
+inline std::string &ltrim(std::string &str, const std::string &filter) {
+    auto it = std::find_if(str.begin(), str.end(), [&filter](char ch) { return filter.find(ch) == std::string::npos; });
+    str.erase(str.begin(), it);
+    return str;
+}
+
+/// Trim whitespace from right of string
+inline std::string &rtrim(std::string &str) {
+    auto it = std::find_if(str.rbegin(), str.rend(), [](char ch) { return !std::isspace<char>(ch, std::locale()); });
+    str.erase(it.base(), str.end());
+    return str;
+}
+
+/// Trim anything from right of string
+inline std::string &rtrim(std::string &str, const std::string &filter) {
+    auto it =
+        std::find_if(str.rbegin(), str.rend(), [&filter](char ch) { return filter.find(ch) == std::string::npos; });
+    str.erase(it.base(), str.end());
+    return str;
+}
+
+/// Trim whitespace from string
+inline std::string &trim(std::string &str) { return ltrim(rtrim(str)); }
+
+/// Trim anything from string
+inline std::string &trim(std::string &str, const std::string filter) { return ltrim(rtrim(str, filter), filter); }
+
+/// Make a copy of the string and then trim it
+inline std::string trim_copy(const std::string &str) {
+    std::string s = str;
+    return trim(s);
+}
+
+/// Make a copy of the string and then trim it, any filter string can be used (any char in string is filtered)
+inline std::string trim_copy(const std::string &str, const std::string &filter) {
+    std::string s = str;
+    return trim(s, filter);
+}
+/// Print a two part "help" string
+inline std::ostream &format_help(std::ostream &out, std::string name, std::string description, size_t wid) {
+    name = "  " + name;
+    out << std::setw(static_cast<int>(wid)) << std::left << name;
+    if(!description.empty()) {
+        if(name.length() >= wid)
+            out << "\n" << std::setw(static_cast<int>(wid)) << "";
+        out << description;
+    }
+    out << "\n";
+    return out;
+}
+
+/// Verify the first character of an option
+template <typename T> bool valid_first_char(T c) { return std::isalpha(c, std::locale()) || c == '_'; }
+
+/// Verify following characters of an option
+template <typename T> bool valid_later_char(T c) {
+    return std::isalnum(c, std::locale()) || c == '_' || c == '.' || c == '-';
+}
+
+/// Verify an option name
+inline bool valid_name_string(const std::string &str) {
+    if(str.empty() || !valid_first_char(str[0]))
+        return false;
+    for(auto c : str.substr(1))
+        if(!valid_later_char(c))
+            return false;
+    return true;
+}
+
+/// Return a lower case version of a string
+inline std::string to_lower(std::string str) {
+    std::transform(std::begin(str), std::end(str), std::begin(str), [](const std::string::value_type &x) {
+        return std::tolower(x, std::locale());
+    });
+    return str;
+}
+
+/// Split a string '"one two" "three"' into 'one two', 'three'
+inline std::vector<std::string> split_up(std::string str) {
+
+    std::vector<char> delims = {'\'', '\"'};
+    auto find_ws = [](char ch) { return std::isspace<char>(ch, std::locale()); };
+    trim(str);
+
+    std::vector<std::string> output;
+
+    while(!str.empty()) {
+        if(str[0] == '\'') {
+            auto end = str.find('\'', 1);
+            if(end != std::string::npos) {
+                output.push_back(str.substr(1, end - 1));
+                str = str.substr(end + 1);
+            } else {
+                output.push_back(str.substr(1));
+                str = "";
+            }
+        } else if(str[0] == '\"') {
+            auto end = str.find('\"', 1);
+            if(end != std::string::npos) {
+                output.push_back(str.substr(1, end - 1));
+                str = str.substr(end + 1);
+            } else {
+                output.push_back(str.substr(1));
+                str = "";
+            }
+
+        } else {
+            auto it = std::find_if(std::begin(str), std::end(str), find_ws);
+            if(it != std::end(str)) {
+                std::string value = std::string(str.begin(), it);
+                output.push_back(value);
+                str = std::string(it, str.end());
+            } else {
+                output.push_back(str);
+                str = "";
+            }
+        }
+        trim(str);
+    }
+
+    return output;
+}
+
+/// Add a leader to the beginning of all new lines (nothing is added
+/// at the start of the first line). `"; "` would be for ini files
+///
+/// Can't use Regex, or this would be a subs.
+inline std::string fix_newlines(std::string leader, std::string input) {
+    std::string::size_type n = 0;
+    while(n != std::string::npos && n < input.size()) {
+        n = input.find('\n', n);
+        if(n != std::string::npos) {
+            input = input.substr(0, n + 1) + leader + input.substr(n + 1);
+            n += leader.size();
+        }
+    }
+    return input;
+}
+
+/// Find and replace a subtring with another substring
+inline std::string find_and_replace(std::string str, std::string from, std::string to) {
+
+    size_t start_pos = 0;
+
+    while((start_pos = str.find(from, start_pos)) != std::string::npos) {
+        str.replace(start_pos, from.length(), to);
+        start_pos += to.length();
+    }
+
+    return str;
+}
+
+} // namespace detail
+} // namespace CLI
+
+// From CLI/Error.hpp:
+
+namespace CLI {
+
+// Use one of these on all error classes.
+// These are temporary and are undef'd at the end of this file.
+#define CLI11_ERROR_DEF(parent, name)                                                                                  \
+  protected:                                                                                                           \
+    name(std::string name, std::string msg, int exit_code) : parent(std::move(name), std::move(msg), exit_code) {}     \
+    name(std::string name, std::string msg, ExitCodes exit_code)                                                       \
+        : parent(std::move(name), std::move(msg), exit_code) {}                                                        \
+                                                                                                                       \
+  public:                                                                                                              \
+    name(std::string msg, ExitCodes exit_code) : parent(#name, std::move(msg), exit_code) {}                           \
+    name(std::string msg, int exit_code) : parent(#name, std::move(msg), exit_code) {}
+
+// This is added after the one above if a class is used directly and builds its own message
+#define CLI11_ERROR_SIMPLE(name)                                                                                       \
+    explicit name(std::string msg) : name(#name, msg, ExitCodes::name) {}
+
+/// These codes are part of every error in CLI. They can be obtained from e using e.exit_code or as a quick shortcut,
+/// int values from e.get_error_code().
+enum class ExitCodes {
+    Success = 0,
+    IncorrectConstruction = 100,
+    BadNameString,
+    OptionAlreadyAdded,
+    FileError,
+    ConversionError,
+    ValidationError,
+    RequiredError,
+    RequiresError,
+    ExcludesError,
+    ExtrasError,
+    ConfigError,
+    InvalidError,
+    HorribleError,
+    OptionNotFound,
+    ArgumentMismatch,
+    BaseClass = 127
+};
+
+// Error definitions
+
+/// @defgroup error_group Errors
+/// @brief Errors thrown by CLI11
+///
+/// These are the errors that can be thrown. Some of them, like CLI::Success, are not really errors.
+/// @{
+
+/// All errors derive from this one
+class Error : public std::runtime_error {
+    int exit_code;
+    std::string name{"Error"};
+
+  public:
+    int get_exit_code() const { return exit_code; }
+
+    std::string get_name() const { return name; }
+
+    Error(std::string name, std::string msg, int exit_code = static_cast<int>(ExitCodes::BaseClass))
+        : runtime_error(msg), exit_code(exit_code), name(std::move(name)) {}
+
+    Error(std::string name, std::string msg, ExitCodes exit_code) : Error(name, msg, static_cast<int>(exit_code)) {}
+};
+
+// Note: Using Error::Error constructors does not work on GCC 4.7
+
+/// Construction errors (not in parsing)
+class ConstructionError : public Error {
+    CLI11_ERROR_DEF(Error, ConstructionError)
+};
+
+/// Thrown when an option is set to conflicting values (non-vector and multi args, for example)
+class IncorrectConstruction : public ConstructionError {
+    CLI11_ERROR_DEF(ConstructionError, IncorrectConstruction)
+    CLI11_ERROR_SIMPLE(IncorrectConstruction)
+    static IncorrectConstruction PositionalFlag(std::string name) {
+        return IncorrectConstruction(name + ": Flags cannot be positional");
+    }
+    static IncorrectConstruction Set0Opt(std::string name) {
+        return IncorrectConstruction(name + ": Cannot set 0 expected, use a flag instead");
+    }
+    static IncorrectConstruction SetFlag(std::string name) {
+        return IncorrectConstruction(name + ": Cannot set an expected number for flags");
+    }
+    static IncorrectConstruction ChangeNotVector(std::string name) {
+        return IncorrectConstruction(name + ": You can only change the expected arguments for vectors");
+    }
+    static IncorrectConstruction AfterMultiOpt(std::string name) {
+        return IncorrectConstruction(
+            name + ": You can't change expected arguments after you've changed the multi option policy!");
+    }
+    static IncorrectConstruction MissingOption(std::string name) {
+        return IncorrectConstruction("Option " + name + " is not defined");
+    }
+    static IncorrectConstruction MultiOptionPolicy(std::string name) {
+        return IncorrectConstruction(name + ": multi_option_policy only works for flags and exact value options");
+    }
+};
+
+/// Thrown on construction of a bad name
+class BadNameString : public ConstructionError {
+    CLI11_ERROR_DEF(ConstructionError, BadNameString)
+    CLI11_ERROR_SIMPLE(BadNameString)
+    static BadNameString OneCharName(std::string name) { return BadNameString("Invalid one char name: " + name); }
+    static BadNameString BadLongName(std::string name) { return BadNameString("Bad long name: " + name); }
+    static BadNameString DashesOnly(std::string name) {
+        return BadNameString("Must have a name, not just dashes: " + name);
+    }
+    static BadNameString MultiPositionalNames(std::string name) {
+        return BadNameString("Only one positional name allowed, remove: " + name);
+    }
+};
+
+/// Thrown when an option already exists
+class OptionAlreadyAdded : public ConstructionError {
+    CLI11_ERROR_DEF(ConstructionError, OptionAlreadyAdded)
+    explicit OptionAlreadyAdded(std::string name)
+        : OptionAlreadyAdded(name + " is already added", ExitCodes::OptionAlreadyAdded) {}
+    static OptionAlreadyAdded Requires(std::string name, std::string other) {
+        return OptionAlreadyAdded(name + " requires " + other, ExitCodes::OptionAlreadyAdded);
+    }
+    static OptionAlreadyAdded Excludes(std::string name, std::string other) {
+        return OptionAlreadyAdded(name + " excludes " + other, ExitCodes::OptionAlreadyAdded);
+    }
+};
+
+// Parsing errors
+
+/// Anything that can error in Parse
+class ParseError : public Error {
+    CLI11_ERROR_DEF(Error, ParseError)
+};
+
+// Not really "errors"
+
+/// This is a successful completion on parsing, supposed to exit
+class Success : public ParseError {
+    CLI11_ERROR_DEF(ParseError, Success)
+    Success() : Success("Successfully completed, should be caught and quit", ExitCodes::Success) {}
+};
+
+/// -h or --help on command line
+class CallForHelp : public ParseError {
+    CLI11_ERROR_DEF(ParseError, CallForHelp)
+    CallForHelp() : CallForHelp("This should be caught in your main function, see examples", ExitCodes::Success) {}
+};
+
+/// Usually somethign like --help-all on command line
+class CallForAllHelp : public ParseError {
+    CLI11_ERROR_DEF(ParseError, CallForAllHelp)
+    CallForAllHelp()
+        : CallForAllHelp("This should be caught in your main function, see examples", ExitCodes::Success) {}
+};
+
+/// Does not output a diagnostic in CLI11_PARSE, but allows to return from main() with a specific error code.
+class RuntimeError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, RuntimeError)
+    explicit RuntimeError(int exit_code = 1) : RuntimeError("Runtime error", exit_code) {}
+};
+
+/// Thrown when parsing an INI file and it is missing
+class FileError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, FileError)
+    CLI11_ERROR_SIMPLE(FileError)
+    static FileError Missing(std::string name) { return FileError(name + " was not readable (missing?)"); }
+};
+
+/// Thrown when conversion call back fails, such as when an int fails to coerce to a string
+class ConversionError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, ConversionError)
+    CLI11_ERROR_SIMPLE(ConversionError)
+    ConversionError(std::string member, std::string name)
+        : ConversionError("The value " + member + " is not an allowed value for " + name) {}
+    ConversionError(std::string name, std::vector<std::string> results)
+        : ConversionError("Could not convert: " + name + " = " + detail::join(results)) {}
+    static ConversionError TooManyInputsFlag(std::string name) {
+        return ConversionError(name + ": too many inputs for a flag");
+    }
+    static ConversionError TrueFalse(std::string name) {
+        return ConversionError(name + ": Should be true/false or a number");
+    }
+};
+
+/// Thrown when validation of results fails
+class ValidationError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, ValidationError)
+    CLI11_ERROR_SIMPLE(ValidationError)
+    explicit ValidationError(std::string name, std::string msg) : ValidationError(name + ": " + msg) {}
+};
+
+/// Thrown when a required option is missing
+class RequiredError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, RequiredError)
+    explicit RequiredError(std::string name) : RequiredError(name + " is required", ExitCodes::RequiredError) {}
+    static RequiredError Subcommand(size_t min_subcom) {
+        if(min_subcom == 1)
+            return RequiredError("A subcommand");
+        else
+            return RequiredError("Requires at least " + std::to_string(min_subcom) + " subcommands",
+                                 ExitCodes::RequiredError);
+    }
+};
+
+/// Thrown when the wrong number of arguments has been received
+class ArgumentMismatch : public ParseError {
+    CLI11_ERROR_DEF(ParseError, ArgumentMismatch)
+    CLI11_ERROR_SIMPLE(ArgumentMismatch)
+    ArgumentMismatch(std::string name, int expected, size_t recieved)
+        : ArgumentMismatch(expected > 0 ? ("Expected exactly " + std::to_string(expected) + " arguments to " + name +
+                                           ", got " + std::to_string(recieved))
+                                        : ("Expected at least " + std::to_string(-expected) + " arguments to " + name +
+                                           ", got " + std::to_string(recieved)),
+                           ExitCodes::ArgumentMismatch) {}
+
+    static ArgumentMismatch AtLeast(std::string name, int num) {
+        return ArgumentMismatch(name + ": At least " + std::to_string(num) + " required");
+    }
+    static ArgumentMismatch TypedAtLeast(std::string name, int num, std::string type) {
+        return ArgumentMismatch(name + ": " + std::to_string(num) + " required " + type + " missing");
+    }
+};
+
+/// Thrown when a requires option is missing
+class RequiresError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, RequiresError)
+    RequiresError(std::string curname, std::string subname)
+        : RequiresError(curname + " requires " + subname, ExitCodes::RequiresError) {}
+};
+
+/// Thrown when an excludes option is present
+class ExcludesError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, ExcludesError)
+    ExcludesError(std::string curname, std::string subname)
+        : ExcludesError(curname + " excludes " + subname, ExitCodes::ExcludesError) {}
+};
+
+/// Thrown when too many positionals or options are found
+class ExtrasError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, ExtrasError)
+    explicit ExtrasError(std::vector<std::string> args)
+        : ExtrasError((args.size() > 1 ? "The following arguments were not expected: "
+                                       : "The following argument was not expected: ") +
+                          detail::rjoin(args, " "),
+                      ExitCodes::ExtrasError) {}
+};
+
+/// Thrown when extra values are found in an INI file
+class ConfigError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, ConfigError)
+    CLI11_ERROR_SIMPLE(ConfigError)
+    static ConfigError Extras(std::string item) { return ConfigError("INI was not able to parse " + item); }
+    static ConfigError NotConfigurable(std::string item) {
+        return ConfigError(item + ": This option is not allowed in a configuration file");
+    }
+};
+
+/// Thrown when validation fails before parsing
+class InvalidError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, InvalidError)
+    explicit InvalidError(std::string name)
+        : InvalidError(name + ": Too many positional arguments with unlimited expected args", ExitCodes::InvalidError) {
+    }
+};
+
+/// This is just a safety check to verify selection and parsing match - you should not ever see it
+/// Strings are directly added to this error, but again, it should never be seen.
+class HorribleError : public ParseError {
+    CLI11_ERROR_DEF(ParseError, HorribleError)
+    CLI11_ERROR_SIMPLE(HorribleError)
+};
+
+// After parsing
+
+/// Thrown when counting a non-existent option
+class OptionNotFound : public Error {
+    CLI11_ERROR_DEF(Error, OptionNotFound)
+    explicit OptionNotFound(std::string name) : OptionNotFound(name + " not found", ExitCodes::OptionNotFound) {}
+};
+
+#undef CLI11_ERROR_DEF
+#undef CLI11_ERROR_SIMPLE
+
+/// @}
+
+} // namespace CLI
+
+// From CLI/TypeTools.hpp:
+
+namespace CLI {
+
+// Type tools
+
+/// A copy of enable_if_t from C++14, compatible with C++11.
+///
+/// We could check to see if C++14 is being used, but it does not hurt to redefine this
+/// (even Google does this: https://github.com/google/skia/blob/master/include/private/SkTLogic.h)
+/// It is not in the std namespace anyway, so no harm done.
+
+template <bool B, class T = void> using enable_if_t = typename std::enable_if<B, T>::type;
+
+/// Check to see if something is a vector (fail check by default)
+template <typename T> struct is_vector { static const bool value = false; };
+
+/// Check to see if something is a vector (true if actually a vector)
+template <class T, class A> struct is_vector<std::vector<T, A>> { static bool const value = true; };
+
+/// Check to see if something is bool (fail check by default)
+template <typename T> struct is_bool { static const bool value = false; };
+
+/// Check to see if something is bool (true if actually a bool)
+template <> struct is_bool<bool> { static bool const value = true; };
+
+namespace detail {
+// Based generally on https://rmf.io/cxx11/almost-static-if
+/// Simple empty scoped class
+enum class enabler {};
+
+/// An instance to use in EnableIf
+constexpr enabler dummy = {};
+
+// Type name print
+
+/// Was going to be based on
+///  http://stackoverflow.com/questions/1055452/c-get-name-of-type-in-template
+/// But this is cleaner and works better in this case
+
+template <typename T,
+          enable_if_t<std::is_integral<T>::value && std::is_signed<T>::value, detail::enabler> = detail::dummy>
+constexpr const char *type_name() {
+    return "INT";
+}
+
+template <typename T,
+          enable_if_t<std::is_integral<T>::value && std::is_unsigned<T>::value, detail::enabler> = detail::dummy>
+constexpr const char *type_name() {
+    return "UINT";
+}
+
+template <typename T, enable_if_t<std::is_floating_point<T>::value, detail::enabler> = detail::dummy>
+constexpr const char *type_name() {
+    return "FLOAT";
+}
+
+/// This one should not be used, since vector types print the internal type
+template <typename T, enable_if_t<is_vector<T>::value, detail::enabler> = detail::dummy>
+constexpr const char *type_name() {
+    return "VECTOR";
+}
+
+template <typename T,
+          enable_if_t<!std::is_floating_point<T>::value && !std::is_integral<T>::value && !is_vector<T>::value,
+                      detail::enabler> = detail::dummy>
+constexpr const char *type_name() {
+    return "TEXT";
+}
+
+// Lexical cast
+
+/// Signed integers / enums
+template <typename T,
+          enable_if_t<(std::is_integral<T>::value && std::is_signed<T>::value), detail::enabler> = detail::dummy>
+bool lexical_cast(std::string input, T &output) {
+    try {
+        size_t n = 0;
+        long long output_ll = std::stoll(input, &n, 0);
+        output = static_cast<T>(output_ll);
+        return n == input.size() && static_cast<long long>(output) == output_ll;
+    } catch(const std::invalid_argument &) {
+        return false;
+    } catch(const std::out_of_range &) {
+        return false;
+    }
+}
+
+/// Unsigned integers
+template <typename T,
+          enable_if_t<std::is_integral<T>::value && std::is_unsigned<T>::value, detail::enabler> = detail::dummy>
+bool lexical_cast(std::string input, T &output) {
+    if(!input.empty() && input.front() == '-')
+        return false; // std::stoull happily converts negative values to junk without any errors.
+
+    try {
+        size_t n = 0;
+        unsigned long long output_ll = std::stoull(input, &n, 0);
+        output = static_cast<T>(output_ll);
+        return n == input.size() && static_cast<unsigned long long>(output) == output_ll;
+    } catch(const std::invalid_argument &) {
+        return false;
+    } catch(const std::out_of_range &) {
+        return false;
+    }
+}
+
+/// Floats
+template <typename T, enable_if_t<std::is_floating_point<T>::value, detail::enabler> = detail::dummy>
+bool lexical_cast(std::string input, T &output) {
+    try {
+        size_t n = 0;
+        output = static_cast<T>(std::stold(input, &n));
+        return n == input.size();
+    } catch(const std::invalid_argument &) {
+        return false;
+    } catch(const std::out_of_range &) {
+        return false;
+    }
+}
+
+/// String and similar
+template <typename T,
+          enable_if_t<!std::is_floating_point<T>::value && !std::is_integral<T>::value &&
+                          std::is_assignable<T &, std::string>::value,
+                      detail::enabler> = detail::dummy>
+bool lexical_cast(std::string input, T &output) {
+    output = input;
+    return true;
+}
+
+/// Non-string parsable
+template <typename T,
+          enable_if_t<!std::is_floating_point<T>::value && !std::is_integral<T>::value &&
+                          !std::is_assignable<T &, std::string>::value,
+                      detail::enabler> = detail::dummy>
+bool lexical_cast(std::string input, T &output) {
+    std::istringstream is;
+
+    is.str(input);
+    is >> output;
+    return !is.fail() && !is.rdbuf()->in_avail();
+}
+
+} // namespace detail
+} // namespace CLI
+
+// From CLI/Split.hpp:
+
+namespace CLI {
+namespace detail {
+
+// Returns false if not a short option. Otherwise, sets opt name and rest and returns true
+inline bool split_short(const std::string &current, std::string &name, std::string &rest) {
+    if(current.size() > 1 && current[0] == '-' && valid_first_char(current[1])) {
+        name = current.substr(1, 1);
+        rest = current.substr(2);
+        return true;
+    } else
+        return false;
+}
+
+// Returns false if not a long option. Otherwise, sets opt name and other side of = and returns true
+inline bool split_long(const std::string &current, std::string &name, std::string &value) {
+    if(current.size() > 2 && current.substr(0, 2) == "--" && valid_first_char(current[2])) {
+        auto loc = current.find("=");
+        if(loc != std::string::npos) {
+            name = current.substr(2, loc - 2);
+            value = current.substr(loc + 1);
+        } else {
+            name = current.substr(2);
+            value = "";
+        }
+        return true;
+    } else
+        return false;
+}
+
+// Splits a string into multiple long and short names
+inline std::vector<std::string> split_names(std::string current) {
+    std::vector<std::string> output;
+    size_t val;
+    while((val = current.find(",")) != std::string::npos) {
+        output.push_back(trim_copy(current.substr(0, val)));
+        current = current.substr(val + 1);
+    }
+    output.push_back(trim_copy(current));
+    return output;
+}
+
+/// Get a vector of short names, one of long names, and a single name
+inline std::tuple<std::vector<std::string>, std::vector<std::string>, std::string>
+get_names(const std::vector<std::string> &input) {
+
+    std::vector<std::string> short_names;
+    std::vector<std::string> long_names;
+    std::string pos_name;
+
+    for(std::string name : input) {
+        if(name.length() == 0)
+            continue;
+        else if(name.length() > 1 && name[0] == '-' && name[1] != '-') {
+            if(name.length() == 2 && valid_first_char(name[1]))
+                short_names.emplace_back(1, name[1]);
+            else
+                throw BadNameString::OneCharName(name);
+        } else if(name.length() > 2 && name.substr(0, 2) == "--") {
+            name = name.substr(2);
+            if(valid_name_string(name))
+                long_names.push_back(name);
+            else
+                throw BadNameString::BadLongName(name);
+        } else if(name == "-" || name == "--") {
+            throw BadNameString::DashesOnly(name);
+        } else {
+            if(pos_name.length() > 0)
+                throw BadNameString::MultiPositionalNames(name);
+            pos_name = name;
+        }
+    }
+
+    return std::tuple<std::vector<std::string>, std::vector<std::string>, std::string>(
+        short_names, long_names, pos_name);
+}
+
+} // namespace detail
+} // namespace CLI
+
+// From CLI/ConfigFwd.hpp:
+
+namespace CLI {
+
+class App;
+
+namespace detail {
+
+/// Comma separated join, adds quotes if needed
+inline std::string ini_join(std::vector<std::string> args) {
+    std::ostringstream s;
+    size_t start = 0;
+    for(const auto &arg : args) {
+        if(start++ > 0)
+            s << " ";
+
+        auto it = std::find_if(arg.begin(), arg.end(), [](char ch) { return std::isspace<char>(ch, std::locale()); });
+        if(it == arg.end())
+            s << arg;
+        else if(arg.find(R"(")") == std::string::npos)
+            s << R"(")" << arg << R"(")";
+        else
+            s << R"(')" << arg << R"(')";
+    }
+
+    return s.str();
+}
+
+} // namespace detail
+
+/// Holds values to load into Options
+struct ConfigItem {
+    /// This is the list of parents
+    std::vector<std::string> parents;
+
+    /// This is the name
+    std::string name;
+
+    /// Listing of inputs
+    std::vector<std::string> inputs;
+
+    /// The list of parents and name joined by "."
+    std::string fullname() const {
+        std::vector<std::string> tmp = parents;
+        tmp.emplace_back(name);
+        return detail::join(tmp, ".");
+    }
+};
+
+/// This class provides a converter for configuration files.
+class Config {
+  protected:
+    std::vector<ConfigItem> items;
+
+  public:
+    /// Convert an app into a configuration
+    virtual std::string to_config(const App *, bool, bool, std::string) const = 0;
+
+    /// Convert a configuration into an app
+    virtual std::vector<ConfigItem> from_config(std::istream &) const = 0;
+
+    /// Convert a flag to a bool
+    virtual std::vector<std::string> to_flag(const ConfigItem &item) const {
+        if(item.inputs.size() == 1) {
+            std::string val = item.inputs.at(0);
+            val = detail::to_lower(val);
+
+            if(val == "true" || val == "on" || val == "yes") {
+                return std::vector<std::string>(1);
+            } else if(val == "false" || val == "off" || val == "no") {
+                return std::vector<std::string>();
+            } else {
+                try {
+                    size_t ui = std::stoul(val);
+                    return std::vector<std::string>(ui);
+                } catch(const std::invalid_argument &) {
+                    throw ConversionError::TrueFalse(item.fullname());
+                }
+            }
+        } else {
+            throw ConversionError::TooManyInputsFlag(item.fullname());
+        }
+    }
+
+    /// Parse a config file, throw an error (ParseError:ConfigParseError or FileError) on failure
+    std::vector<ConfigItem> from_file(const std::string &name) {
+        std::ifstream input{name};
+        if(!input.good())
+            throw FileError::Missing(name);
+
+        return from_config(input);
+    }
+
+    /// virtual destructor
+    virtual ~Config() = default;
+};
+
+/// This converter works with INI files
+class ConfigINI : public Config {
+  public:
+    std::string to_config(const App *, bool default_also, bool write_description, std::string prefix) const override;
+
+    std::vector<ConfigItem> from_config(std::istream &input) const override {
+        std::string line;
+        std::string section = "default";
+
+        std::vector<ConfigItem> output;
+
+        while(getline(input, line)) {
+            std::vector<std::string> items_buffer;
+
+            detail::trim(line);
+            size_t len = line.length();
+            if(len > 1 && line[0] == '[' && line[len - 1] == ']') {
+                section = line.substr(1, len - 2);
+            } else if(len > 0 && line[0] != ';') {
+                output.emplace_back();
+                ConfigItem &out = output.back();
+
+                // Find = in string, split and recombine
+                auto pos = line.find('=');
+                if(pos != std::string::npos) {
+                    out.name = detail::trim_copy(line.substr(0, pos));
+                    std::string item = detail::trim_copy(line.substr(pos + 1));
+                    items_buffer = detail::split_up(item);
+                } else {
+                    out.name = detail::trim_copy(line);
+                    items_buffer = {"ON"};
+                }
+
+                if(detail::to_lower(section) != "default") {
+                    out.parents = {section};
+                }
+
+                if(out.name.find('.') != std::string::npos) {
+                    std::vector<std::string> plist = detail::split(out.name, '.');
+                    out.name = plist.back();
+                    plist.pop_back();
+                    out.parents.insert(out.parents.end(), plist.begin(), plist.end());
+                }
+
+                out.inputs.insert(std::end(out.inputs), std::begin(items_buffer), std::end(items_buffer));
+            }
+        }
+        return output;
+    }
+};
+
+} // namespace CLI
+
+// From CLI/Validators.hpp:
+
+namespace CLI {
+
+/// @defgroup validator_group Validators
+
+/// @brief Some validators that are provided
+///
+/// These are simple `std::string(const std::string&)` validators that are useful. They return
+/// a string if the validation fails. A custom struct is provided, as well, with the same user
+/// semantics, but with the ability to provide a new type name.
+/// @{
+
+///
+struct Validator {
+    /// This is the type name, if empty the type name will not be changed
+    std::string tname;
+
+    /// This it the base function that is to be called.
+    /// Returns a string error message if validation fails.
+    std::function<std::string(const std::string &)> func;
+
+    /// This is the required operator for a validator - provided to help
+    /// users (CLI11 uses the member `func` directly)
+    std::string operator()(const std::string &str) const { return func(str); };
+
+    /// Combining validators is a new validator
+    Validator operator&(const Validator &other) const {
+        Validator newval;
+        newval.tname = (tname == other.tname ? tname : "");
+
+        // Give references (will make a copy in lambda function)
+        const std::function<std::string(const std::string &filename)> &f1 = func;
+        const std::function<std::string(const std::string &filename)> &f2 = other.func;
+
+        newval.func = [f1, f2](const std::string &filename) {
+            std::string s1 = f1(filename);
+            std::string s2 = f2(filename);
+            if(!s1.empty() && !s2.empty())
+                return s1 + " & " + s2;
+            else
+                return s1 + s2;
+        };
+        return newval;
+    }
+
+    /// Combining validators is a new validator
+    Validator operator|(const Validator &other) const {
+        Validator newval;
+        newval.tname = (tname == other.tname ? tname : "");
+
+        // Give references (will make a copy in lambda function)
+        const std::function<std::string(const std::string &filename)> &f1 = func;
+        const std::function<std::string(const std::string &filename)> &f2 = other.func;
+
+        newval.func = [f1, f2](const std::string &filename) {
+            std::string s1 = f1(filename);
+            std::string s2 = f2(filename);
+            if(s1.empty() || s2.empty())
+                return std::string();
+            else
+                return s1 + " & " + s2;
+        };
+        return newval;
+    }
+};
+
+// The implementation of the built in validators is using the Validator class;
+// the user is only expected to use the const (static) versions (since there's no setup).
+// Therefore, this is in detail.
+namespace detail {
+
+/// Check for an existing file (returns error message if check fails)
+struct ExistingFileValidator : public Validator {
+    ExistingFileValidator() {
+        tname = "FILE";
+        func = [](const std::string &filename) {
+            struct stat buffer;
+            bool exist = stat(filename.c_str(), &buffer) == 0;
+            bool is_dir = (buffer.st_mode & S_IFDIR) != 0;
+            if(!exist) {
+                return "File does not exist: " + filename;
+            } else if(is_dir) {
+                return "File is actually a directory: " + filename;
+            }
+            return std::string();
+        };
+    }
+};
+
+/// Check for an existing directory (returns error message if check fails)
+struct ExistingDirectoryValidator : public Validator {
+    ExistingDirectoryValidator() {
+        tname = "DIR";
+        func = [](const std::string &filename) {
+            struct stat buffer;
+            bool exist = stat(filename.c_str(), &buffer) == 0;
+            bool is_dir = (buffer.st_mode & S_IFDIR) != 0;
+            if(!exist) {
+                return "Directory does not exist: " + filename;
+            } else if(!is_dir) {
+                return "Directory is actually a file: " + filename;
+            }
+            return std::string();
+        };
+    }
+};
+
+/// Check for an existing path
+struct ExistingPathValidator : public Validator {
+    ExistingPathValidator() {
+        tname = "PATH";
+        func = [](const std::string &filename) {
+            struct stat buffer;
+            bool const exist = stat(filename.c_str(), &buffer) == 0;
+            if(!exist) {
+                return "Path does not exist: " + filename;
+            }
+            return std::string();
+        };
+    }
+};
+
+/// Check for an non-existing path
+struct NonexistentPathValidator : public Validator {
+    NonexistentPathValidator() {
+        tname = "PATH";
+        func = [](const std::string &filename) {
+            struct stat buffer;
+            bool exist = stat(filename.c_str(), &buffer) == 0;
+            if(exist) {
+                return "Path already exists: " + filename;
+            }
+            return std::string();
+        };
+    }
+};
+} // namespace detail
+
+// Static is not needed here, because global const implies static.
+
+/// Check for existing file (returns error message if check fails)
+const detail::ExistingFileValidator ExistingFile;
+
+/// Check for an existing directory (returns error message if check fails)
+const detail::ExistingDirectoryValidator ExistingDirectory;
+
+/// Check for an existing path
+const detail::ExistingPathValidator ExistingPath;
+
+/// Check for an non-existing path
+const detail::NonexistentPathValidator NonexistentPath;
+
+///  Produce a range (factory). Min and max are inclusive.
+struct Range : public Validator {
+    /// This produces a range with min and max inclusive.
+    ///
+    /// Note that the constructor is templated, but the struct is not, so C++17 is not
+    /// needed to provide nice syntax for Range(a,b).
+    template <typename T> Range(T min, T max) {
+        std::stringstream out;
+        out << detail::type_name<T>() << " in [" << min << " - " << max << "]";
+
+        tname = out.str();
+        func = [min, max](std::string input) {
+            T val;
+            detail::lexical_cast(input, val);
+            if(val < min || val > max)
+                return "Value " + input + " not in range " + std::to_string(min) + " to " + std::to_string(max);
+
+            return std::string();
+        };
+    }
+
+    /// Range of one value is 0 to value
+    template <typename T> explicit Range(T max) : Range(static_cast<T>(0), max) {}
+};
+
+/// @}
+
+} // namespace CLI
+
+// From CLI/FormatterFwd.hpp:
+
+namespace CLI {
+
+class Option;
+class App;
+
+/// This enum signifies the type of help requested
+///
+/// This is passed in by App; all user classes must accept this as
+/// the second argument.
+
+enum class AppFormatMode {
+    Normal, //< The normal, detailed help
+    All,    //< A fully expanded help
+    Sub,    //< Used when printed as part of expanded subcommand
+};
+
+/// This is the minimum requirements to run a formatter.
+///
+/// A user can subclass this is if they do not care at all
+/// about the structure in CLI::Formatter.
+class FormatterBase {
+  protected:
+    /// @name Options
+    ///@{
+
+    /// The width of the first column
+    size_t column_width_{30};
+
+    /// @brief The required help printout labels (user changeable)
+    /// Values are Needs, Excludes, etc.
+    std::map<std::string, std::string> labels_;
+
+    ///@}
+    /// @name Basic
+    ///@{
+
+  public:
+    FormatterBase() = default;
+    FormatterBase(const FormatterBase &) = default;
+    FormatterBase(FormatterBase &&) = default;
+    virtual ~FormatterBase() = default;
+
+    /// This is the key method that puts together help
+    virtual std::string make_help(const App *, std::string, AppFormatMode) const = 0;
+
+    ///@}
+    /// @name Setters
+    ///@{
+
+    /// Set the "REQUIRED" label
+    void label(std::string key, std::string val) { labels_[key] = val; }
+
+    /// Set the column width
+    void column_width(size_t val) { column_width_ = val; }
+
+    ///@}
+    /// @name Getters
+    ///@{
+
+    /// Get the current value of a name (REQUIRED, etc.)
+    std::string get_label(std::string key) const {
+        if(labels_.find(key) == labels_.end())
+            return key;
+        else
+            return labels_.at(key);
+    }
+
+    /// Get the current column width
+    size_t get_column_width() const { return column_width_; }
+
+    ///@}
+};
+
+/// This is a specialty override for lambda functions
+class FormatterLambda final : public FormatterBase {
+    using funct_t = std::function<std::string(const App *, std::string, AppFormatMode)>;
+
+    /// The lambda to hold and run
+    funct_t lambda_;
+
+  public:
+    /// Create a FormatterLambda with a lambda function
+    explicit FormatterLambda(funct_t funct) : lambda_(std::move(funct)) {}
+
+    /// This will simply call the lambda function
+    std::string make_help(const App *app, std::string name, AppFormatMode mode) const override {
+        return lambda_(app, name, mode);
+    }
+};
+
+/// This is the default Formatter for CLI11. It pretty prints help output, and is broken into quite a few
+/// overridable methods, to be highly customizable with minimal effort.
+class Formatter : public FormatterBase {
+  public:
+    Formatter() = default;
+    Formatter(const Formatter &) = default;
+    Formatter(Formatter &&) = default;
+
+    /// @name Overridables
+    ///@{
+
+    /// This prints out a group of options with title
+    ///
+    virtual std::string make_group(std::string group, bool is_positional, std::vector<const Option *> opts) const;
+
+    /// This prints out just the positionals "group"
+    virtual std::string make_positionals(const App *app) const;
+
+    /// This prints out all the groups of options
+    std::string make_groups(const App *app, AppFormatMode mode) const;
+
+    /// This prints out all the subcommands
+    virtual std::string make_subcommands(const App *app, AppFormatMode mode) const;
+
+    /// This prints out a subcommand
+    virtual std::string make_subcommand(const App *sub) const;
+
+    /// This prints out a subcommand in help-all
+    virtual std::string make_expanded(const App *sub) const;
+
+    /// This prints out all the groups of options
+    virtual std::string make_footer(const App *app) const;
+
+    /// This displays the description line
+    virtual std::string make_description(const App *app) const;
+
+    /// This displays the usage line
+    virtual std::string make_usage(const App *app, std::string name) const;
+
+    /// This puts everything together
+    std::string make_help(const App *, std::string, AppFormatMode) const override;
+
+    ///@}
+    /// @name Options
+    ///@{
+
+    /// This prints out an option help line, either positional or optional form
+    virtual std::string make_option(const Option *opt, bool is_positional) const {
+        std::stringstream out;
+        detail::format_help(
+            out, make_option_name(opt, is_positional) + make_option_opts(opt), make_option_desc(opt), column_width_);
+        return out.str();
+    }
+
+    /// @brief This is the name part of an option, Default: left column
+    virtual std::string make_option_name(const Option *, bool) const;
+
+    /// @brief This is the options part of the name, Default: combined into left column
+    virtual std::string make_option_opts(const Option *) const;
+
+    /// @brief This is the description. Default: Right column, on new line if left column too large
+    virtual std::string make_option_desc(const Option *) const;
+
+    /// @brief This is used to print the name on the USAGE line
+    virtual std::string make_option_usage(const Option *opt) const;
+
+    ///@}
+};
+
+} // namespace CLI
+
+// From CLI/Option.hpp:
+
+namespace CLI {
+
+using results_t = std::vector<std::string>;
+using callback_t = std::function<bool(results_t)>;
+
+class Option;
+class App;
+
+using Option_p = std::unique_ptr<Option>;
+
+enum class MultiOptionPolicy { Throw, TakeLast, TakeFirst, Join };
+
+/// This is the CRTP base class for Option and OptionDefaults. It was designed this way
+/// to share parts of the class; an OptionDefaults can copy to an Option.
+template <typename CRTP> class OptionBase {
+    friend App;
+
+  protected:
+    /// The group membership
+    std::string group_ = std::string("Options");
+
+    /// True if this is a required option
+    bool required_{false};
+
+    /// Ignore the case when matching (option, not value)
+    bool ignore_case_{false};
+
+    /// Allow this option to be given in a configuration file
+    bool configurable_{true};
+
+    /// Policy for multiple arguments when `expected_ == 1`  (can be set on bool flags, too)
+    MultiOptionPolicy multi_option_policy_{MultiOptionPolicy::Throw};
+
+    /// Copy the contents to another similar class (one based on OptionBase)
+    template <typename T> void copy_to(T *other) const {
+        other->group(group_);
+        other->required(required_);
+        other->ignore_case(ignore_case_);
+        other->configurable(configurable_);
+        other->multi_option_policy(multi_option_policy_);
+    }
+
+  public:
+    // setters
+
+    /// Changes the group membership
+    CRTP *group(std::string name) {
+        group_ = name;
+        return static_cast<CRTP *>(this);
+        ;
+    }
+
+    /// Set the option as required
+    CRTP *required(bool value = true) {
+        required_ = value;
+        return static_cast<CRTP *>(this);
+    }
+
+    /// Support Plumbum term
+    CRTP *mandatory(bool value = true) { return required(value); }
+
+    // Getters
+
+    /// Get the group of this option
+    const std::string &get_group() const { return group_; }
+
+    /// True if this is a required option
+    bool get_required() const { return required_; }
+
+    /// The status of ignore case
+    bool get_ignore_case() const { return ignore_case_; }
+
+    /// The status of configurable
+    bool get_configurable() const { return configurable_; }
+
+    /// The status of the multi option policy
+    MultiOptionPolicy get_multi_option_policy() const { return multi_option_policy_; }
+
+    // Shortcuts for multi option policy
+
+    /// Set the multi option policy to take last
+    CRTP *take_last() {
+        auto self = static_cast<CRTP *>(this);
+        self->multi_option_policy(MultiOptionPolicy::TakeLast);
+        return self;
+    }
+
+    /// Set the multi option policy to take last
+    CRTP *take_first() {
+        auto self = static_cast<CRTP *>(this);
+        self->multi_option_policy(MultiOptionPolicy::TakeFirst);
+        return self;
+    }
+
+    /// Set the multi option policy to take last
+    CRTP *join() {
+        auto self = static_cast<CRTP *>(this);
+        self->multi_option_policy(MultiOptionPolicy::Join);
+        return self;
+    }
+
+    /// Allow in a configuration file
+    CRTP *configurable(bool value = true) {
+        configurable_ = value;
+        return static_cast<CRTP *>(this);
+    }
+};
+
+/// This is a version of OptionBase that only supports setting values,
+/// for defaults. It is stored as the default option in an App.
+class OptionDefaults : public OptionBase<OptionDefaults> {
+  public:
+    OptionDefaults() = default;
+
+    // Methods here need a different implementation if they are Option vs. OptionDefault
+
+    /// Take the last argument if given multiple times
+    OptionDefaults *multi_option_policy(MultiOptionPolicy value = MultiOptionPolicy::Throw) {
+        multi_option_policy_ = value;
+        return this;
+    }
+
+    /// Ignore the case of the option name
+    OptionDefaults *ignore_case(bool value = true) {
+        ignore_case_ = value;
+        return this;
+    }
+};
+
+class Option : public OptionBase<Option> {
+    friend App;
+
+  protected:
+    /// @name Names
+    ///@{
+
+    /// A list of the short names (`-a`) without the leading dashes
+    std::vector<std::string> snames_;
+
+    /// A list of the long names (`--a`) without the leading dashes
+    std::vector<std::string> lnames_;
+
+    /// A positional name
+    std::string pname_;
+
+    /// If given, check the environment for this option
+    std::string envname_;
+
+    ///@}
+    /// @name Help
+    ///@{
+
+    /// The description for help strings
+    std::string description_;
+
+    /// A human readable default value, usually only set if default is true in creation
+    std::string defaultval_;
+
+    /// A human readable type value, set when App creates this
+    ///
+    /// This is a lambda function so "types" can be dynamic, such as when a set prints its contents.
+    std::function<std::string()> type_name_{[]() { return std::string(); }};
+
+    /// True if this option has a default
+    bool default_{false};
+
+    ///@}
+    /// @name Configuration
+    ///@{
+
+    /// The number of arguments that make up one option. -1=unlimited (vector-like), 0=flag, 1=normal option,
+    /// 2=complex/pair, etc. Set only when the option is created; this is intrinsic to the type. Eventually, -2 may mean
+    /// vector of pairs.
+    int type_size_{1};
+
+    /// The number of expected values, type_size_ must be < 0. Ignored for flag. N < 0 means at least -N values.
+    int expected_{1};
+
+    /// A list of validators to run on each value parsed
+    std::vector<std::function<std::string(std::string &)>> validators_;
+
+    /// A list of options that are required with this option
+    std::set<Option *> needs_;
+
+    /// A list of options that are excluded with this option
+    std::set<Option *> excludes_;
+
+    ///@}
+    /// @name Other
+    ///@{
+
+    /// Remember the parent app
+    App *parent_;
+
+    /// Options store a callback to do all the work
+    callback_t callback_;
+
+    /// Options can short-circuit for help options or similar (called before parsing is validated)
+    bool short_circuit_{false};
+
+    ///@}
+    /// @name Parsing results
+    ///@{
+
+    /// Results of parsing
+    results_t results_;
+
+    /// Whether the callback has run (needed for INI parsing)
+    bool callback_run_{false};
+
+    ///@}
+
+    /// Making an option by hand is not defined, it must be made by the App class
+    Option(
+        std::string name, std::string description, std::function<bool(results_t)> callback, bool defaulted, App *parent)
+        : description_(std::move(description)), default_(defaulted), parent_(parent),
+          callback_(callback ? std::move(callback) : [](results_t) { return true; }) {
+        std::tie(snames_, lnames_, pname_) = detail::get_names(detail::split_names(name));
+    }
+
+  public:
+    /// @name Basic
+    ///@{
+
+    /// Count the total number of times an option was passed
+    size_t count() const { return results_.size(); }
+
+    /// True if the option was not passed
+    size_t empty() const { return results_.empty(); }
+
+    /// This class is true if option is passed.
+    operator bool() const { return !empty(); }
+
+    /// Clear the parsed results (mostly for testing)
+    void clear() { results_.clear(); }
+
+    ///@}
+    /// @name Setting options
+    ///@{
+
+    /// Set the number of expected arguments (Flags don't use this)
+    Option *expected(int value) {
+        // Break if this is a flag
+        if(type_size_ == 0)
+            throw IncorrectConstruction::SetFlag(get_name(true, true));
+
+        // Setting 0 is not allowed
+        else if(value == 0)
+            throw IncorrectConstruction::Set0Opt(get_name());
+
+        // No change is okay, quit now
+        else if(expected_ == value)
+            return this;
+
+        // Type must be a vector
+        else if(type_size_ >= 0)
+            throw IncorrectConstruction::ChangeNotVector(get_name());
+
+        // TODO: Can support multioption for non-1 values (except for join)
+        else if(value != 1 && multi_option_policy_ != MultiOptionPolicy::Throw)
+            throw IncorrectConstruction::AfterMultiOpt(get_name());
+
+        expected_ = value;
+        return this;
+    }
+
+    /// Adds a validator with a built in type name
+    Option *check(const Validator &validator) {
+        validators_.emplace_back(validator.func);
+        if(!validator.tname.empty())
+            type_name(validator.tname);
+        return this;
+    }
+
+    /// Adds a validator. Takes a const string& and returns an error message (empty if conversion/check is okay).
+    Option *check(std::function<std::string(const std::string &)> validator) {
+        validators_.emplace_back(validator);
+        return this;
+    }
+
+    /// Adds a validator-like function that can change result
+    Option *transform(std::function<std::string(std::string)> func) {
+        validators_.emplace_back([func](std::string &inout) {
+            try {
+                inout = func(inout);
+            } catch(const ValidationError &e) {
+                return std::string(e.what());
+            }
+            return std::string();
+        });
+        return this;
+    }
+
+    /// Adds a user supplied function to run on each item passed in (communicate though lambda capture)
+    Option *each(std::function<void(std::string)> func) {
+        validators_.emplace_back([func](std::string &inout) {
+            func(inout);
+            return std::string();
+        });
+        return this;
+    }
+
+    /// Sets required options
+    Option *needs(Option *opt) {
+        auto tup = needs_.insert(opt);
+        if(!tup.second)
+            throw OptionAlreadyAdded::Requires(get_name(), opt->get_name());
+        return this;
+    }
+
+    /// Can find a string if needed
+    template <typename T = App> Option *needs(std::string opt_name) {
+        for(const Option_p &opt : dynamic_cast<T *>(parent_)->options_)
+            if(opt.get() != this && opt->check_name(opt_name))
+                return needs(opt.get());
+        throw IncorrectConstruction::MissingOption(opt_name);
+    }
+
+    /// Any number supported, any mix of string and Opt
+    template <typename A, typename B, typename... ARG> Option *needs(A opt, B opt1, ARG... args) {
+        needs(opt);
+        return needs(opt1, args...);
+    }
+
+    /// Remove needs link from an option. Returns true if the option really was in the needs list.
+    bool remove_needs(Option *opt) {
+        auto iterator = std::find(std::begin(needs_), std::end(needs_), opt);
+
+        if(iterator != std::end(needs_)) {
+            needs_.erase(iterator);
+            return true;
+        } else {
+            return false;
+        }
+    }
+
+    /// Sets excluded options
+    Option *excludes(Option *opt) {
+        excludes_.insert(opt);
+
+        // Help text should be symmetric - excluding a should exclude b
+        opt->excludes_.insert(this);
+
+        // Ignoring the insert return value, excluding twice is now allowed.
+        // (Mostly to allow both directions to be excluded by user, even though the library does it for you.)
+
+        return this;
+    }
+
+    /// Can find a string if needed
+    template <typename T = App> Option *excludes(std::string opt_name) {
+        for(const Option_p &opt : dynamic_cast<T *>(parent_)->options_)
+            if(opt.get() != this && opt->check_name(opt_name))
+                return excludes(opt.get());
+        throw IncorrectConstruction::MissingOption(opt_name);
+    }
+
+    /// Any number supported, any mix of string and Opt
+    template <typename A, typename B, typename... ARG> Option *excludes(A opt, B opt1, ARG... args) {
+        excludes(opt);
+        return excludes(opt1, args...);
+    }
+
+    /// Remove needs link from an option. Returns true if the option really was in the needs list.
+    bool remove_excludes(Option *opt) {
+        auto iterator = std::find(std::begin(excludes_), std::end(excludes_), opt);
+
+        if(iterator != std::end(excludes_)) {
+            excludes_.erase(iterator);
+            return true;
+        } else {
+            return false;
+        }
+    }
+
+    /// Sets environment variable to read if no option given
+    Option *envname(std::string name) {
+        envname_ = name;
+        return this;
+    }
+
+    /// Ignore case
+    ///
+    /// The template hides the fact that we don't have the definition of App yet.
+    /// You are never expected to add an argument to the template here.
+    template <typename T = App> Option *ignore_case(bool value = true) {
+        ignore_case_ = value;
+        auto *parent = dynamic_cast<T *>(parent_);
+
+        for(const Option_p &opt : parent->options_)
+            if(opt.get() != this && *opt == *this)
+                throw OptionAlreadyAdded(opt->get_name(true, true));
+
+        return this;
+    }
+
+    /// Take the last argument if given multiple times (or another policy)
+    Option *multi_option_policy(MultiOptionPolicy value = MultiOptionPolicy::Throw) {
+
+        if(get_items_expected() < 0)
+            throw IncorrectConstruction::MultiOptionPolicy(get_name());
+        multi_option_policy_ = value;
+        return this;
+    }
+
+    /// Options with a short circuit set will run this function before parsing is finished.
+    ///
+    /// This is set on help functions, for example, to escape the normal validation.
+    Option *short_circuit(bool value = true) {
+        short_circuit_ = value;
+        return this;
+    }
+
+    ///@}
+    /// @name Accessors
+    ///@{
+
+    /// The number of arguments the option expects
+    int get_type_size() const { return type_size_; }
+
+    /// The environment variable associated to this value
+    std::string get_envname() const { return envname_; }
+
+    /// The set of options needed
+    std::set<Option *> get_needs() const { return needs_; }
+
+    /// The set of options excluded
+    std::set<Option *> get_excludes() const { return excludes_; }
+
+    /// The default value (for help printing)
+    std::string get_defaultval() const { return defaultval_; }
+
+    /// See if this is supposed to short circuit (skip validation, INI, etc) (Used for help flags)
+    bool get_short_circuit() const { return short_circuit_; }
+
+    /// Get the callback function
+    callback_t get_callback() const { return callback_; }
+
+    /// Get the long names
+    const std::vector<std::string> get_lnames() const { return lnames_; }
+
+    /// Get the short names
+    const std::vector<std::string> get_snames() const { return snames_; }
+
+    /// The number of times the option expects to be included
+    int get_expected() const { return expected_; }
+
+    /// \brief The total number of expected values (including the type)
+    /// This is positive if exactly this number is expected, and negitive for at least N values
+    ///
+    /// v = fabs(size_type*expected)
+    /// !MultiOptionPolicy::Throw
+    ///           | Expected < 0  | Expected == 0 | Expected > 0
+    /// Size < 0  |      -v       |       0       |     -v
+    /// Size == 0 |       0       |       0       |      0
+    /// Size > 0  |      -v       |       0       |     -v       // Expected must be 1
+    ///
+    /// MultiOptionPolicy::Throw
+    ///           | Expected < 0  | Expected == 0 | Expected > 0
+    /// Size < 0  |      -v       |       0       |      v
+    /// Size == 0 |       0       |       0       |      0
+    /// Size > 0  |       v       |       0       |      v      // Expected must be 1
+    ///
+    int get_items_expected() const {
+        return std::abs(type_size_ * expected_) *
+               ((multi_option_policy_ != MultiOptionPolicy::Throw || (expected_ < 0 && type_size_ < 0) ? -1 : 1));
+    }
+
+    /// True if this has a default value
+    int get_default() const { return default_; }
+
+    /// True if the argument can be given directly
+    bool get_positional() const { return pname_.length() > 0; }
+
+    /// True if option has at least one non-positional name
+    bool nonpositional() const { return (snames_.size() + lnames_.size()) > 0; }
+
+    /// True if option has description
+    bool has_description() const { return description_.length() > 0; }
+
+    /// Get the description
+    const std::string &get_description() const { return description_; }
+
+    ///@}
+    /// @name Help tools
+    ///@{
+
+    /// \brief Gets a comma seperated list of names.
+    /// Will include / prefer the positional name if positional is true.
+    /// If all_options is false, pick just the most descriptive name to show.
+    /// Use `get_name(true)` to get the positional name (replaces `get_pname`)
+    std::string get_name(bool positional = false, //<[input] Show the positional name
+                         bool all_options = false //<[input] Show every option
+                         ) const {
+
+        if(all_options) {
+
+            std::vector<std::string> name_list;
+
+            /// The all list wil never include a positional unless asked or that's the only name.
+            if((positional && pname_.length()) || (snames_.empty() && lnames_.empty()))
+                name_list.push_back(pname_);
+
+            for(const std::string &sname : snames_)
+                name_list.push_back("-" + sname);
+
+            for(const std::string &lname : lnames_)
+                name_list.push_back("--" + lname);
+
+            return detail::join(name_list);
+
+        } else {
+
+            // This returns the positional name no matter what
+            if(positional)
+                return pname_;
+
+            // Prefer long name
+            else if(!lnames_.empty())
+                return std::string("--") + lnames_[0];
+
+            // Or short name if no long name
+            else if(!snames_.empty())
+                return std::string("-") + snames_[0];
+
+            // If positional is the only name, it's okay to use that
+            else
+                return pname_;
+        }
+    }
+
+    ///@}
+    /// @name Parser tools
+    ///@{
+
+    /// Process the callback
+    void run_callback() {
+
+        // Run the validators (can change the string)
+        if(!validators_.empty()) {
+            for(std::string &result : results_)
+                for(const std::function<std::string(std::string &)> &vali : validators_) {
+                    std::string err_msg = vali(result);
+                    if(!err_msg.empty())
+                        throw ValidationError(get_name(), err_msg);
+                }
+        }
+
+        bool local_result;
+
+        // Num items expected or length of vector, always at least 1
+        // Only valid for a trimming policy
+        int trim_size =
+            std::min<int>(std::max<int>(std::abs(get_items_expected()), 1), static_cast<int>(results_.size()));
+
+        // Operation depends on the policy setting
+        if(multi_option_policy_ == MultiOptionPolicy::TakeLast) {
+            // Allow multi-option sizes (including 0)
+            results_t partial_result{results_.end() - trim_size, results_.end()};
+            local_result = !callback_(partial_result);
+
+        } else if(multi_option_policy_ == MultiOptionPolicy::TakeFirst) {
+            results_t partial_result{results_.begin(), results_.begin() + trim_size};
+            local_result = !callback_(partial_result);
+
+        } else if(multi_option_policy_ == MultiOptionPolicy::Join) {
+            results_t partial_result = {detail::join(results_, "\n")};
+            local_result = !callback_(partial_result);
+
+        } else {
+            // Exact number required
+            if(get_items_expected() > 0) {
+                if(results_.size() != static_cast<size_t>(get_items_expected()))
+                    throw ArgumentMismatch(get_name(), get_items_expected(), results_.size());
+                // Variable length list
+            } else if(get_items_expected() < 0) {
+                // Require that this be a multiple of expected size and at least as many as expected
+                if(results_.size() < static_cast<size_t>(-get_items_expected()) ||
+                   results_.size() % static_cast<size_t>(std::abs(get_type_size())) != 0)
+                    throw ArgumentMismatch(get_name(), get_items_expected(), results_.size());
+            }
+            local_result = !callback_(results_);
+        }
+
+        if(local_result)
+            throw ConversionError(get_name(), results_);
+    }
+
+    /// If options share any of the same names, they are equal (not counting positional)
+    bool operator==(const Option &other) const {
+        for(const std::string &sname : snames_)
+            if(other.check_sname(sname))
+                return true;
+        for(const std::string &lname : lnames_)
+            if(other.check_lname(lname))
+                return true;
+        // We need to do the inverse, just in case we are ignore_case
+        for(const std::string &sname : other.snames_)
+            if(check_sname(sname))
+                return true;
+        for(const std::string &lname : other.lnames_)
+            if(check_lname(lname))
+                return true;
+        return false;
+    }
+
+    /// Check a name. Requires "-" or "--" for short / long, supports positional name
+    bool check_name(std::string name) const {
+
+        if(name.length() > 2 && name.substr(0, 2) == "--")
+            return check_lname(name.substr(2));
+        else if(name.length() > 1 && name.substr(0, 1) == "-")
+            return check_sname(name.substr(1));
+        else {
+            std::string local_pname = pname_;
+            if(ignore_case_) {
+                local_pname = detail::to_lower(local_pname);
+                name = detail::to_lower(name);
+            }
+            return name == local_pname;
+        }
+    }
+
+    /// Requires "-" to be removed from string
+    bool check_sname(std::string name) const {
+        if(ignore_case_) {
+            name = detail::to_lower(name);
+            return std::find_if(std::begin(snames_), std::end(snames_), [&name](std::string local_sname) {
+                       return detail::to_lower(local_sname) == name;
+                   }) != std::end(snames_);
+        } else
+            return std::find(std::begin(snames_), std::end(snames_), name) != std::end(snames_);
+    }
+
+    /// Requires "--" to be removed from string
+    bool check_lname(std::string name) const {
+        if(ignore_case_) {
+            name = detail::to_lower(name);
+            return std::find_if(std::begin(lnames_), std::end(lnames_), [&name](std::string local_sname) {
+                       return detail::to_lower(local_sname) == name;
+                   }) != std::end(lnames_);
+        } else
+            return std::find(std::begin(lnames_), std::end(lnames_), name) != std::end(lnames_);
+    }
+
+    /// Puts a result at the end
+    Option *add_result(std::string s) {
+        results_.push_back(s);
+        callback_run_ = false;
+        return this;
+    }
+
+    /// Set the results vector all at once
+    Option *set_results(std::vector<std::string> results) {
+        results_ = results;
+        callback_run_ = false;
+        return this;
+    }
+
+    /// Get a copy of the results
+    std::vector<std::string> results() const { return results_; }
+
+    /// See if the callback has been run already
+    bool get_callback_run() const { return callback_run_; }
+
+    ///@}
+    /// @name Custom options
+    ///@{
+
+    /// Set the type function to run when displayed on this option
+    Option *type_name_fn(std::function<std::string()> typefun) {
+        type_name_ = typefun;
+        return this;
+    }
+
+    /// Set a custom option typestring
+    Option *type_name(std::string typeval) {
+        type_name_fn([typeval]() { return typeval; });
+        return this;
+    }
+
+    /// Provided for backward compatibility \deprecated
+    CLI11_DEPRECATED("Please use type_name instead")
+    Option *set_type_name(std::string typeval) { return type_name(typeval); }
+
+    /// Set a custom option size
+    Option *type_size(int type_size) {
+        type_size_ = type_size;
+        if(type_size_ == 0)
+            required_ = false;
+        if(type_size < 0)
+            expected_ = -1;
+        return this;
+    }
+
+    /// Set the default value string representation
+    Option *default_str(std::string val) {
+        defaultval_ = val;
+        return this;
+    }
+
+    /// Set the default value string representation and evaluate
+    Option *default_val(std::string val) {
+        default_str(val);
+        auto old_results = results_;
+        results_ = {val};
+        run_callback();
+        results_ = std::move(old_results);
+        return this;
+    }
+
+    /// Get the typename for this option
+    std::string get_type_name() const { return type_name_(); }
+};
+
+} // namespace CLI
+
+// From CLI/App.hpp:
+
+namespace CLI {
+
+#ifndef CLI11_PARSE
+#define CLI11_PARSE(app, argc, argv)                                                                                   \
+    try {                                                                                                              \
+        (app).parse((argc), (argv));                                                                                   \
+    } catch(const CLI::ParseError &e) {                                                                                \
+        return (app).exit(e);                                                                                          \
+    }
+#endif
+
+namespace detail {
+enum class Classifer { NONE, POSITIONAL_MARK, SHORT, LONG, SUBCOMMAND };
+struct AppFriend;
+} // namespace detail
+
+namespace FailureMessage {
+std::string simple(const App *app, const Error &e);
+std::string help(const App *app, const Error &e);
+} // namespace FailureMessage
+
+class App;
+
+using App_p = std::unique_ptr<App>;
+
+/// Creates a command line program, with very few defaults.
+/** To use, create a new `Program()` instance with `argc`, `argv`, and a help description. The templated
+ *  add_option methods make it easy to prepare options. Remember to call `.start` before starting your
+ * program, so that the options can be evaluated and the help option doesn't accidentally run your program. */
+class App {
+    friend Option;
+    friend detail::AppFriend;
+
+  protected:
+    // This library follows the Google style guide for member names ending in underscores
+
+    /// @name Basics
+    ///@{
+
+    /// Subcommand name or program name (from parser if name is empty)
+    std::string name_;
+
+    /// Description of the current program/subcommand
+    std::string description_;
+
+    /// If true, allow extra arguments (ie, don't throw an error). INHERITABLE
+    bool allow_extras_{false};
+
+    /// If true, allow extra arguments in the ini file (ie, don't throw an error). INHERITABLE
+    bool allow_config_extras_{false};
+
+    ///  If true, return immediately on an unrecognised option (implies allow_extras) INHERITABLE
+    bool prefix_command_{false};
+
+    /// This is a function that runs when complete. Great for subcommands. Can throw.
+    std::function<void()> callback_;
+
+    ///@}
+    /// @name Options
+    ///@{
+
+    /// The default values for options, customizable and changeable INHERITABLE
+    OptionDefaults option_defaults_;
+
+    /// The list of options, stored locally
+    std::vector<Option_p> options_;
+
+    ///@}
+    /// @name Help
+    ///@{
+
+    /// Footer to put after all options in the help output INHERITABLE
+    std::string footer_;
+
+    /// A pointer to the help flag if there is one INHERITABLE
+    Option *help_ptr_{nullptr};
+
+    /// A pointer to the help all flag if there is one INHERITABLE
+    Option *help_all_ptr_{nullptr};
+
+    /// This is the formatter for help printing. Default provided. INHERITABLE (same pointer)
+    std::shared_ptr<FormatterBase> formatter_{new Formatter()};
+
+    /// The error message printing function INHERITABLE
+    std::function<std::string(const App *, const Error &e)> failure_message_ = FailureMessage::simple;
+
+    ///@}
+    /// @name Parsing
+    ///@{
+
+    using missing_t = std::vector<std::pair<detail::Classifer, std::string>>;
+
+    /// Pair of classifier, string for missing options. (extra detail is removed on returning from parse)
+    ///
+    /// This is faster and cleaner than storing just a list of strings and reparsing. This may contain the -- separator.
+    missing_t missing_;
+
+    /// This is a list of pointers to options with the original parse order
+    std::vector<Option *> parse_order_;
+
+    /// This is a list of the subcommands collected, in order
+    std::vector<App *> parsed_subcommands_;
+
+    ///@}
+    /// @name Subcommands
+    ///@{
+
+    /// Storage for subcommand list
+    std::vector<App_p> subcommands_;
+
+    /// If true, the program name is not case sensitive INHERITABLE
+    bool ignore_case_{false};
+
+    /// Allow subcommand fallthrough, so that parent commands can collect commands after subcommand.  INHERITABLE
+    bool fallthrough_{false};
+
+    /// A pointer to the parent if this is a subcommand
+    App *parent_{nullptr};
+
+    /// True if this command/subcommand was parsed
+    bool parsed_{false};
+
+    /// Minimum required subcommands (not inheritable!)
+    size_t require_subcommand_min_ = 0;
+
+    /// Max number of subcommands allowed (parsing stops after this number). 0 is unlimited INHERITABLE
+    size_t require_subcommand_max_ = 0;
+
+    /// The group membership INHERITABLE
+    std::string group_{"Subcommands"};
+
+    ///@}
+    /// @name Config
+    ///@{
+
+    /// The name of the connected config file
+    std::string config_name_;
+
+    /// True if ini is required (throws if not present), if false simply keep going.
+    bool config_required_{false};
+
+    /// Pointer to the config option
+    Option *config_ptr_{nullptr};
+
+    /// This is the formatter for help printing. Default provided. INHERITABLE (same pointer)
+    std::shared_ptr<Config> config_formatter_{new ConfigINI()};
+
+    ///@}
+
+    /// Special private constructor for subcommand
+    App(std::string description_, std::string name, App *parent)
+        : name_(std::move(name)), description_(std::move(description_)), parent_(parent) {
+        // Inherit if not from a nullptr
+        if(parent_ != nullptr) {
+            if(parent_->help_ptr_ != nullptr)
+                set_help_flag(parent_->help_ptr_->get_name(false, true), parent_->help_ptr_->get_description());
+            if(parent_->help_all_ptr_ != nullptr)
+                set_help_all_flag(parent_->help_all_ptr_->get_name(false, true),
+                                  parent_->help_all_ptr_->get_description());
+
+            /// OptionDefaults
+            option_defaults_ = parent_->option_defaults_;
+
+            // INHERITABLE
+            failure_message_ = parent_->failure_message_;
+            allow_extras_ = parent_->allow_extras_;
+            allow_config_extras_ = parent_->allow_config_extras_;
+            prefix_command_ = parent_->prefix_command_;
+            ignore_case_ = parent_->ignore_case_;
+            fallthrough_ = parent_->fallthrough_;
+            group_ = parent_->group_;
+            footer_ = parent_->footer_;
+            formatter_ = parent_->formatter_;
+            config_formatter_ = parent_->config_formatter_;
+            require_subcommand_max_ = parent_->require_subcommand_max_;
+        }
+    }
+
+  public:
+    /// @name Basic
+    ///@{
+
+    /// Create a new program. Pass in the same arguments as main(), along with a help string.
+    explicit App(std::string description_ = "", std::string name = "") : App(description_, name, nullptr) {
+        set_help_flag("-h,--help", "Print this help message and exit");
+    }
+
+    /// virtual destructor
+    virtual ~App() = default;
+
+    /// Set a callback for the end of parsing.
+    ///
+    /// Due to a bug in c++11,
+    /// it is not possible to overload on std::function (fixed in c++14
+    /// and backported to c++11 on newer compilers). Use capture by reference
+    /// to get a pointer to App if needed.
+    App *callback(std::function<void()> callback) {
+        callback_ = callback;
+        return this;
+    }
+
+    /// Set a name for the app (empty will use parser to set the name)
+    App *name(std::string name = "") {
+        name_ = name;
+        return this;
+    }
+
+    /// Remove the error when extras are left over on the command line.
+    App *allow_extras(bool allow = true) {
+        allow_extras_ = allow;
+        return this;
+    }
+
+    /// Remove the error when extras are left over on the command line.
+    /// Will also call App::allow_extras().
+    App *allow_config_extras(bool allow = true) {
+        allow_extras(allow);
+        allow_config_extras_ = allow;
+        return this;
+    }
+
+    /// Do not parse anything after the first unrecognised option and return
+    App *prefix_command(bool allow = true) {
+        prefix_command_ = allow;
+        return this;
+    }
+
+    /// Ignore case. Subcommand inherit value.
+    App *ignore_case(bool value = true) {
+        ignore_case_ = value;
+        if(parent_ != nullptr) {
+            for(const auto &subc : parent_->subcommands_) {
+                if(subc.get() != this && (this->check_name(subc->name_) || subc->check_name(this->name_)))
+                    throw OptionAlreadyAdded(subc->name_);
+            }
+        }
+        return this;
+    }
+
+    /// Set the help formatter
+    App *formatter(std::shared_ptr<FormatterBase> fmt) {
+        formatter_ = fmt;
+        return this;
+    }
+
+    /// Set the help formatter
+    App *formatter_fn(std::function<std::string(const App *, std::string, AppFormatMode)> fmt) {
+        formatter_ = std::make_shared<FormatterLambda>(fmt);
+        return this;
+    }
+
+    /// Set the config formatter
+    App *config_formatter(std::shared_ptr<Config> fmt) {
+        config_formatter_ = fmt;
+        return this;
+    }
+
+    /// Check to see if this subcommand was parsed, true only if received on command line.
+    bool parsed() const { return parsed_; }
+
+    /// Get the OptionDefault object, to set option defaults
+    OptionDefaults *option_defaults() { return &option_defaults_; }
+
+    ///@}
+    /// @name Adding options
+    ///@{
+
+    /// Add an option, will automatically understand the type for common types.
+    ///
+    /// To use, create a variable with the expected type, and pass it in after the name.
+    /// After start is called, you can use count to see if the value was passed, and
+    /// the value will be initialized properly. Numbers, vectors, and strings are supported.
+    ///
+    /// ->required(), ->default, and the validators are options,
+    /// The positional options take an optional number of arguments.
+    ///
+    /// For example,
+    ///
+    ///     std::string filename;
+    ///     program.add_option("filename", filename, "description of filename");
+    ///
+    Option *add_option(std::string name, callback_t callback, std::string description = "", bool defaulted = false) {
+        Option myopt{name, description, callback, defaulted, this};
+
+        if(std::find_if(std::begin(options_), std::end(options_), [&myopt](const Option_p &v) {
+               return *v == myopt;
+           }) == std::end(options_)) {
+            options_.emplace_back();
+            Option_p &option = options_.back();
+            option.reset(new Option(name, description, callback, defaulted, this));
+            option_defaults_.copy_to(option.get());
+            return option.get();
+        } else
+            throw OptionAlreadyAdded(myopt.get_name());
+    }
+
+    /// Add option for non-vectors (duplicate copy needed without defaulted to avoid `iostream << value`)
+    template <typename T, enable_if_t<!is_vector<T>::value, detail::enabler> = detail::dummy>
+    Option *add_option(std::string name,
+                       T &variable, ///< The variable to set
+                       std::string description = "") {
+
+        CLI::callback_t fun = [&variable](CLI::results_t res) { return detail::lexical_cast(res[0], variable); };
+
+        Option *opt = add_option(name, fun, description, false);
+        opt->type_name(detail::type_name<T>());
+        return opt;
+    }
+
+    /// Add option for non-vectors with a default print
+    template <typename T, enable_if_t<!is_vector<T>::value, detail::enabler> = detail::dummy>
+    Option *add_option(std::string name,
+                       T &variable, ///< The variable to set
+                       std::string description,
+                       bool defaulted) {
+
+        CLI::callback_t fun = [&variable](CLI::results_t res) { return detail::lexical_cast(res[0], variable); };
+
+        Option *opt = add_option(name, fun, description, defaulted);
+        opt->type_name(detail::type_name<T>());
+        if(defaulted) {
+            std::stringstream out;
+            out << variable;
+            opt->default_str(out.str());
+        }
+        return opt;
+    }
+
+    /// Add option for vectors (no default)
+    template <typename T>
+    Option *add_option(std::string name,
+                       std::vector<T> &variable, ///< The variable vector to set
+                       std::string description = "") {
+
+        CLI::callback_t fun = [&variable](CLI::results_t res) {
+            bool retval = true;
+            variable.clear();
+            for(const auto &a : res) {
+                variable.emplace_back();
+                retval &= detail::lexical_cast(a, variable.back());
+            }
+            return (!variable.empty()) && retval;
+        };
+
+        Option *opt = add_option(name, fun, description, false);
+        opt->type_name(detail::type_name<T>())->type_size(-1);
+        return opt;
+    }
+
+    /// Add option for vectors
+    template <typename T>
+    Option *add_option(std::string name,
+                       std::vector<T> &variable, ///< The variable vector to set
+                       std::string description,
+                       bool defaulted) {
+
+        CLI::callback_t fun = [&variable](CLI::results_t res) {
+            bool retval = true;
+            variable.clear();
+            for(const auto &a : res) {
+                variable.emplace_back();
+                retval &= detail::lexical_cast(a, variable.back());
+            }
+            return (!variable.empty()) && retval;
+        };
+
+        Option *opt = add_option(name, fun, description, defaulted);
+        opt->type_name(detail::type_name<T>())->type_size(-1);
+        if(defaulted)
+            opt->default_str("[" + detail::join(variable) + "]");
+        return opt;
+    }
+
+    /// Set a help flag, replace the existing one if present
+    Option *set_help_flag(std::string name = "", std::string description = "") {
+        if(help_ptr_ != nullptr) {
+            remove_option(help_ptr_);
+            help_ptr_ = nullptr;
+        }
+
+        // Empty name will simply remove the help flag
+        if(!name.empty()) {
+            help_ptr_ = add_flag_function(name, [](size_t) -> void { throw CallForHelp(); }, description);
+            help_ptr_->short_circuit(true);
+            help_ptr_->configurable(false);
+        }
+
+        return help_ptr_;
+    }
+
+    /// Set a help all flag, replaced the existing one if present
+    Option *set_help_all_flag(std::string name = "", std::string description = "") {
+        if(help_all_ptr_ != nullptr) {
+            remove_option(help_all_ptr_);
+            help_all_ptr_ = nullptr;
+        }
+
+        // Empty name will simply remove the help all flag
+        if(!name.empty()) {
+            help_all_ptr_ = add_flag_function(name, [](size_t) -> void { throw CallForAllHelp(); }, description);
+            help_all_ptr_->short_circuit(true);
+            help_all_ptr_->configurable(false);
+        }
+
+        return help_all_ptr_;
+    }
+
+    /// Add option for flag
+    Option *add_flag(std::string name, std::string description = "") {
+        CLI::callback_t fun = [](CLI::results_t) { return true; };
+
+        Option *opt = add_option(name, fun, description, false);
+        if(opt->get_positional())
+            throw IncorrectConstruction::PositionalFlag(name);
+        opt->type_size(0);
+        return opt;
+    }
+
+    /// Add option for flag integer
+    template <typename T,
+              enable_if_t<std::is_integral<T>::value && !is_bool<T>::value, detail::enabler> = detail::dummy>
+    Option *add_flag(std::string name,
+                     T &count, ///< A variable holding the count
+                     std::string description = "") {
+
+        count = 0;
+        CLI::callback_t fun = [&count](CLI::results_t res) {
+            count = static_cast<T>(res.size());
+            return true;
+        };
+
+        Option *opt = add_option(name, fun, description, false);
+        if(opt->get_positional())
+            throw IncorrectConstruction::PositionalFlag(name);
+        opt->type_size(0);
+        return opt;
+    }
+
+    /// Bool version - defaults to allowing multiple passings, but can be forced to one if
+    /// `multi_option_policy(CLI::MultiOptionPolicy::Throw)` is used.
+    template <typename T, enable_if_t<is_bool<T>::value, detail::enabler> = detail::dummy>
+    Option *add_flag(std::string name,
+                     T &count, ///< A variable holding true if passed
+                     std::string description = "") {
+
+        count = false;
+        CLI::callback_t fun = [&count](CLI::results_t res) {
+            count = true;
+            return res.size() == 1;
+        };
+
+        Option *opt = add_option(name, fun, description, false);
+        if(opt->get_positional())
+            throw IncorrectConstruction::PositionalFlag(name);
+        opt->type_size(0);
+        opt->multi_option_policy(CLI::MultiOptionPolicy::TakeLast);
+        return opt;
+    }
+
+    /// Add option for callback
+    Option *add_flag_function(std::string name,
+                              std::function<void(size_t)> function, ///< A function to call, void(size_t)
+                              std::string description = "") {
+
+        CLI::callback_t fun = [function](CLI::results_t res) {
+            auto count = static_cast<size_t>(res.size());
+            function(count);
+            return true;
+        };
+
+        Option *opt = add_option(name, fun, description, false);
+        if(opt->get_positional())
+            throw IncorrectConstruction::PositionalFlag(name);
+        opt->type_size(0);
+        return opt;
+    }
+
+#ifdef CLI11_CPP14
+    /// Add option for callback (C++14 or better only)
+    Option *add_flag(std::string name,
+                     std::function<void(size_t)> function, ///< A function to call, void(size_t)
+                     std::string description = "") {
+        return add_flag_function(name, function, description);
+    }
+#endif
+
+    /// Add set of options (No default, temp refernce, such as an inline set)
+    template <typename T>
+    Option *add_set(std::string name,
+                    T &member,                   ///< The selected member of the set
+                    const std::set<T> &&options, ///< The set of possibilities
+                    std::string description = "") {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&member, options, simple_name](CLI::results_t res) {
+            bool retval = detail::lexical_cast(res[0], member);
+            if(!retval)
+                throw ConversionError(res[0], simple_name);
+            return std::find(std::begin(options), std::end(options), member) != std::end(options);
+        };
+
+        Option *opt = add_option(name, fun, description, false);
+        std::string typeval = detail::type_name<T>();
+        typeval += " in {" + detail::join(options) + "}";
+        opt->type_name(typeval);
+        return opt;
+    }
+
+    /// Add set of options (No default, non-temp refernce, such as an existing set)
+    template <typename T>
+    Option *add_set(std::string name,
+                    T &member,                  ///< The selected member of the set
+                    const std::set<T> &options, ///< The set of possibilities
+                    std::string description = "") {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&member, &options, simple_name](CLI::results_t res) {
+            bool retval = detail::lexical_cast(res[0], member);
+            if(!retval)
+                throw ConversionError(res[0], simple_name);
+            return std::find(std::begin(options), std::end(options), member) != std::end(options);
+        };
+
+        Option *opt = add_option(name, fun, description, false);
+        opt->type_name_fn(
+            [&options]() { return std::string(detail::type_name<T>()) + " in {" + detail::join(options) + "}"; });
+
+        return opt;
+    }
+
+    /// Add set of options (with default, R value, such as an inline set)
+    template <typename T>
+    Option *add_set(std::string name,
+                    T &member,                   ///< The selected member of the set
+                    const std::set<T> &&options, ///< The set of posibilities
+                    std::string description,
+                    bool defaulted) {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&member, options, simple_name](CLI::results_t res) {
+            bool retval = detail::lexical_cast(res[0], member);
+            if(!retval)
+                throw ConversionError(res[0], simple_name);
+            return std::find(std::begin(options), std::end(options), member) != std::end(options);
+        };
+
+        Option *opt = add_option(name, fun, description, defaulted);
+        std::string typeval = detail::type_name<T>();
+        typeval += " in {" + detail::join(options) + "}";
+        opt->type_name(typeval);
+        if(defaulted) {
+            std::stringstream out;
+            out << member;
+            opt->default_str(out.str());
+        }
+        return opt;
+    }
+
+    /// Add set of options (with default, L value refernce, such as an existing set)
+    template <typename T>
+    Option *add_set(std::string name,
+                    T &member,                  ///< The selected member of the set
+                    const std::set<T> &options, ///< The set of posibilities
+                    std::string description,
+                    bool defaulted) {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&member, &options, simple_name](CLI::results_t res) {
+            bool retval = detail::lexical_cast(res[0], member);
+            if(!retval)
+                throw ConversionError(res[0], simple_name);
+            return std::find(std::begin(options), std::end(options), member) != std::end(options);
+        };
+
+        Option *opt = add_option(name, fun, description, defaulted);
+        opt->type_name_fn(
+            [&options]() { return std::string(detail::type_name<T>()) + " in {" + detail::join(options) + "}"; });
+        if(defaulted) {
+            std::stringstream out;
+            out << member;
+            opt->default_str(out.str());
+        }
+        return opt;
+    }
+
+    /// Add set of options, string only, ignore case (no default, R value)
+    Option *add_set_ignore_case(std::string name,
+                                std::string &member,                   ///< The selected member of the set
+                                const std::set<std::string> &&options, ///< The set of possibilities
+                                std::string description = "") {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&member, options, simple_name](CLI::results_t res) {
+            member = detail::to_lower(res[0]);
+            auto iter = std::find_if(std::begin(options), std::end(options), [&member](std::string val) {
+                return detail::to_lower(val) == member;
+            });
+            if(iter == std::end(options))
+                throw ConversionError(member, simple_name);
+            else {
+                member = *iter;
+                return true;
+            }
+        };
+
+        Option *opt = add_option(name, fun, description, false);
+        std::string typeval = detail::type_name<std::string>();
+        typeval += " in {" + detail::join(options) + "}";
+        opt->type_name(typeval);
+
+        return opt;
+    }
+
+    /// Add set of options, string only, ignore case (no default, L value)
+    Option *add_set_ignore_case(std::string name,
+                                std::string &member,                  ///< The selected member of the set
+                                const std::set<std::string> &options, ///< The set of possibilities
+                                std::string description = "") {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&member, &options, simple_name](CLI::results_t res) {
+            member = detail::to_lower(res[0]);
+            auto iter = std::find_if(std::begin(options), std::end(options), [&member](std::string val) {
+                return detail::to_lower(val) == member;
+            });
+            if(iter == std::end(options))
+                throw ConversionError(member, simple_name);
+            else {
+                member = *iter;
+                return true;
+            }
+        };
+
+        Option *opt = add_option(name, fun, description, false);
+        opt->type_name_fn([&options]() {
+            return std::string(detail::type_name<std::string>()) + " in {" + detail::join(options) + "}";
+        });
+
+        return opt;
+    }
+
+    /// Add set of options, string only, ignore case (default, R value)
+    Option *add_set_ignore_case(std::string name,
+                                std::string &member,                   ///< The selected member of the set
+                                const std::set<std::string> &&options, ///< The set of posibilities
+                                std::string description,
+                                bool defaulted) {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&member, options, simple_name](CLI::results_t res) {
+            member = detail::to_lower(res[0]);
+            auto iter = std::find_if(std::begin(options), std::end(options), [&member](std::string val) {
+                return detail::to_lower(val) == member;
+            });
+            if(iter == std::end(options))
+                throw ConversionError(member, simple_name);
+            else {
+                member = *iter;
+                return true;
+            }
+        };
+
+        Option *opt = add_option(name, fun, description, defaulted);
+        std::string typeval = detail::type_name<std::string>();
+        typeval += " in {" + detail::join(options) + "}";
+        opt->type_name(typeval);
+        if(defaulted) {
+            opt->default_str(member);
+        }
+        return opt;
+    }
+
+    /// Add set of options, string only, ignore case (default, L value)
+    Option *add_set_ignore_case(std::string name,
+                                std::string &member,                  ///< The selected member of the set
+                                const std::set<std::string> &options, ///< The set of posibilities
+                                std::string description,
+                                bool defaulted) {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&member, &options, simple_name](CLI::results_t res) {
+            member = detail::to_lower(res[0]);
+            auto iter = std::find_if(std::begin(options), std::end(options), [&member](std::string val) {
+                return detail::to_lower(val) == member;
+            });
+            if(iter == std::end(options))
+                throw ConversionError(member, simple_name);
+            else {
+                member = *iter;
+                return true;
+            }
+        };
+
+        Option *opt = add_option(name, fun, description, defaulted);
+        opt->type_name_fn([&options]() {
+            return std::string(detail::type_name<std::string>()) + " in {" + detail::join(options) + "}";
+        });
+        if(defaulted) {
+            opt->default_str(member);
+        }
+        return opt;
+    }
+
+    /// Add a complex number
+    template <typename T>
+    Option *add_complex(std::string name,
+                        T &variable,
+                        std::string description = "",
+                        bool defaulted = false,
+                        std::string label = "COMPLEX") {
+
+        std::string simple_name = CLI::detail::split(name, ',').at(0);
+        CLI::callback_t fun = [&variable, simple_name, label](results_t res) {
+            if(res[1].back() == 'i')
+                res[1].pop_back();
+            double x, y;
+            bool worked = detail::lexical_cast(res[0], x) && detail::lexical_cast(res[1], y);
+            if(worked)
+                variable = T(x, y);
+            return worked;
+        };
+
+        CLI::Option *opt = add_option(name, fun, description, defaulted);
+        opt->type_name(label)->type_size(2);
+        if(defaulted) {
+            std::stringstream out;
+            out << variable;
+            opt->default_str(out.str());
+        }
+        return opt;
+    }
+
+    /// Set a configuration ini file option, or clear it if no name passed
+    Option *set_config(std::string name = "",
+                       std::string default_filename = "",
+                       std::string help = "Read an ini file",
+                       bool required = false) {
+
+        // Remove existing config if present
+        if(config_ptr_ != nullptr)
+            remove_option(config_ptr_);
+
+        // Only add config if option passed
+        if(!name.empty()) {
+            config_name_ = default_filename;
+            config_required_ = required;
+            config_ptr_ = add_option(name, config_name_, help, !default_filename.empty());
+            config_ptr_->configurable(false);
+        }
+
+        return config_ptr_;
+    }
+
+    /// Removes an option from the App. Takes an option pointer. Returns true if found and removed.
+    bool remove_option(Option *opt) {
+        // Make sure no links exist
+        for(Option_p &op : options_) {
+            op->remove_needs(opt);
+            op->remove_excludes(opt);
+        }
+
+        auto iterator =
+            std::find_if(std::begin(options_), std::end(options_), [opt](const Option_p &v) { return v.get() == opt; });
+        if(iterator != std::end(options_)) {
+            options_.erase(iterator);
+            return true;
+        }
+        return false;
+    }
+
+    ///@}
+    /// @name Subcommmands
+    ///@{
+
+    /// Add a subcommand. Inherits INHERITABLE and OptionDefaults, and help flag
+    App *add_subcommand(std::string name, std::string description = "") {
+        subcommands_.emplace_back(new App(description, name, this));
+        for(const auto &subc : subcommands_)
+            if(subc.get() != subcommands_.back().get())
+                if(subc->check_name(subcommands_.back()->name_) || subcommands_.back()->check_name(subc->name_))
+                    throw OptionAlreadyAdded(subc->name_);
+        return subcommands_.back().get();
+    }
+
+    /// Check to see if a subcommand is part of this command (doesn't have to be in command line)
+    App *get_subcommand(App *subcom) const {
+        for(const App_p &subcomptr : subcommands_)
+            if(subcomptr.get() == subcom)
+                return subcom;
+        throw OptionNotFound(subcom->get_name());
+    }
+
+    /// Check to see if a subcommand is part of this command (text version)
+    App *get_subcommand(std::string subcom) const {
+        for(const App_p &subcomptr : subcommands_)
+            if(subcomptr->check_name(subcom))
+                return subcomptr.get();
+        throw OptionNotFound(subcom);
+    }
+
+    /// Changes the group membership
+    App *group(std::string name) {
+        group_ = name;
+        return this;
+    }
+
+    /// The argumentless form of require subcommand requires 1 or more subcommands
+    App *require_subcommand() {
+        require_subcommand_min_ = 1;
+        require_subcommand_max_ = 0;
+        return this;
+    }
+
+    /// Require a subcommand to be given (does not affect help call)
+    /// The number required can be given. Negative values indicate maximum
+    /// number allowed (0 for any number). Max number inheritable.
+    App *require_subcommand(int value) {
+        if(value < 0) {
+            require_subcommand_min_ = 0;
+            require_subcommand_max_ = static_cast<size_t>(-value);
+        } else {
+            require_subcommand_min_ = static_cast<size_t>(value);
+            require_subcommand_max_ = static_cast<size_t>(value);
+        }
+        return this;
+    }
+
+    /// Explicitly control the number of subcommands required. Setting 0
+    /// for the max means unlimited number allowed. Max number inheritable.
+    App *require_subcommand(size_t min, size_t max) {
+        require_subcommand_min_ = min;
+        require_subcommand_max_ = max;
+        return this;
+    }
+
+    /// Stop subcommand fallthrough, so that parent commands cannot collect commands after subcommand.
+    /// Default from parent, usually set on parent.
+    App *fallthrough(bool value = true) {
+        fallthrough_ = value;
+        return this;
+    }
+
+    /// Check to see if this subcommand was parsed, true only if received on command line.
+    /// This allows the subcommand to be directly checked.
+    operator bool() const { return parsed_; }
+
+    ///@}
+    /// @name Extras for subclassing
+    ///@{
+
+    /// This allows subclasses to inject code before callbacks but after parse.
+    ///
+    /// This does not run if any errors or help is thrown.
+    virtual void pre_callback() {}
+
+    ///@}
+    /// @name Parsing
+    ///@{
+    //
+    /// Reset the parsed data
+    void clear() {
+
+        parsed_ = false;
+        missing_.clear();
+        parsed_subcommands_.clear();
+
+        for(const Option_p &opt : options_) {
+            opt->clear();
+        }
+        for(const App_p &app : subcommands_) {
+            app->clear();
+        }
+    }
+
+    /// Parses the command line - throws errors
+    /// This must be called after the options are in but before the rest of the program.
+    void parse(int argc, const char *const *argv) {
+        // If the name is not set, read from command line
+        if(name_.empty())
+            name_ = argv[0];
+
+        std::vector<std::string> args;
+        for(int i = argc - 1; i > 0; i--)
+            args.emplace_back(argv[i]);
+        parse(args);
+    }
+
+    /// The real work is done here. Expects a reversed vector.
+    /// Changes the vector to the remaining options.
+    void parse(std::vector<std::string> &args) {
+        // Clear if parsed
+        if(parsed_)
+            clear();
+
+        // Redundant (set by _parse on commands/subcommands)
+        // but placed here to make sure this is cleared when
+        // running parse after an error is thrown, even by _validate.
+        parsed_ = true;
+
+        _validate();
+        _parse(args);
+        run_callback();
+    }
+
+    /// Provide a function to print a help message. The function gets access to the App pointer and error.
+    void failure_message(std::function<std::string(const App *, const Error &e)> function) {
+        failure_message_ = function;
+    }
+
+    /// Print a nice error message and return the exit code
+    int exit(const Error &e, std::ostream &out = std::cout, std::ostream &err = std::cerr) const {
+
+        /// Avoid printing anything if this is a CLI::RuntimeError
+        if(dynamic_cast<const CLI::RuntimeError *>(&e) != nullptr)
+            return e.get_exit_code();
+
+        if(dynamic_cast<const CLI::CallForHelp *>(&e) != nullptr) {
+            out << help();
+            return e.get_exit_code();
+        }
+
+        if(dynamic_cast<const CLI::CallForAllHelp *>(&e) != nullptr) {
+            out << help("", AppFormatMode::All);
+            return e.get_exit_code();
+        }
+
+        if(e.get_exit_code() != static_cast<int>(ExitCodes::Success)) {
+            if(failure_message_)
+                err << failure_message_(this, e) << std::flush;
+        }
+
+        return e.get_exit_code();
+    }
+
+    ///@}
+    /// @name Post parsing
+    ///@{
+
+    /// Counts the number of times the given option was passed.
+    size_t count(std::string name) const {
+        for(const Option_p &opt : options_) {
+            if(opt->check_name(name)) {
+                return opt->count();
+            }
+        }
+        throw OptionNotFound(name);
+    }
+
+    /// Get a subcommand pointer list to the currently selected subcommands (after parsing by by default, in command
+    /// line order; use parsed = false to get the original definition list.)
+    std::vector<App *> get_subcommands() const { return parsed_subcommands_; }
+
+    /// Get a filtered subcommand pointer list from the original definition list. An empty function will provide all
+    /// subcommands (const)
+    std::vector<const App *> get_subcommands(const std::function<bool(const App *)> &filter) const {
+        std::vector<const App *> subcomms(subcommands_.size());
+        std::transform(std::begin(subcommands_), std::end(subcommands_), std::begin(subcomms), [](const App_p &v) {
+            return v.get();
+        });
+
+        if(filter) {
+            subcomms.erase(std::remove_if(std::begin(subcomms),
+                                          std::end(subcomms),
+                                          [&filter](const App *app) { return !filter(app); }),
+                           std::end(subcomms));
+        }
+
+        return subcomms;
+    }
+
+    /// Get a filtered subcommand pointer list from the original definition list. An empty function will provide all
+    /// subcommands
+    std::vector<App *> get_subcommands(const std::function<bool(App *)> &filter) {
+        std::vector<App *> subcomms(subcommands_.size());
+        std::transform(std::begin(subcommands_), std::end(subcommands_), std::begin(subcomms), [](const App_p &v) {
+            return v.get();
+        });
+
+        if(filter) {
+            subcomms.erase(
+                std::remove_if(std::begin(subcomms), std::end(subcomms), [&filter](App *app) { return !filter(app); }),
+                std::end(subcomms));
+        }
+
+        return subcomms;
+    }
+
+    /// Check to see if given subcommand was selected
+    bool got_subcommand(App *subcom) const {
+        // get subcom needed to verify that this was a real subcommand
+        return get_subcommand(subcom)->parsed_;
+    }
+
+    /// Check with name instead of pointer to see if subcommand was selected
+    bool got_subcommand(std::string name) const { return get_subcommand(name)->parsed_; }
+
+    ///@}
+    /// @name Help
+    ///@{
+
+    /// Set footer.
+    App *footer(std::string footer) {
+        footer_ = footer;
+        return this;
+    }
+
+    /// Produce a string that could be read in as a config of the current values of the App. Set default_also to include
+    /// default arguments. Prefix will add a string to the beginning of each option.
+    std::string config_to_str(bool default_also = false, bool write_description = false) const {
+        return config_formatter_->to_config(this, default_also, write_description, "");
+    }
+
+    /// Makes a help message, using the currently configured formatter
+    /// Will only do one subcommand at a time
+    std::string help(std::string prev = "", AppFormatMode mode = AppFormatMode::Normal) const {
+        if(prev.empty())
+            prev = get_name();
+        else
+            prev += " " + get_name();
+
+        // Delegate to subcommand if needed
+        auto selected_subcommands = get_subcommands();
+        if(!selected_subcommands.empty())
+            return selected_subcommands.at(0)->help(prev, mode);
+        else
+            return formatter_->make_help(this, prev, mode);
+    }
+
+    /// Provided for backwards compatibility \deprecated
+    CLI11_DEPRECATED("Please use footer instead")
+    App *set_footer(std::string msg) { return footer(msg); }
+
+    /// Provided for backwards compatibility \deprecated
+    CLI11_DEPRECATED("Please use name instead")
+    App *set_name(std::string msg) { return name(msg); }
+
+    /// Provided for backwards compatibility \deprecated
+    CLI11_DEPRECATED("Please use callback instead")
+    App *set_callback(std::function<void()> fn) { return callback(fn); }
+
+    ///@}
+    /// @name Getters
+    ///@{
+
+    /// Access the formatter
+    std::shared_ptr<FormatterBase> get_formatter() const { return formatter_; }
+
+    /// Access the config formatter
+    std::shared_ptr<Config> get_config_formatter() const { return config_formatter_; }
+
+    /// Get the app or subcommand description
+    std::string get_description() const { return description_; }
+
+    /// Get the list of options (user facing function, so returns raw pointers), has optional filter function
+    std::vector<const Option *> get_options(const std::function<bool(const Option *)> filter = {}) const {
+        std::vector<const Option *> options(options_.size());
+        std::transform(std::begin(options_), std::end(options_), std::begin(options), [](const Option_p &val) {
+            return val.get();
+        });
+
+        if(filter) {
+            options.erase(std::remove_if(std::begin(options),
+                                         std::end(options),
+                                         [&filter](const Option *opt) { return !filter(opt); }),
+                          std::end(options));
+        }
+
+        return options;
+    }
+
+    /// Get an option by name
+    const Option *get_option(std::string name) const {
+        for(const Option_p &opt : options_) {
+            if(opt->check_name(name)) {
+                return opt.get();
+            }
+        }
+        throw OptionNotFound(name);
+    }
+
+    /// Get an option by name (non-const version)
+    Option *get_option(std::string name) {
+        for(Option_p &opt : options_) {
+            if(opt->check_name(name)) {
+                return opt.get();
+            }
+        }
+        throw OptionNotFound(name);
+    }
+
+    /// Check the status of ignore_case
+    bool get_ignore_case() const { return ignore_case_; }
+
+    /// Check the status of fallthrough
+    bool get_fallthrough() const { return fallthrough_; }
+
+    /// Get the group of this subcommand
+    const std::string &get_group() const { return group_; }
+
+    /// Get footer.
+    std::string get_footer() const { return footer_; }
+
+    /// Get the required min subcommand value
+    size_t get_require_subcommand_min() const { return require_subcommand_min_; }
+
+    /// Get the required max subcommand value
+    size_t get_require_subcommand_max() const { return require_subcommand_max_; }
+
+    /// Get the prefix command status
+    bool get_prefix_command() const { return prefix_command_; }
+
+    /// Get the status of allow extras
+    bool get_allow_extras() const { return allow_extras_; }
+
+    /// Get the status of allow extras
+    bool get_allow_config_extras() const { return allow_config_extras_; }
+
+    /// Get a pointer to the help flag.
+    Option *get_help_ptr() { return help_ptr_; }
+
+    /// Get a pointer to the help flag. (const)
+    const Option *get_help_ptr() const { return help_ptr_; }
+
+    /// Get a pointer to the help all flag. (const)
+    const Option *get_help_all_ptr() const { return help_all_ptr_; }
+
+    /// Get a pointer to the config option.
+    Option *get_config_ptr() { return config_ptr_; }
+
+    /// Get a pointer to the config option. (const)
+    const Option *get_config_ptr() const { return config_ptr_; }
+
+    /// Get the parent of this subcommand (or nullptr if master app)
+    App *get_parent() { return parent_; }
+
+    /// Get the parent of this subcommand (or nullptr if master app) (const version)
+    const App *get_parent() const { return parent_; }
+
+    /// Get the name of the current app
+    std::string get_name() const { return name_; }
+
+    /// Check the name, case insensitive if set
+    bool check_name(std::string name_to_check) const {
+        std::string local_name = name_;
+        if(ignore_case_) {
+            local_name = detail::to_lower(name_);
+            name_to_check = detail::to_lower(name_to_check);
+        }
+
+        return local_name == name_to_check;
+    }
+
+    /// Get the groups available directly from this option (in order)
+    std::vector<std::string> get_groups() const {
+        std::vector<std::string> groups;
+
+        for(const Option_p &opt : options_) {
+            // Add group if it is not already in there
+            if(std::find(groups.begin(), groups.end(), opt->get_group()) == groups.end()) {
+                groups.push_back(opt->get_group());
+            }
+        }
+
+        return groups;
+    }
+
+    /// This gets a vector of pointers with the original parse order
+    const std::vector<Option *> &parse_order() const { return parse_order_; }
+
+    /// This returns the missing options from the current subcommand
+    std::vector<std::string> remaining(bool recurse = false) const {
+        std::vector<std::string> miss_list;
+        for(const std::pair<detail::Classifer, std::string> &miss : missing_) {
+            miss_list.push_back(std::get<1>(miss));
+        }
+
+        // Recurse into subcommands
+        if(recurse) {
+            for(const App *sub : parsed_subcommands_) {
+                std::vector<std::string> output = sub->remaining(recurse);
+                std::copy(std::begin(output), std::end(output), std::back_inserter(miss_list));
+            }
+        }
+        return miss_list;
+    }
+
+    /// This returns the number of remaining options, minus the -- seperator
+    size_t remaining_size(bool recurse = false) const {
+        auto count = static_cast<size_t>(std::count_if(
+            std::begin(missing_), std::end(missing_), [](const std::pair<detail::Classifer, std::string> &val) {
+                return val.first != detail::Classifer::POSITIONAL_MARK;
+            }));
+        if(recurse) {
+            for(const App_p &sub : subcommands_) {
+                count += sub->remaining_size(recurse);
+            }
+        }
+        return count;
+    }
+
+    ///@}
+
+  protected:
+    /// Check the options to make sure there are no conflicts.
+    ///
+    /// Currently checks to see if multiple positionals exist with -1 args
+    void _validate() const {
+        auto count = std::count_if(std::begin(options_), std::end(options_), [](const Option_p &opt) {
+            return opt->get_items_expected() < 0 && opt->get_positional();
+        });
+        if(count > 1)
+            throw InvalidError(name_);
+        for(const App_p &app : subcommands_)
+            app->_validate();
+    }
+
+    /// Internal function to run (App) callback, top down
+    void run_callback() {
+        pre_callback();
+        if(callback_)
+            callback_();
+        for(App *subc : get_subcommands()) {
+            subc->run_callback();
+        }
+    }
+
+    /// Check to see if a subcommand is valid. Give up immediately if subcommand max has been reached.
+    bool _valid_subcommand(const std::string &current) const {
+        // Don't match if max has been reached - but still check parents
+        if(require_subcommand_max_ != 0 && parsed_subcommands_.size() >= require_subcommand_max_) {
+            return parent_ != nullptr && parent_->_valid_subcommand(current);
+        }
+
+        for(const App_p &com : subcommands_)
+            if(com->check_name(current) && !*com)
+                return true;
+
+        // Check parent if exists, else return false
+        return parent_ != nullptr && parent_->_valid_subcommand(current);
+    }
+
+    /// Selects a Classifier enum based on the type of the current argument
+    detail::Classifer _recognize(const std::string &current) const {
+        std::string dummy1, dummy2;
+
+        if(current == "--")
+            return detail::Classifer::POSITIONAL_MARK;
+        if(_valid_subcommand(current))
+            return detail::Classifer::SUBCOMMAND;
+        if(detail::split_long(current, dummy1, dummy2))
+            return detail::Classifer::LONG;
+        if(detail::split_short(current, dummy1, dummy2))
+            return detail::Classifer::SHORT;
+        return detail::Classifer::NONE;
+    }
+
+    /// Internal parse function
+    void _parse(std::vector<std::string> &args) {
+        parsed_ = true;
+        bool positional_only = false;
+
+        while(!args.empty()) {
+            _parse_single(args, positional_only);
+        }
+
+        for(const Option_p &opt : options_)
+            if(opt->get_short_circuit() && opt->count() > 0)
+                opt->run_callback();
+
+        // Process an INI file
+        if(config_ptr_ != nullptr) {
+            if(*config_ptr_) {
+                config_ptr_->run_callback();
+                config_required_ = true;
+            }
+            if(!config_name_.empty()) {
+                try {
+                    std::vector<ConfigItem> values = config_formatter_->from_file(config_name_);
+                    _parse_config(values);
+                } catch(const FileError &) {
+                    if(config_required_)
+                        throw;
+                }
+            }
+        }
+
+        // Get envname options if not yet passed
+        for(const Option_p &opt : options_) {
+            if(opt->count() == 0 && !opt->envname_.empty()) {
+                char *buffer = nullptr;
+                std::string ename_string;
+
+#ifdef _MSC_VER
+                // Windows version
+                size_t sz = 0;
+                if(_dupenv_s(&buffer, &sz, opt->envname_.c_str()) == 0 && buffer != nullptr) {
+                    ename_string = std::string(buffer);
+                    free(buffer);
+                }
+#else
+                // This also works on Windows, but gives a warning
+                buffer = std::getenv(opt->envname_.c_str());
+                if(buffer != nullptr)
+                    ename_string = std::string(buffer);
+#endif
+
+                if(!ename_string.empty()) {
+                    opt->add_result(ename_string);
+                }
+            }
+        }
+
+        // Process callbacks
+        for(const Option_p &opt : options_) {
+            if(opt->count() > 0 && !opt->get_callback_run()) {
+                opt->run_callback();
+            }
+        }
+
+        // Verify required options
+        for(const Option_p &opt : options_) {
+            // Exit if a help flag was passed (requirements not required in that case)
+            if(_any_help_flag())
+                break;
+
+            // Required or partially filled
+            if(opt->get_required() || opt->count() != 0) {
+                // Make sure enough -N arguments parsed (+N is already handled in parsing function)
+                if(opt->get_items_expected() < 0 && opt->count() < static_cast<size_t>(-opt->get_items_expected()))
+                    throw ArgumentMismatch::AtLeast(opt->get_name(), -opt->get_items_expected());
+
+                // Required but empty
+                if(opt->get_required() && opt->count() == 0)
+                    throw RequiredError(opt->get_name());
+            }
+            // Requires
+            for(const Option *opt_req : opt->needs_)
+                if(opt->count() > 0 && opt_req->count() == 0)
+                    throw RequiresError(opt->get_name(), opt_req->get_name());
+            // Excludes
+            for(const Option *opt_ex : opt->excludes_)
+                if(opt->count() > 0 && opt_ex->count() != 0)
+                    throw ExcludesError(opt->get_name(), opt_ex->get_name());
+        }
+
+        auto selected_subcommands = get_subcommands();
+        if(require_subcommand_min_ > selected_subcommands.size())
+            throw RequiredError::Subcommand(require_subcommand_min_);
+
+        // Convert missing (pairs) to extras (string only)
+        if(!(allow_extras_ || prefix_command_)) {
+            size_t num_left_over = remaining_size();
+            if(num_left_over > 0) {
+                args = remaining(false);
+                throw ExtrasError(args);
+            }
+        }
+
+        if(parent_ == nullptr) {
+            args = remaining(false);
+        }
+    }
+
+    /// Return True if a help flag detected (checks all parents) (only run if help called before subcommand)
+    bool _any_help_flag() const {
+        bool result = false;
+        const Option *help_ptr = get_help_ptr();
+        const Option *help_all_ptr = get_help_all_ptr();
+        if(help_ptr != nullptr && help_ptr->count() > 0)
+            result = true;
+        if(help_all_ptr != nullptr && help_all_ptr->count() > 0)
+            result = true;
+        if(parent_ != nullptr)
+            return result || parent_->_any_help_flag();
+        else
+            return result;
+    }
+
+    /// Parse one config param, return false if not found in any subcommand, remove if it is
+    ///
+    /// If this has more than one dot.separated.name, go into the subcommand matching it
+    /// Returns true if it managed to find the option, if false you'll need to remove the arg manually.
+    void _parse_config(std::vector<ConfigItem> &args) {
+        for(ConfigItem item : args) {
+            if(!_parse_single_config(item) && !allow_config_extras_)
+                throw ConfigError::Extras(item.fullname());
+        }
+    }
+
+    /// Fill in a single config option
+    bool _parse_single_config(const ConfigItem &item, size_t level = 0) {
+        if(level < item.parents.size()) {
+            App *subcom;
+            try {
+                subcom = get_subcommand(item.parents.at(level));
+            } catch(const OptionNotFound &) {
+                return false;
+            }
+            return subcom->_parse_single_config(item, level + 1);
+        }
+
+        Option *op;
+        try {
+            op = get_option("--" + item.name);
+        } catch(const OptionNotFound &) {
+            // If the option was not present
+            if(get_allow_config_extras())
+                // Should we worry about classifying the extras properly?
+                missing_.emplace_back(detail::Classifer::NONE, item.fullname());
+            return false;
+        }
+
+        if(!op->get_configurable())
+            throw ConfigError::NotConfigurable(item.fullname());
+
+        if(op->empty()) {
+            // Flag parsing
+            if(op->get_type_size() == 0) {
+                op->set_results(config_formatter_->to_flag(item));
+            } else {
+                op->set_results(item.inputs);
+                op->run_callback();
+            }
+        }
+
+        return true;
+    }
+
+    /// Parse "one" argument (some may eat more than one), delegate to parent if fails, add to missing if missing from
+    /// master
+    void _parse_single(std::vector<std::string> &args, bool &positional_only) {
+
+        detail::Classifer classifer = positional_only ? detail::Classifer::NONE : _recognize(args.back());
+        switch(classifer) {
+        case detail::Classifer::POSITIONAL_MARK:
+            missing_.emplace_back(classifer, args.back());
+            args.pop_back();
+            positional_only = true;
+            break;
+        case detail::Classifer::SUBCOMMAND:
+            _parse_subcommand(args);
+            break;
+        case detail::Classifer::LONG:
+            // If already parsed a subcommand, don't accept options_
+            _parse_arg(args, true);
+            break;
+        case detail::Classifer::SHORT:
+            // If already parsed a subcommand, don't accept options_
+            _parse_arg(args, false);
+            break;
+        case detail::Classifer::NONE:
+            // Probably a positional or something for a parent (sub)command
+            _parse_positional(args);
+        }
+    }
+
+    /// Count the required remaining positional arguments
+    size_t _count_remaining_positionals(bool required = false) const {
+        size_t retval = 0;
+        for(const Option_p &opt : options_)
+            if(opt->get_positional() && (!required || opt->get_required()) && opt->get_items_expected() > 0 &&
+               static_cast<int>(opt->count()) < opt->get_items_expected())
+                retval = static_cast<size_t>(opt->get_items_expected()) - opt->count();
+
+        return retval;
+    }
+
+    /// Parse a positional, go up the tree to check
+    void _parse_positional(std::vector<std::string> &args) {
+
+        std::string positional = args.back();
+        for(const Option_p &opt : options_) {
+            // Eat options, one by one, until done
+            if(opt->get_positional() &&
+               (static_cast<int>(opt->count()) < opt->get_items_expected() || opt->get_items_expected() < 0)) {
+
+                opt->add_result(positional);
+                parse_order_.push_back(opt.get());
+                args.pop_back();
+                return;
+            }
+        }
+
+        if(parent_ != nullptr && fallthrough_)
+            return parent_->_parse_positional(args);
+        else {
+            args.pop_back();
+            missing_.emplace_back(detail::Classifer::NONE, positional);
+
+            if(prefix_command_) {
+                while(!args.empty()) {
+                    missing_.emplace_back(detail::Classifer::NONE, args.back());
+                    args.pop_back();
+                }
+            }
+        }
+    }
+
+    /// Parse a subcommand, modify args and continue
+    ///
+    /// Unlike the others, this one will always allow fallthrough
+    void _parse_subcommand(std::vector<std::string> &args) {
+        if(_count_remaining_positionals(/* required */ true) > 0)
+            return _parse_positional(args);
+        for(const App_p &com : subcommands_) {
+            if(com->check_name(args.back())) {
+                args.pop_back();
+                if(std::find(std::begin(parsed_subcommands_), std::end(parsed_subcommands_), com.get()) ==
+                   std::end(parsed_subcommands_))
+                    parsed_subcommands_.push_back(com.get());
+                com->_parse(args);
+                return;
+            }
+        }
+        if(parent_ != nullptr)
+            return parent_->_parse_subcommand(args);
+        else
+            throw HorribleError("Subcommand " + args.back() + " missing");
+    }
+
+    /// Parse a short (false) or long (true) argument, must be at the top of the list
+    void _parse_arg(std::vector<std::string> &args, bool second_dash) {
+
+        detail::Classifer current_type = second_dash ? detail::Classifer::LONG : detail::Classifer::SHORT;
+
+        std::string current = args.back();
+
+        std::string name;
+        std::string value;
+        std::string rest;
+
+        if(second_dash) {
+            if(!detail::split_long(current, name, value))
+                throw HorribleError("Long parsed but missing (you should not see this):" + args.back());
+        } else {
+            if(!detail::split_short(current, name, rest))
+                throw HorribleError("Short parsed but missing! You should not see this");
+        }
+
+        auto op_ptr = std::find_if(std::begin(options_), std::end(options_), [name, second_dash](const Option_p &opt) {
+            return second_dash ? opt->check_lname(name) : opt->check_sname(name);
+        });
+
+        // Option not found
+        if(op_ptr == std::end(options_)) {
+            // If a subcommand, try the master command
+            if(parent_ != nullptr && fallthrough_)
+                return parent_->_parse_arg(args, second_dash);
+            // Otherwise, add to missing
+            else {
+                args.pop_back();
+                missing_.emplace_back(current_type, current);
+                return;
+            }
+        }
+
+        args.pop_back();
+
+        // Get a reference to the pointer to make syntax bearable
+        Option_p &op = *op_ptr;
+
+        int num = op->get_items_expected();
+
+        // Make sure we always eat the minimum for unlimited vectors
+        int collected = 0;
+
+        // --this=value
+        if(!value.empty()) {
+            // If exact number expected
+            if(num > 0)
+                num--;
+            op->add_result(value);
+            parse_order_.push_back(op.get());
+            collected += 1;
+        } else if(num == 0) {
+            op->add_result("");
+            parse_order_.push_back(op.get());
+            // -Trest
+        } else if(!rest.empty()) {
+            if(num > 0)
+                num--;
+            op->add_result(rest);
+            parse_order_.push_back(op.get());
+            rest = "";
+            collected += 1;
+        }
+
+        // Unlimited vector parser
+        if(num < 0) {
+            while(!args.empty() && _recognize(args.back()) == detail::Classifer::NONE) {
+                if(collected >= -num) {
+                    // We could break here for allow extras, but we don't
+
+                    // If any positionals remain, don't keep eating
+                    if(_count_remaining_positionals() > 0)
+                        break;
+                }
+                op->add_result(args.back());
+                parse_order_.push_back(op.get());
+                args.pop_back();
+                collected++;
+            }
+
+            // Allow -- to end an unlimited list and "eat" it
+            if(!args.empty() && _recognize(args.back()) == detail::Classifer::POSITIONAL_MARK)
+                args.pop_back();
+
+        } else {
+            while(num > 0 && !args.empty()) {
+                num--;
+                std::string current_ = args.back();
+                args.pop_back();
+                op->add_result(current_);
+                parse_order_.push_back(op.get());
+            }
+
+            if(num > 0) {
+                throw ArgumentMismatch::TypedAtLeast(op->get_name(), num, op->get_type_name());
+            }
+        }
+
+        if(!rest.empty()) {
+            rest = "-" + rest;
+            args.push_back(rest);
+        }
+    }
+};
+
+namespace FailureMessage {
+
+/// Printout a clean, simple message on error (the default in CLI11 1.5+)
+inline std::string simple(const App *app, const Error &e) {
+    std::string header = std::string(e.what()) + "\n";
+    if(app->get_help_ptr() != nullptr)
+        header += "Run with " + app->get_help_ptr()->get_name() + " for more information.\n";
+    return header;
+}
+
+/// Printout the full help string on error (if this fn is set, the old default for CLI11)
+inline std::string help(const App *app, const Error &e) {
+    std::string header = std::string("ERROR: ") + e.get_name() + ": " + e.what() + "\n";
+    header += app->help();
+    return header;
+}
+
+} // namespace FailureMessage
+
+namespace detail {
+/// This class is simply to allow tests access to App's protected functions
+struct AppFriend {
+
+    /// Wrap _parse_short, perfectly forward arguments and return
+    template <typename... Args>
+    static auto parse_arg(App *app, Args &&... args) ->
+        typename std::result_of<decltype (&App::_parse_arg)(App, Args...)>::type {
+        return app->_parse_arg(std::forward<Args>(args)...);
+    }
+
+    /// Wrap _parse_subcommand, perfectly forward arguments and return
+    template <typename... Args>
+    static auto parse_subcommand(App *app, Args &&... args) ->
+        typename std::result_of<decltype (&App::_parse_subcommand)(App, Args...)>::type {
+        return app->_parse_subcommand(std::forward<Args>(args)...);
+    }
+};
+} // namespace detail
+
+} // namespace CLI
+
+// From CLI/Config.hpp:
+
+namespace CLI {
+
+inline std::string
+ConfigINI::to_config(const App *app, bool default_also, bool write_description, std::string prefix) const {
+    std::stringstream out;
+    for(const Option *opt : app->get_options({})) {
+
+        // Only process option with a long-name and configurable
+        if(!opt->get_lnames().empty() && opt->get_configurable()) {
+            std::string name = prefix + opt->get_lnames()[0];
+            std::string value;
+
+            // Non-flags
+            if(opt->get_type_size() != 0) {
+
+                // If the option was found on command line
+                if(opt->count() > 0)
+                    value = detail::ini_join(opt->results());
+
+                // If the option has a default and is requested by optional argument
+                else if(default_also && !opt->get_defaultval().empty())
+                    value = opt->get_defaultval();
+                // Flag, one passed
+            } else if(opt->count() == 1) {
+                value = "true";
+
+                // Flag, multiple passed
+            } else if(opt->count() > 1) {
+                value = std::to_string(opt->count());
+
+                // Flag, not present
+            } else if(opt->count() == 0 && default_also) {
+                value = "false";
+            }
+
+            if(!value.empty()) {
+                if(write_description && opt->has_description()) {
+                    if(static_cast<int>(out.tellp()) != 0) {
+                        out << std::endl;
+                    }
+                    out << "; " << detail::fix_newlines("; ", opt->get_description()) << std::endl;
+                }
+                out << name << "=" << value << std::endl;
+            }
+        }
+    }
+
+    for(const App *subcom : app->get_subcommands({}))
+        out << to_config(subcom, default_also, write_description, prefix + subcom->get_name() + ".");
+
+    return out.str();
+}
+
+} // namespace CLI
+
+// From CLI/Formatter.hpp:
+
+namespace CLI {
+
+inline std::string
+Formatter::make_group(std::string group, bool is_positional, std::vector<const Option *> opts) const {
+    std::stringstream out;
+
+    out << "\n" << group << ":\n";
+    for(const Option *opt : opts) {
+        out << make_option(opt, is_positional);
+    }
+
+    return out.str();
+}
+
+inline std::string Formatter::make_positionals(const App *app) const {
+    std::vector<const Option *> opts =
+        app->get_options([](const Option *opt) { return !opt->get_group().empty() && opt->get_positional(); });
+
+    if(opts.empty())
+        return std::string();
+    else
+        return make_group(get_label("Positionals"), true, opts);
+}
+
+inline std::string Formatter::make_groups(const App *app, AppFormatMode mode) const {
+    std::stringstream out;
+    std::vector<std::string> groups = app->get_groups();
+
+    // Options
+    for(const std::string &group : groups) {
+        std::vector<const Option *> opts = app->get_options([app, mode, &group](const Option *opt) {
+            return opt->get_group() == group                    // Must be in the right group
+                   && opt->nonpositional()                      // Must not be a positional
+                   && (mode != AppFormatMode::Sub               // If mode is Sub, then
+                       || (app->get_help_ptr() != opt           // Ignore help pointer
+                           && app->get_help_all_ptr() != opt)); // Ignore help all pointer
+        });
+        if(!group.empty() && !opts.empty()) {
+            out << make_group(group, false, opts);
+
+            if(group != groups.back())
+                out << "\n";
+        }
+    }
+
+    return out.str();
+}
+
+inline std::string Formatter::make_description(const App *app) const {
+    std::string desc = app->get_description();
+
+    if(!desc.empty())
+        return desc + "\n";
+    else
+        return "";
+}
+
+inline std::string Formatter::make_usage(const App *app, std::string name) const {
+    std::stringstream out;
+
+    out << get_label("Usage") << ":" << (name.empty() ? "" : " ") << name;
+
+    std::vector<std::string> groups = app->get_groups();
+
+    // Print an Options badge if any options exist
+    std::vector<const Option *> non_pos_options =
+        app->get_options([](const Option *opt) { return opt->nonpositional(); });
+    if(!non_pos_options.empty())
+        out << " [" << get_label("OPTIONS") << "]";
+
+    // Positionals need to be listed here
+    std::vector<const Option *> positionals = app->get_options([](const Option *opt) { return opt->get_positional(); });
+
+    // Print out positionals if any are left
+    if(!positionals.empty()) {
+        // Convert to help names
+        std::vector<std::string> positional_names(positionals.size());
+        std::transform(positionals.begin(), positionals.end(), positional_names.begin(), [this](const Option *opt) {
+            return make_option_usage(opt);
+        });
+
+        out << " " << detail::join(positional_names, " ");
+    }
+
+    // Add a marker if subcommands are expected or optional
+    if(!app->get_subcommands({}).empty()) {
+        out << " " << (app->get_require_subcommand_min() == 0 ? "[" : "")
+            << get_label(app->get_require_subcommand_max() < 2 || app->get_require_subcommand_min() > 1 ? "SUBCOMMAND"
+                                                                                                        : "SUBCOMMANDS")
+            << (app->get_require_subcommand_min() == 0 ? "]" : "");
+    }
+
+    out << std::endl;
+
+    return out.str();
+}
+
+inline std::string Formatter::make_footer(const App *app) const {
+    std::string footer = app->get_footer();
+    if(!footer.empty())
+        return footer + "\n";
+    else
+        return "";
+}
+
+inline std::string Formatter::make_help(const App *app, std::string name, AppFormatMode mode) const {
+
+    // This immediatly forwards to the make_expanded method. This is done this way so that subcommands can
+    // have overridden formatters
+    if(mode == AppFormatMode::Sub)
+        return make_expanded(app);
+
+    std::stringstream out;
+
+    out << make_description(app);
+    out << make_usage(app, name);
+    out << make_positionals(app);
+    out << make_groups(app, mode);
+    out << make_subcommands(app, mode);
+    out << make_footer(app);
+
+    return out.str();
+}
+
+inline std::string Formatter::make_subcommands(const App *app, AppFormatMode mode) const {
+    std::stringstream out;
+
+    std::vector<const App *> subcommands = app->get_subcommands({});
+
+    // Make a list in definition order of the groups seen
+    std::vector<std::string> subcmd_groups_seen;
+    for(const App *com : subcommands) {
+        std::string group_key = com->get_group();
+        if(!group_key.empty() &&
+           std::find_if(subcmd_groups_seen.begin(), subcmd_groups_seen.end(), [&group_key](std::string a) {
+               return detail::to_lower(a) == detail::to_lower(group_key);
+           }) == subcmd_groups_seen.end())
+            subcmd_groups_seen.push_back(group_key);
+    }
+
+    // For each group, filter out and print subcommands
+    for(const std::string &group : subcmd_groups_seen) {
+        out << "\n" << group << ":\n";
+        std::vector<const App *> subcommands_group = app->get_subcommands(
+            [&group](const App *app) { return detail::to_lower(app->get_group()) == detail::to_lower(group); });
+        for(const App *new_com : subcommands_group) {
+            if(mode != AppFormatMode::All) {
+                out << make_subcommand(new_com);
+            } else {
+                out << new_com->help(new_com->get_name(), AppFormatMode::Sub);
+                out << "\n";
+            }
+        }
+    }
+
+    return out.str();
+}
+
+inline std::string Formatter::make_subcommand(const App *sub) const {
+    std::stringstream out;
+    detail::format_help(out, sub->get_name(), sub->get_description(), column_width_);
+    return out.str();
+}
+
+inline std::string Formatter::make_expanded(const App *sub) const {
+    std::stringstream out;
+    out << sub->get_name() << "\n";
+
+    out << make_description(sub);
+    out << make_positionals(sub);
+    out << make_groups(sub, AppFormatMode::Sub);
+    out << make_subcommands(sub, AppFormatMode::Sub);
+
+    // Drop blank spaces
+    std::string tmp = detail::find_and_replace(out.str(), "\n\n", "\n");
+    tmp = tmp.substr(0, tmp.size() - 1); // Remove the final '\n'
+
+    // Indent all but the first line (the name)
+    return detail::find_and_replace(tmp, "\n", "\n  ") + "\n";
+}
+
+inline std::string Formatter::make_option_name(const Option *opt, bool is_positional) const {
+    if(is_positional)
+        return opt->get_name(true, false);
+    else
+        return opt->get_name(false, true);
+}
+
+inline std::string Formatter::make_option_opts(const Option *opt) const {
+    std::stringstream out;
+
+    if(opt->get_type_size() != 0) {
+        if(!opt->get_type_name().empty())
+            out << " " << get_label(opt->get_type_name());
+        if(!opt->get_defaultval().empty())
+            out << "=" << opt->get_defaultval();
+        if(opt->get_expected() > 1)
+            out << " x " << opt->get_expected();
+        if(opt->get_expected() == -1)
+            out << " ...";
+        if(opt->get_required())
+            out << " " << get_label("REQUIRED");
+    }
+    if(!opt->get_envname().empty())
+        out << " (" << get_label("Env") << ":" << opt->get_envname() << ")";
+    if(!opt->get_needs().empty()) {
+        out << " " << get_label("Needs") << ":";
+        for(const Option *op : opt->get_needs())
+            out << " " << op->get_name();
+    }
+    if(!opt->get_excludes().empty()) {
+        out << " " << get_label("Excludes") << ":";
+        for(const Option *op : opt->get_excludes())
+            out << " " << op->get_name();
+    }
+    return out.str();
+}
+
+inline std::string Formatter::make_option_desc(const Option *opt) const { return opt->get_description(); }
+
+inline std::string Formatter::make_option_usage(const Option *opt) const {
+    // Note that these are positionals usages
+    std::stringstream out;
+    out << make_option_name(opt, true);
+
+    if(opt->get_expected() > 1)
+        out << "(" << std::to_string(opt->get_expected()) << "x)";
+    else if(opt->get_expected() < 0)
+        out << "...";
+
+    return opt->get_required() ? out.str() : "[" + out.str() + "]";
+}
+
+} // namespace CLI
+