From 782e6066f48d5f25932fe7d4b844f82f29c6b748 Mon Sep 17 00:00:00 2001
From: "Jadebeck, Johann Fredrik" <j.jadebeck@fz-juelich.de>
Date: Fri, 12 Jul 2024 00:54:45 +0200
Subject: [PATCH] add simple example

---
 CLA.md                  |  66 ++++++++++++++++++++++
 examples/CMakeLists.txt |   1 +
 examples/Demo.cpp       | 121 ++++++++++++++++++++++++++++++++++++++++
 3 files changed, 188 insertions(+)
 create mode 100644 CLA.md
 create mode 100644 examples/Demo.cpp

diff --git a/CLA.md b/CLA.md
new file mode 100644
index 00000000..fe893f71
--- /dev/null
+++ b/CLA.md
@@ -0,0 +1,66 @@
+# `hops` CLA
+
+Thank you for your interest in contributing to Forschungszentrum Jülich GmbH - Institute for Bio- and Geosciences: IBG-1's `calibr8` project ("We" or "Us").
+
+The purpose of this contributor agreement is to clarify and document the rights granted by contributors to Us.
+To make this document effective, please follow the instructions below.
+
+# How to use this CLA
+
+If You are an employee and have created the Contribution as part of your employment, You need to have Your employers approval as a Legal Entity.
+If You do not own the Copyright in the entire work of authorship, any other author of the Contribution must also sign this.
+
+# Definitions
+
+**"You"** means the individual Copyright owner who Submits a Contribution to Us.
+
+**"Legal Entity"** means an entity that is not a natural person.
+
+**"Contribution"** means any original work of authorship, including any original modifications or additions to an existing work of authorship, Submitted by You to Us, in which You own the Copyright.
+
+**"Copyright"** means all rights protecting works of authorship, including copyright, moral and neighboring rights, as appropriate, for the full term of their existence.
+
+**"Material"** means the software or documentation made available by Us to third parties.
+When this Agreement covers more than one software project, the Material means the software or documentation to which the Contribution was Submitted.
+After You Submit the Contribution, it may be included in the Material.
+
+**"Submit"** means any act by which a Contribution is transferred to Us by You by means of tangible or intangible media, including but not limited to electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, Us, but excluding any transfer that is conspicuously marked or otherwise designated in writing by You as "Not a Contribution."
+
+**"Documentation"** means any non-software portion of a Contribution.
+
+# Rights and Obligations
+
+You hereby grant to Us a worldwide, royalty-free, non-exclusive, perpetual and irrevocable license, with the right to transfer an unlimited number of licenses or to grant sublicenses to third parties, under the Copyright covering the Contribution to use the Contribution by all means.
+
+We agree to (sub)license the Contribution or any Materials containing it, based on or derived from your Contribution non-exclusively under the terms of any licenses the Free Software Foundation classifies as Free Software License and which are approved by the Open Source Initiative as Open Source licenses.
+
+# Copyright
+
+You warrant, that you are legitimated to license the Contribution to us.
+
+# Acceptance of the CLA
+
+By submitting your Contribution via the [Repository](https://jugit.fz-juelich.de/IBG-1/ModSim/hops) you accept this agreement.
+
+
+# Liability
+
+Except in cases of willful misconduct or physical damage directly caused to natural persons, the Parties will not be liable for any direct or indirect, material or moral, damages of any kind, arising out of the Licence or of the use of the Material, including, without limitation, damages for loss of goodwill, work stoppage, computer failure or malfunction, loss of data or any commercial damage.
+However, the Licensor will be liable under statutory product liability laws as far such laws apply to the Material.
+
+# Warranty
+
+The Material is a work in progress, which is continuously being improved by numerous Contributors.
+It is not a finished work and may therefore contain defects or "bugs" inherent to this type of development.
+
+For the above reason, the Material is provided on an "as is" basis and without warranties of any kind concerning the Material, including without limitation,
+merchantability, fitness for a particular purpose, absence of defects or errors, accuracy, non-infringement of intellectual property rights other than copyright.
+
+# Miscellaneous
+
+This Agreement and all disputes, claims, actions, suits or other proceedings arising out of this agreement or relating in any way to it shall be governed by the laws of Germany excluding its private international law provisions.
+
+If any provision of this Agreement is found void and unenforceable, such provision will be replaced to the extent possible with a provision that comes closest to the meaning of the original provision and that is enforceable.
+The terms and conditions set forth in this Agreement shall apply notwithstanding any failure of essential purpose of this Agreement or any limited remedy to the maximum extent possible under law.
+
+You agree to notify Us of any facts or circumstances of which You become aware that would make this Agreement inaccurate in any respect.
diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt
index d21f6299..a751c3ce 100644
--- a/examples/CMakeLists.txt
+++ b/examples/CMakeLists.txt
@@ -5,6 +5,7 @@ if (NOT HOPS_LIBRARY_TYPE STREQUAL "HEADER_ONLY")
             SamplingGaussianTarget.cpp
             SamplingGaussianSignificanceTarget.cpp
             TruncatedGaussianDemo.cpp
+            Demo.cpp
     )
 
     if (HOPS_MPI)
