diff --git a/CMakeLists.txt b/CMakeLists.txt
index 882ae32ca531d634a2d90886f33d1cd544a5ec25..e1d017f06c817abfb615ed93d85af1d4a1f47da8 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -216,6 +216,7 @@ target_link_libraries (
   CONAN_PKG::boost
   CONAN_PKG::yaml-cpp
   CONAN_PKG::arrow
+  CONAN_PKG::cli11
   cnpy # for SaveBoostHistogram
   )
 
diff --git a/conanfile.txt b/conanfile.txt
index 3579442077988e68869ddb4634d9010f118fac4a..f02bb7b39738b61ccec1561cd778dc81312c71c7 100644
--- a/conanfile.txt
+++ b/conanfile.txt
@@ -7,6 +7,7 @@ zlib/1.2.11
 proposal/7.0.4
 yaml-cpp/0.6.3
 arrow/2.0.0
+cli11/1.9.1
 
 [build_requires]
 readline/8.0 # this is only needed to fix a missing dependency in "bison" which is pulled-in by arrow
diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt
index 2d20738b31793611db793fa90bf1cc1bca1b1dd5..7aa0ad850d41aad0cf6e19a84422f0b74845cf81 100644
--- a/examples/CMakeLists.txt
+++ b/examples/CMakeLists.txt
@@ -59,4 +59,3 @@ CORSIKA_REGISTER_EXAMPLE (hybrid_MC RUN_OPTIONS 4 2 10000.)
 
 add_executable (corsika corsika.cpp)
 target_link_libraries (corsika CORSIKA8::CORSIKA8)
-CORSIKA_REGISTER_EXAMPLE (corsika RUN_OPTIONS 4 2 10000. 1)
diff --git a/examples/corsika.cpp b/examples/corsika.cpp
index a67236dd5cd712cb2c4413f120fd5d0c8297f9fc..9bb4ee59244dd87a1b364876218c753ddef31064 100644
--- a/examples/corsika.cpp
+++ b/examples/corsika.cpp
@@ -56,6 +56,10 @@
 #include <corsika/setup/SetupStack.hpp>
 #include <corsika/setup/SetupTrajectory.hpp>
 
+#include <CLI/App.hpp>
+#include <CLI/Formatter.hpp>
+#include <CLI/Config.hpp>
+
 #include <iomanip>
 #include <iostream>
 #include <limits>
