diff --git a/Environment/CMakeLists.txt b/Environment/CMakeLists.txt
index d5e69be6dd383bd5c1ce126331a59c66bbdb3be6..e90160a67a2c74c55c4a40fbad479cfe86d05416 100644
--- a/Environment/CMakeLists.txt
+++ b/Environment/CMakeLists.txt
@@ -23,6 +23,8 @@ set (
   ShowerAxis.h
   IMagneticFieldModel.h
   UniformMagneticField.h
+  IRefractiveIndexModel.h
+  UniformRefractiveIndex.h
   )
 
 set (
diff --git a/Environment/IRefractiveIndexModel.h b/Environment/IRefractiveIndexModel.h
new file mode 100644
index 0000000000000000000000000000000000000000..ed8944e670e8f8654242378edcfdeb0e9b6a8120
--- /dev/null
+++ b/Environment/IRefractiveIndexModel.h
@@ -0,0 +1,45 @@
+/*
+ * (c) Copyright 2018 CORSIKA Project, corsika-project@lists.kit.edu
+ *
+ * See file AUTHORS for a list of contributors.
+ *
+ * This software is distributed under the terms of the GNU General Public
+ * Licence version 3 (GPL Version 3). See file LICENSE for a full version of
+ * the license.
+ */
+#pragma once
+
+#include <corsika/geometry/Point.h>
+#include <corsika/units/PhysicalUnits.h>
+
+namespace corsika::environment {
+
+  /**
+   * An interface for refractive index models.
+   *
+   * This is the base interface for refractive index mixins.
+   *
+   */
+  template <typename Model>
+  class IRefractiveIndexModel : public Model {
+
+    // a type-alias for a dimensionless refractive index
+    using RefractiveIndex = corsika::units::si::RefractiveIndexType;
+
+  public:
+    /**
+     * Evaluate the refractive index at a given location.
+     *
+     * @param  point    The location to evaluate at.
+     * @returns    The refractive index at this point.
+     */
+    virtual RefractiveIndex GetRefractiveIndex(corsika::geometry::Point const&) const = 0;
+
+    /**
+     * A virtual default destructor.
+     */
+    virtual ~IRefractiveIndexModel() = default;
+
+  }; // END: class IRefractiveIndexModel
+
+} // namespace corsika::environment
diff --git a/Environment/UniformRefractiveIndex.h b/Environment/UniformRefractiveIndex.h
new file mode 100644
index 0000000000000000000000000000000000000000..a64f07c4f3a33293ff9b922b949deb2fd72292c2
--- /dev/null
+++ b/Environment/UniformRefractiveIndex.h
@@ -0,0 +1,66 @@
+/*
+ * (c) Copyright 2018 CORSIKA Project, corsika-project@lists.kit.edu
+ *
+ * See file AUTHORS for a list of contributors.
+ *
+ * This software is distributed under the terms of the GNU General Public
+ * Licence version 3 (GPL Version 3). See file LICENSE for a full version of
+ * the license.
+ */
+#pragma once
+
+#include <corsika/environment/IRefractiveIndexModel.h>
+
+namespace corsika::environment {
+
+  /**
+   * A uniform refractive index.
+   *
+   * This class returns the same refractive index
+   * for all evaluated locations.
+   *
+   */
+  template <typename T>
+  class UniformRefractiveIndex : public T {
+
+    // a type-alias for a dimensionless refractive index
+    using RefractiveIndex = corsika::units::si::RefractiveIndexType;
+
+    RefractiveIndex n_; ///< The constant refractive index that we use.
+
+  public:
+    /**
+     * Construct a UniformRefractiveIndex.
+     *
+     * This is initialized with a fixed refractive index
+     * and returns this refractive index at all locations.
+     *
+     * @param field    The refractive index to return everywhere.
+     */
+    template <typename... Args>
+    UniformRefractiveIndex(RefractiveIndex const n, Args&&... args)
+        : T(std::forward<Args>(args)...)
+        , n_(n) {}
+
+    /**
+     * Evaluate the refractive index at a given location.
+     *
+     * @param  point    The location to evaluate at.
+     * @returns    The refractive index at this point.
+     */
+    RefractiveIndex GetRefractiveIndex(
+        corsika::geometry::Point const&) const final override {
+      return n_;
+    }
+
+    /**
+     * Set the refractive index returned by this instance.
+     *
+     * @param  point    The location to evaluate at.
+     * @returns    The refractive index at this location.
+     */
+    void SetRefractiveIndex(RefractiveIndex const& n) { n_ = n; }
+
+  }; // END: class RefractiveIndex
+
+} // namespace corsika::environment
diff --git a/Environment/testEnvironment.cc b/Environment/testEnvironment.cc
index 55ef5fba4f2473e6e59141953c1b191d2ceadaec..b30e5ffd42e283a14475dd708934372bc9a9df31 100644
--- a/Environment/testEnvironment.cc
+++ b/Environment/testEnvironment.cc
@@ -13,12 +13,17 @@
 #include <corsika/environment/HomogeneousMedium.h>
 #include <corsika/environment/IMagneticFieldModel.h>
 #include <corsika/environment/IMediumModel.h>
+#include <corsika/environment/IRefractiveIndexModel.h>
 #include <corsika/environment/InhomogeneousMedium.h>
 #include <corsika/environment/LayeredSphericalAtmosphereBuilder.h>
 #include <corsika/environment/LinearApproximationIntegrator.h>
 #include <corsika/environment/NuclearComposition.h>
 #include <corsika/environment/SlidingPlanarExponential.h>
+<<<<<<< HEAD
 #include <corsika/environment/UniformMagneticField.h>
+=======
+#include <corsika/environment/UniformRefractiveIndex.h>
+>>>>>>> a9fca8f... Initial refractive index interface and models with unittests.
 #include <corsika/environment/VolumeTreeNode.h>
 #include <corsika/geometry/Line.h>
 #include <corsika/geometry/RootCoordinateSystem.h>
@@ -249,8 +254,14 @@ TEST_CASE("UniformMagneticField w/ Homogeneous Medium") {
   const auto density{19.2_g / cube(1_cm)};
 
   // create our atmospheric model
+<<<<<<< HEAD
   AtmModel medium(B0, density, protonComposition);
 
+  // and test at several locations
+  REQUIRE(B0 == medium.GetMagneticField(Point(gCS, -10_m, 4_m, 35_km)));
+  REQUIRE(B0 == medium.GetMagneticField(Point(gCS, 1000_km, -1000_km, 1000_km)));
+  REQUIRE(B0 == medium.GetMagneticField(Point(gCS, 0_m, 0_m, 0_m)));
+
   // create a new magnetic field vector
   QuantityVector B1(23_T, 57_T, -4_T);
 
@@ -279,4 +290,99 @@ TEST_CASE("UniformMagneticField w/ Homogeneous Medium") {
   // and check the integrated grammage
   REQUIRE((medium.IntegratedGrammage(trajectory, 3_m) / (density * 3_m)) == Approx(1));
   REQUIRE((medium.ArclengthFromGrammage(trajectory, density * 5_m) / 5_m) == Approx(1));
+=======
+  AtmModel medium(n, 19.2_g / cube(1_cm), protonComposition);
+
+  // and require that it is constant
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, -10_m, 4_m, 35_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, +210_m, 0_m, 7_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, 0_m, 0_m, 0_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, 100_km, 400_km, 350_km)));
+
+  // a new refractive index
+  RefractiveIndexType n2 = 2.3472123__;
+
+  // update the refractive index of this atmospheric model
+  medium.SetRefractiveIndex(n2);
+
+  // check that the returned refractive index is correct
+  REQUIRE(n2 == medium.GetRefractiveIndex(Point(gCS, -10_m, 4_m, 35_km)));
+  REQUIRE(n2 == medium.GetRefractiveIndex(Point(gCS, +210_m, 0_m, 7_km)));
+  REQUIRE(n2 == medium.GetRefractiveIndex(Point(gCS, 0_m, 0_m, 0_km)));
+  REQUIRE(n2 == medium.GetRefractiveIndex(Point(gCS, 100_km, 400_km, 350_km)));
+>>>>>>> 12a0be6... Add test for SetRefractiveIndex
+}
+
+TEST_CASE("UniformRefractiveIndex w/ FlatExponential") {
+
+  // setup our interface types
+  using IModelInterface = IRefractiveIndexModel<IMediumModel>;
+  using AtmModel = UniformRefractiveIndex<FlatExponential<IModelInterface>>;
+
+  // the composition we use for the homogenous medium
+  NuclearComposition const protonComposition(std::vector<Code>{Code::Proton},
+                                             std::vector<float>{1.f});
+
+  // define our axis vector
+  Vector const axis(gCS, QuantityVector<dimensionless_d>(0, 0, 1));
+
+  // the parameters of our exponential model
+  LengthType const lambda = 3_m;
+  auto const rho0 = 1_g / units::si::detail::static_pow<3>(1_cm);
+
+  // the refractive index we want returned everywhere
+  RefractiveIndexType n = 1.000327__;
+
+  // create our atmospheric model
+  AtmModel medium(n, gOrigin, axis, rho0, lambda, protonComposition);
+
+  // check that the returned refractive index is correct
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, -10_m, 4_m, 35_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, +210_m, 0_m, 7_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, 0_m, 0_m, 0_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, 100_km, 400_km, 350_km)));
+
+<<<<<<< HEAD
+  // the refractive index we want returned everywhere
+  RefractiveIndexType n = 1.1234__;
+
+  // create our atmospheric model
+  AtmModel const medium(n, 19.2_g / cube(1_cm), protonComposition);
+
+  // and require that it is constant
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, -10_m, 4_m, 35_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, +210_m, 0_m, 7_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, 0_m, 0_m, 0_km)));
+  REQUIRE(n == medium.GetRefractiveIndex(Point(gCS, 100_km, 400_km, 350_km)));
+
+  // check the density and nuclear composition
+  REQUIRE(density == medium.GetMassDensity(Point(gCS, 0_m, 0_m, 0_m)));
+  medium.GetNuclearComposition();
+
+  // create a line of length 1 m
+  Line const line(gOrigin, Vector<SpeedType::dimension_type>(
+                               gCS, {1_m / second, 0_m / second, 0_m / second}));
+
+  // the end time of our line
+  auto const tEnd = 1_s;
+
+  // and the associated trajectory
+  Trajectory<Line> const trajectory(line, tEnd);
+
+  // and check the integrated grammage
+  REQUIRE((medium.IntegratedGrammage(trajectory, 3_m) / (density * 3_m)) == Approx(1));
+  REQUIRE((medium.ArclengthFromGrammage(trajectory, density * 5_m) / 5_m) == Approx(1));
+=======
+  // a new refractive index
+  RefractiveIndexType n2 = 1.123456__;
+
+  // update the refractive index of this atmospheric model
+  medium.SetRefractiveIndex(n2);
+
+  // check that the returned refractive index is correct
+  REQUIRE(n2 == medium.GetRefractiveIndex(Point(gCS, -10_m, 4_m, 35_km)));
+  REQUIRE(n2 == medium.GetRefractiveIndex(Point(gCS, +210_m, 0_m, 7_km)));
+  REQUIRE(n2 == medium.GetRefractiveIndex(Point(gCS, 0_m, 0_m, 0_km)));
+  REQUIRE(n2 == medium.GetRefractiveIndex(Point(gCS, 100_km, 400_km, 350_km)));
+>>>>>>> 12a0be6... Add test for SetRefractiveIndex
 }
diff --git a/Framework/Units/PhysicalUnits.h b/Framework/Units/PhysicalUnits.h
index e701928f8cc38b3fca6e91a2de3bbee12e02861a..35d4cd7a1b16fb218bb305b1edfbbd21779ae660 100644
--- a/Framework/Units/PhysicalUnits.h
+++ b/Framework/Units/PhysicalUnits.h
@@ -69,6 +69,12 @@ namespace corsika::units::si {
       phys::units::quantity<phys::units::dimensions<0, 0, -1>, double>;
   using InverseGrammageType =
       phys::units::quantity<phys::units::dimensions<2, -1, 0>, double>;
+  using RefractiveIndexType = phys::units::quantity<dimensionless_d, double>;
+
+  // define a new literal to construct dimensionless types
+  // This is a 'double underscore' literal, So 3__ is a
+  // dimensionless quantity = 3
+  QUANTITY_DEFINE_LITERALS(_, dimensionless_d)
 
   namespace detail {
     template <int N, typename T>