diff --git a/examples/Demo.cpp b/examples/Demo.cpp
new file mode 100644
index 00000000..c1387c3a
--- /dev/null
+++ b/examples/Demo.cpp
@@ -0,0 +1,121 @@
+#include "hops/hops.hpp"
+#include <chrono>
+#include <numeric>
+#include <tuple>
+
+
+int main() {
+    const long numberOfSamples = 10'000;
+    const long thinning = 5;
+    const long numberOfCheckPoints = 5;
+    const std::size_t dims = 3;
+
+    double covScale = 1e-2;
+    Eigen::MatrixXd baseCov = covScale * Eigen::MatrixXd::Random(dims, dims);
+    // makes covariance positive definite
+    const Eigen::MatrixXd covariance = baseCov.transpose() * baseCov + covScale * Eigen::MatrixXd::Identity(dims, dims);
+    ;
+
+
+    Eigen::MatrixXd A(2 * dims, dims);
+    A << Eigen::MatrixXd::Identity(dims, dims), -Eigen::MatrixXd::Identity(dims, dims);
+
+    Eigen::VectorXd b(2 * dims);
+    b << 5 * Eigen::VectorXd::Ones(dims), Eigen::VectorXd::Zero(dims);
+
+
+    Eigen::VectorXd mean = Eigen::VectorXd::Zero(dims);
+
+    hops::Gaussian model(mean, covariance);
+    mean = Eigen::VectorXd::Ones(dims);
+
+    // Rounding the polytope using the maximum volue ellipsoid is recommended to deal with anistropy.
+    auto MVE = hops::MaximumVolumeEllipsoid<double>::construct(A, b, 100000);
+    auto startPointFinder = hops::LinearProgramFactory::createLinearProgram(A, b);
+    hops::LinearProgramSolution startPointSolution = startPointFinder->computeChebyshevCenter();
+    assert(startPointSolution.status == hops::LinearProgramStatus::OPTIMAL);
+    Eigen::VectorXd startPoint = startPointSolution.optimalParameters;
+    Eigen::MatrixXd roundingTrafo = MVE.getRoundingTransformation();
+    Eigen::VectorXd startPointRounded = roundingTrafo.template triangularView<Eigen::Lower>().solve(startPoint);
+    Eigen::MatrixXd Arounded = A * roundingTrafo;
+
+    hops::MarkovChainType chainType = hops::MarkovChainType::CoordinateHitAndRun;
+
+    auto gaussianSampler = hops::MarkovChainFactory::createMarkovChain(
+            chainType,
+            Arounded,
+            b,
+            startPointRounded,
+            roundingTrafo,
+            // we skipped shifting the polytopes chebyshev center to the origin in this example
+            Eigen::VectorXd(Eigen::VectorXd::Zero(roundingTrafo.cols())),
+            model);
+
+    auto uniformSampler = hops::MarkovChainFactory::createMarkovChain(
+            chainType,
+            Arounded,
+            b,
+            startPointRounded,
+            roundingTrafo,
+            // we skipped shifting the polytopes chebyshev center to the origin in this example
+            Eigen::VectorXd(Eigen::VectorXd::Zero(roundingTrafo.cols())));
+
+    hops::RandomNumberGenerator randomNumberGenerator((std::random_device()()));
+
+    std::vector<double> acceptanceRates;
+    std::vector<Eigen::VectorXd> states;
+    std::vector<long> timestamps;
+
+    acceptanceRates.reserve(numberOfSamples);
+    states.reserve(numberOfSamples);
+    timestamps.reserve(numberOfSamples);
+
+
+    auto uniform_results_writer = hops::FileWriterFactory::createFileWriter("uniform_sampling_results", hops::FileWriterType::CSV);
+    auto gaussian_results_writer = hops::FileWriterFactory::createFileWriter("gaussian_sampling_results", hops::FileWriterType::CSV);
+
+
+    // draws uniform samples
+    for (long checkPoint = 0; checkPoint < numberOfCheckPoints; ++checkPoint) {
+        for (long i = 0; i < numberOfSamples / numberOfCheckPoints; ++i) {
+            ABORTABLE
+            auto [acceptanceRate, state] = uniformSampler->draw(randomNumberGenerator, thinning);
+            acceptanceRates.emplace_back(acceptanceRate);
+            states.emplace_back(state);
+            timestamps.emplace_back(std::chrono::duration_cast<std::chrono::milliseconds>(
+                                            std::chrono::high_resolution_clock::now().time_since_epoch())
+                                            .count());
+        }
+
+        uniform_results_writer->write("states", states);
+        uniform_results_writer->write("acceptance_rates", std::vector<double>{
+                                                                  std::reduce(acceptanceRates.begin(), acceptanceRates.end(), 0.) / acceptanceRates.size()});
+        uniform_results_writer->write("timestamps", timestamps);
+
+        states.clear();
+        acceptanceRates.clear();
+        timestamps.clear();
+    }
+
+    // draws gaussian samples
+    for (long checkPoint = 0; checkPoint < numberOfCheckPoints; ++checkPoint) {
+        for (long i = 0; i < numberOfSamples / numberOfCheckPoints; ++i) {
+            ABORTABLE
+            auto [acceptanceRate, state] = gaussianSampler->draw(randomNumberGenerator, thinning);
+            acceptanceRates.emplace_back(acceptanceRate);
+            states.emplace_back(state);
+            timestamps.emplace_back(std::chrono::duration_cast<std::chrono::milliseconds>(
+                                            std::chrono::high_resolution_clock::now().time_since_epoch())
+                                            .count());
+        }
+
+        gaussian_results_writer->write("states", states);
+        gaussian_results_writer->write("acceptance_rates", std::vector<double>{
+                                                                   std::reduce(acceptanceRates.begin(), acceptanceRates.end(), 0.) / acceptanceRates.size()});
+        gaussian_results_writer->write("timestamps", timestamps);
+
+        states.clear();
+        acceptanceRates.clear();
+        timestamps.clear();
+    }
+}
-- 
GitLab