diff --git a/ProcessLib/ComponentTransport/ComponentTransportProcess.cpp b/ProcessLib/ComponentTransport/ComponentTransportProcess.cpp
index 6736b76e3505fc1952bf2e3ad68823fa890519ce..d29078e8d83a977eebd602e509a14572f2211d69 100644
--- a/ProcessLib/ComponentTransport/ComponentTransportProcess.cpp
+++ b/ProcessLib/ComponentTransport/ComponentTransportProcess.cpp
@@ -197,6 +197,31 @@ void ComponentTransportProcess::
         _local_assemblers, pv.getActiveElementIDs(), _coupled_solutions);
 }
 
+void ComponentTransportProcess::solveReactionEquation(
+    std::vector<GlobalVector*>& x, double const t, double const dt)
+{
+    if (!_chemical_solver_interface)
+    {
+        return;
+    }
+
+    // Sequential non-iterative approach applied here to perform water
+    // chemistry calculation followed by resolving component transport
+    // process.
+    // TODO: move into a global loop to consider both mass balance over
+    // space and localized chemical equilibrium between solutes.
+    BaseLib::RunTime time_phreeqc;
+    time_phreeqc.start();
+
+    _chemical_solver_interface->doWaterChemistryCalculation(
+        interpolateNodalValuesToIntegrationPoints(x), dt);
+
+    extrapolateIntegrationPointValuesToNodes(
+        t, _chemical_solver_interface->getIntPtProcessSolutions(), x);
+
+    INFO("[time] Phreeqc took {:g} s.", time_phreeqc.elapsed());
+}
+
 std::vector<GlobalVector>
 ComponentTransportProcess::interpolateNodalValuesToIntegrationPoints(
     std::vector<GlobalVector*> const& nodal_values_vectors) const
diff --git a/ProcessLib/ComponentTransport/ComponentTransportProcess.h b/ProcessLib/ComponentTransport/ComponentTransportProcess.h
index 873eed1e1124d3b714ca12b623253c269c2abae4..3f1a89caeeb29156705de2ffd0db72b681b15a72 100644
--- a/ProcessLib/ComponentTransport/ComponentTransportProcess.h
+++ b/ProcessLib/ComponentTransport/ComponentTransportProcess.h
@@ -124,6 +124,9 @@ public:
     void setCoupledTermForTheStaggeredSchemeToLocalAssemblers(
         int const process_id) override;
 
+    void solveReactionEquation(std::vector<GlobalVector*>& x, double const t,
+                               double const dt) override;
+
     std::vector<GlobalVector> interpolateNodalValuesToIntegrationPoints(
         std::vector<GlobalVector*> const& nodal_values_vectors) const override;
 
diff --git a/ProcessLib/Process.h b/ProcessLib/Process.h
index c01cf00daa59d94710ee74d8eb45e9dae55f4202..ce4db97ebb08036b044fc4e089f0266cebac760e 100644
--- a/ProcessLib/Process.h
+++ b/ProcessLib/Process.h
@@ -179,6 +179,11 @@ public:
         return Eigen::Vector3d{};
     }
 
+    virtual void solveReactionEquation(std::vector<GlobalVector*>& /*x*/,
+                                       double const /*t*/, double const /*dt*/)
+    {
+    }
+
 protected:
     NumLib::Extrapolator& getExtrapolator() const
     {
diff --git a/ProcessLib/TimeLoop.cpp b/ProcessLib/TimeLoop.cpp
index 3e2329718012b8dd0741377d6ea12183cc0093c4..b8114d76bcca4056cd57c2461d9ee78787395163 100644
--- a/ProcessLib/TimeLoop.cpp
+++ b/ProcessLib/TimeLoop.cpp
@@ -823,26 +823,9 @@ TimeLoop::solveCoupledEquationSystemsByStaggeredScheme(
             timestep_id, t);
     }
 
-    if (_chemical_solver_interface)
     {
-        // Sequential non-iterative approach applied here to perform water
-        // chemistry calculation followed by resolving component transport
-        // process.
-        // TODO: move into a global loop to consider both mass balance over
-        // space and localized chemical equilibrium between solutes.
-        BaseLib::RunTime time_phreeqc;
-        time_phreeqc.start();
-
         auto& pcs = _per_process_data[0]->process;
-        _chemical_solver_interface->doWaterChemistryCalculation(
-            pcs.interpolateNodalValuesToIntegrationPoints(_process_solutions),
-            dt);
-
-        pcs.extrapolateIntegrationPointValuesToNodes(
-            t, _chemical_solver_interface->getIntPtProcessSolutions(),
-            _process_solutions);
-
-        INFO("[time] Phreeqc took {:g} s.", time_phreeqc.elapsed());
+        pcs.solveReactionEquation(_process_solutions, t, dt);
     }
 
     return nonlinear_solver_status;