IODA
Exception.h
Go to the documentation of this file.
1 #pragma once
2 /*
3  * (C) Copyright 2020-2021 UCAR
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  */
8 /*! @file Exception.h
9 * @brief IODA's error system
10 */
11 #include <exception>
12 #include <iostream>
13 #include <list>
14 #include <sstream>
15 #include <string>
16 #include <utility>
17 
18 #include "./defs.h"
19 #include "Misc/Options.h"
21 
22 namespace ioda {
23 /*! @brief The ioda exception class.
24 *
25 * @details
26 *
27 * IODA is used to ingest data from external sources (ioda-converters), provides the runtime
28 * interface for JEDI's data access, and is used to analyze files for offline diagnostics.
29 * As a result, ioda is perhaps the one jedi component that must build everywhere with minimal
30 * dependencies. We do not want to deploy the full stack in all situations, particularly in
31 * difficult-to-configure Python environments (Anaconda), so we do not want to use eckit
32 * exceptions in the core ioda-engines code.
33 *
34 * This class is the default exception class for ioda. All ioda errors should use this class.
35 *
36 * <h3>USAGE EXAMPLES:</h3>
37 *
38 * 1. To throw a regular error:
39 * ```cpp
40 * throw ioda::Exception();
41 * ```
42 *
43 * 2. To throw a basic error with a single-line message:
44 * ```cpp
45 * throw ioda::Exception("This is an error");
46 * ```
47 *
48 * 3. To throw an error with some data:
49 * ```cpp
50 * throw ioda::Exception().add<std::string>("Reason", "Some descriptive error goes here.")
51 * .add<int>("status-code", 42);
52 * ```
53 */
54 class IODA_DL Exception : virtual public std::exception {
56  mutable std::string emessage_;
57  void invalidate();
58  void add_source_location(const ::ioda::source_location& loc);
59 
60 public:
61  Exception(const ::ioda::source_location& loc = source_location::current(),
62  const Options& opts = Options{});
63  explicit Exception(const char* msg,
64  const ::ioda::source_location& loc = source_location::current(),
65  const Options& opts = Options{});
66  virtual ~Exception() noexcept;
67  /// \brief Print the error message.
68  /// \todo Make this truly noexcept. Should never throw unless a
69  /// memory corruption error has occurred, in which case the
70  /// program will terminate anyways.
71  virtual const char* what() const noexcept;
72 
73  /// Add a key-value pair to the error message.
74  template <class T>
75  Exception& add(const std::string& key, const T value) {
76  opts_.add<T>(key, value);
77  invalidate();
78  return *this;
79  }
80 };
81 
82 /// Convenience function for unwinding an exception stack.
83 IODA_DL void unwind_exception_stack(const std::exception& e, std::ostream& out = std::cerr,
84  int level = 0);
85 
86 } // namespace ioda
Quick and easy key-value container that stringifies all values.
The ioda exception class.
Definition: Exception.h:54
virtual ~Exception() noexcept
Options opts_
Definition: Exception.h:55
std::string emessage_
Definition: Exception.h:56
Quick and easy key-value container that stringifies all values.
Definition: Options.h:23
Options & add(const std::string &key, const T &value)
Adds an option. Throws if the same name already exists.
Definition: Options.h:73
Common preprocessor definitions used throughout IODA.
#define IODA_DL
A preprocessor tag that indicates that a symbol is to be exported/imported.
Definition: defs.h:110
IODA_DL void unwind_exception_stack(const std::exception &e, std::ostream &out=std::cerr, int level=0)
Convenience function for unwinding an exception stack.
Definition: Exception.cpp:48
Implements ioda::source_location.
static source_location current() noexcept
Determine current source location.