@@ -103,26 +107,74 @@ int main(int argc, char** argv) {
 
   logging::set_level(logging::level::info);
 
-  CORSIKA_LOG_INFO("corsika");
-
-  if (argc < 5) {
-    std::cerr << "usage: corsika <A> <Z> <energy/GeV> <Nevt> [seed] \n"
-                 "       if A=0, Z is interpreted as PDG code \n"
-                 "       if no seed is given, a random seed is chosen \n"
-              << std::endl;
-    return 1;
+  // the main command line description
+  CLI::App app{"Simulate standard (downgoing) showers with CORSIKA 8."};
+
+  // some options that we want to fill in
+  int A, Z, nevent = 0;
+
+  // the following section adds the options to the parser
+
+  // we start by definining a sub-group for the primary ID
+  auto opt_Z = app.add_option("-Z", Z, "Atomic number for primary")
+                   ->check(CLI::Range(0, 26))
+                   ->group("Primary");
+  auto opt_A = app.add_option("-A", A, "Atomic mass number for primary")
+                   ->needs(opt_Z)
+                   ->check(CLI::Range(1, 58))
+                   ->group("Primary");
+  app.add_option("-p,--pdg", "PDG code for primary.")
+      ->excludes(opt_A)
+      ->excludes(opt_Z)
+      ->group("Primary");
+  // the remainding options
+  app.add_option("-E,--energy", "Primary energy in GeV")
+      ->required()
+      ->check(CLI::PositiveNumber)
+      ->group("Primary");
+  app.add_option("-z,--zenith", "Primary zenith angle (deg)")
+      ->required()
+      ->default_val(0.)
+      ->check(CLI::Range(0, 90))
+      ->group("Primary");
+  app.add_option("-a,--azimuth", "Primary azimuth angle (deg)")
+      ->default_val(0.)
+      ->check(CLI::Range(0, 360))
+      ->group("Primary");
+  app.add_option("-N,--nevent", nevent, "The number of events/showers to run.")
+      ->required()
+      ->check(CLI::PositiveNumber)
+      ->group("Library/Output");
+  app.add_option("-f,--filename", "Filename for output library.")
+      ->required()
+      ->default_val("corsika_library")
+      ->check(CLI::NonexistentPath)
+      ->group("Library/Output");
+  app.add_option("-s,--seed", "The random number seed.")
+      ->default_val(12351739)
+      ->check(CLI::NonNegativeNumber)
+      ->group("Misc.");
+  app.add_flag("--force-interaction", "Force the location of the first interaction.")
+      ->group("Misc.");
+
+  // parse the command line options into the variables
+  CLI11_PARSE(app, argc, argv);
+
+  // check that we got either PDG or A/Z
+  // this can be done with option_groups but the ordering
+  // gets all messed up
+  if (app.count("--pdg") == 0) {
+    if ((app.count("-A") == 0) || (app.count("-Z") == 0)) {
+      std::cerr << "If --pdg is not provided, then both -A and -Z are required."
+                << std::endl;
+      return 1;
+    }
   }
-  feenableexcept(FE_INVALID);
-
-  int seed = 0;
-  int number_showers = std::stoi(std::string(argv[4]));
-
-  if (argc > 5) { seed = std::stoi(std::string(argv[5])); }
 
   // initialize random number sequence(s)
-  registerRandomStreams(seed);
+  registerRandomStreams(app["--seed"]->as<int>());
 
-  // setup environment, geometry
+  /* === START: SETUP ENVIRONMENT AND ROOT COORDINATE SYSTEM === */
   using EnvType = setup::Environment;
   EnvType env;
   CoordinateSystemPtr const& rootCS = env.getCoordinateSystem();
@@ -144,43 +196,43 @@ int main(int argc, char** argv) {
   builder.addExponentialLayer(540.1778_g / (1_cm * 1_cm), 772170.16_cm, 100_km);
   builder.addLinearLayer(1e9_cm, 112.8_km + constants::EarthRadius::Mean);
   builder.assemble(env);
+  /* === END: SETUP ENVIRONMENT AND ROOT COORDINATE SYSTEM === */
+
+  /* === START: CONSTRUCT PRIMARY PARTICLE === */
 
-  // pre-setup particle stack
-  unsigned short const A = std::stoi(std::string(argv[1]));
+  // parse the primary ID as a PDG or A/Z code
   Code beamCode;
   HEPEnergyType mass;
-  unsigned short Z = 0;
-  if (A > 0) {
-    beamCode = Code::Nucleus;
-    Z = std::stoi(std::string(argv[2]));
-    mass = get_nucleus_mass(A, Z);
-  } else {
-    int pdg = std::stoi(std::string(argv[2]));
-    beamCode = convert_from_PDG(PDGCode(pdg));
+
+  // check if we want to use a PDG code instead
+  if (app.count("--pdg") > 0) {
+    beamCode = convert_from_PDG(PDGCode(app["--pdg"]->as<int>()));
     mass = get_mass(beamCode);
+  } else {
+    // check manually for proton and neutrons
+    if ((A == 0) && (Z == 1)) beamCode = Code::Proton;
+    if ((A == 1) && (Z == 1)) beamCode = Code::Neutron;
+    mass = get_nucleus_mass(A, Z);
   }
-  HEPEnergyType const E0 = 1_GeV * std::stof(std::string(argv[3]));
-  double theta = 20.;
-  double phi = 180.;
-  auto const thetaRad = theta / 180. * M_PI;
-  auto const phiRad = phi / 180. * M_PI;
-
-  auto elab2plab = [](HEPEnergyType Elab, HEPMassType m) {
-    return sqrt((Elab - m) * (Elab + m));
-  };
-  HEPMomentumType P0 = elab2plab(E0, mass);
-  auto momentumComponents = [](double theta, double phi, HEPMomentumType ptot) {
-    return std::make_tuple(ptot * sin(theta) * cos(phi), ptot * sin(theta) * sin(phi),
-                           -ptot * cos(theta));
-  };
 
-  auto const [px, py, pz] = momentumComponents(thetaRad, phiRad, P0);
+  // particle energy
+  HEPEnergyType const E0 = 1_GeV * app["--energy"]->as<float>();
+
+  // direction of the shower in (theta, phi) space
+  auto const thetaRad = app["--zenith"]->as<float>() / 180. * M_PI;
+  auto const phiRad = app["--azimuth"]->as<float>() / 180. * M_PI;
+
+  // convert Elab to Plab
+  HEPMomentumType P0 = sqrt((E0 - mass) * (E0 + mass));
+
+  // convert the momentum to the zenith and azimuth angle of the primary
+  auto const [px, py, pz] =
+      std::make_tuple(P0 * sin(thetaRad) * cos(phiRad), P0 * sin(thetaRad) * sin(phiRad),
+                      -P0 * cos(thetaRad));
   auto plab = MomentumVector(rootCS, {px, py, pz});
-  cout << "input particle: " << beamCode << endl;
-  cout << "input angles: theta=" << theta << ",phi=" << phi << endl;
-  cout << "input momentum: " << plab.getComponents() / 1_GeV
-       << ", norm = " << plab.getNorm() << endl;
+  /* === END: CONSTRUCT PRIMARY PARTICLE === */
 
+  /* === START: CONSTRUCT GEOMETRY === */
   auto const observationHeight = 0_km + builder.getEarthRadius();
   auto const injectionHeight = 111.75_km + builder.getEarthRadius();
   auto const t = -observationHeight * cos(thetaRad) +
@@ -193,21 +245,15 @@ int main(int argc, char** argv) {
                                     -sin(thetaRad) * sin(phiRad), cos(thetaRad)}} *
                        t;
 
-  std::cout << "point of injection: " << injectionPos.getCoordinates() << std::endl;
-
   // we make the axis much longer than the inj-core distance since the
   // profile will go beyond the core, depending on zenith angle
-  std::cout << "shower axis length: " << (showerCore - injectionPos).getNorm() * 1.2
-            << std::endl;
-
   ShowerAxis const showerAxis{injectionPos, (showerCore - injectionPos) * 1.2, env};
+  /* === END: CONSTRUCT GEOMETRY === */
 
   // create the output manager that we then register outputs with
-  OutputManager output("corsika_outputs");
+  OutputManager output(app["--filename"]->as<std::string>());
 
-  // setup processes, decays and interactions
-
-  // corsika::qgsjetII::Interaction qgsjet;
+  /* === START: SETUP PROCESS LIST === */
   corsika::sibyll::Interaction sibyll;
   InteractionCounter sibyllCounted(sibyll);
 
@@ -261,27 +307,46 @@ int main(int argc, char** argv) {
                                     make_sequence(sibyllNucCounted, sibyllCounted));
   auto decaySequence = make_sequence(decayPythia, decaySibyll);
 
+  // track writer
   TrackWriter trackWriter;
   output.add("tracks", trackWriter); // register TrackWriter
 
+  // observation plane
   Plane const obsPlane(showerCore, DirectionVector(rootCS, {0., 0., 1.}));
   ObservationPlane observationLevel(obsPlane, DirectionVector(rootCS, {1., 0., 0.}));
-  // register the observation plane with the output
-  output.add("particles", observationLevel);
+  output.add("obslevel", observationLevel);
 
+  // assemble the final process sequence
   auto sequence =
       make_sequence(stackInspect, hadronSequence, decaySequence, emCascadeCounted,
                     emContinuous, cut, trackWriter, observationLevel, longprof);
+  /* === END: SETUP PROCESS LIST === */
 
-  // define air shower object, run simulation
+  // create the cascade object using the default stack and tracking implementation
   setup::Tracking tracking;
   setup::Stack stack;
   Cascade EAS(env, tracking, sequence, output, stack);
 
+  // print our primary parameters all in one place
+  if (app["--pdg"]->count() > 0) {
+    CORSIKA_LOG_INFO("Primary PDG ID: {}", app["--pdg"]->as<int>());
+  } else {
+    CORSIKA_LOG_INFO("Primary Z/A: {}/{}", Z, A);
+  }
+  CORSIKA_LOG_INFO("Primary Energy: {}", E0);
+  CORSIKA_LOG_INFO("Primary Momentum: {}", P0);
+  CORSIKA_LOG_INFO("Point of Injection: {}", injectionPos.getCoordinates());
+  CORSIKA_LOG_INFO("Shower Axis Length: {}", (showerCore - injectionPos).getNorm() * 1.2);
+
+  // trigger the output manager to open the library for writing
   output.startOfLibrary();
 
-  for (int i_shower = 1; i_shower < number_showers + 1; i_shower++) {
+  // loop over each shower
+  for (int i_shower = 1; i_shower < nevent + 1; i_shower++) {
+
+    CORSIKA_LOG_INFO("Shower {} / {} ", i_shower, nevent);
 
+    // trigger the start of the outputs for this shower
     output.startOfShower();
 
     // directory for outputs
@@ -289,32 +354,20 @@ int main(int argc, char** argv) {
     string const cMSHist_file = "inthist_cms_verticalEAS_" + to_string(i_shower) + ".npz";
     string const longprof_file = "longprof_verticalEAS_" + to_string(i_shower) + ".txt";
 
-    CORSIKA_LOG_INFO("Shower {} / {} ", i_shower, number_showers);
-
     // setup particle stack, and add primary particle
     stack.clear();
 
+    // add the desired particle to the stack
     if (A > 1) {
       stack.addParticle(std::make_tuple(beamCode, plab, injectionPos, 0_ns, A, Z));
-
     } else {
-      if (A == 1) {
-        if (Z == 1) {
-          stack.addParticle(std::make_tuple(Code::Proton, plab, injectionPos, 0_ns));
-        } else if (Z == 0) {
-          stack.addParticle(std::make_tuple(Code::Neutron, plab, injectionPos, 0_ns));
-        } else {
-          std::cerr << "illegal parameters" << std::endl;
-          return EXIT_FAILURE;
-        }
-      } else {
-        stack.addParticle(std::make_tuple(beamCode, plab, injectionPos, 0_ns));
-      }
+      stack.addParticle(std::make_tuple(beamCode, plab, injectionPos, 0_ns));
     }
 
-    // to fix the point of first interaction, uncomment the following two lines:
-    //  EAS.forceInteraction();
+    // if we want to fix the first location of the shower
+    if (app["--force-interaction"]) EAS.forceInteraction();
 
+    // run the shower
     EAS.run();
 
     cut.showResults();
@@ -335,7 +388,11 @@ int main(int argc, char** argv) {
     save_hist(hists.labHist(), labHist_file, true);
     save_hist(hists.CMSHist(), cMSHist_file, true);
     longprof.save(longprof_file);
+
+    // trigger the output manager to save this shower to disk
     output.endOfShower();
   }
+
+  // and finalize the output on disk
   output.endOfLibrary();
 }