2020-04-26 20:47:41 +00:00
|
|
|
#pragma once
|
2020-03-22 18:25:47 +00:00
|
|
|
|
2022-03-03 09:50:35 +00:00
|
|
|
#include "suggestions.hh"
|
2020-04-26 20:47:41 +00:00
|
|
|
#include "ref.hh"
|
2020-04-03 14:48:20 +00:00
|
|
|
#include "types.hh"
|
2020-07-07 12:37:47 +00:00
|
|
|
#include "fmt.hh"
|
2020-03-27 16:55:09 +00:00
|
|
|
|
2020-06-15 23:35:07 +00:00
|
|
|
#include <cstring>
|
2020-04-26 20:47:41 +00:00
|
|
|
#include <list>
|
|
|
|
#include <memory>
|
|
|
|
#include <map>
|
|
|
|
#include <optional>
|
2020-03-22 18:25:47 +00:00
|
|
|
|
2020-07-07 12:37:47 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
2020-03-22 18:25:47 +00:00
|
|
|
|
2020-04-26 20:47:41 +00:00
|
|
|
/* Before 4.7, gcc's std::exception uses empty throw() specifiers for
|
|
|
|
* its (virtual) destructor and what() in c++11 mode, in violation of spec
|
|
|
|
*/
|
|
|
|
#ifdef __GNUC__
|
|
|
|
#if __GNUC__ < 4 || (__GNUC__ == 4 && __GNUC_MINOR__ < 7)
|
|
|
|
#define EXCEPTION_NEEDS_THROW_SPEC
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
2020-06-15 12:06:58 +00:00
|
|
|
/*
|
2020-06-04 17:53:19 +00:00
|
|
|
|
2020-06-18 21:25:26 +00:00
|
|
|
This file defines two main structs/classes used in nix error handling.
|
2020-06-04 17:53:19 +00:00
|
|
|
|
2020-06-18 21:25:26 +00:00
|
|
|
ErrorInfo provides a standard payload of error information, with conversion to string
|
|
|
|
happening in the logger rather than at the call site.
|
2020-06-04 17:53:19 +00:00
|
|
|
|
2020-06-18 21:25:26 +00:00
|
|
|
BaseError is the ancestor of nix specific exceptions (and Interrupted), and contains
|
|
|
|
an ErrorInfo.
|
2020-06-04 17:53:19 +00:00
|
|
|
|
2020-06-18 21:25:26 +00:00
|
|
|
ErrorInfo structs are sent to the logger as part of an exception, or directly with the
|
|
|
|
logError or logWarning macros.
|
2020-03-24 17:21:35 +00:00
|
|
|
|
2020-11-11 16:21:26 +00:00
|
|
|
See libutil/tests/logging.cc for usage examples.
|
2020-06-04 17:53:19 +00:00
|
|
|
|
2020-06-18 21:25:26 +00:00
|
|
|
*/
|
2020-06-04 17:53:19 +00:00
|
|
|
|
2020-04-29 03:06:08 +00:00
|
|
|
typedef enum {
|
2020-04-26 20:47:41 +00:00
|
|
|
lvlError = 0,
|
|
|
|
lvlWarn,
|
2020-11-27 10:19:36 +00:00
|
|
|
lvlNotice,
|
2020-04-26 20:47:41 +00:00
|
|
|
lvlInfo,
|
|
|
|
lvlTalkative,
|
|
|
|
lvlChatty,
|
|
|
|
lvlDebug,
|
|
|
|
lvlVomit
|
|
|
|
} Verbosity;
|
|
|
|
|
2020-06-24 14:33:53 +00:00
|
|
|
// the lines of code surrounding an error.
|
|
|
|
struct LinesOfCode {
|
2022-02-25 15:00:00 +00:00
|
|
|
std::optional<std::string> prevLineOfCode;
|
|
|
|
std::optional<std::string> errLineOfCode;
|
|
|
|
std::optional<std::string> nextLineOfCode;
|
2020-06-24 14:33:53 +00:00
|
|
|
};
|
|
|
|
|
2022-12-12 23:48:04 +00:00
|
|
|
/* An abstract type that represents a location in a source file. */
|
|
|
|
struct AbstractPos
|
|
|
|
{
|
|
|
|
uint32_t line = 0;
|
|
|
|
uint32_t column = 0;
|
2020-04-26 20:47:41 +00:00
|
|
|
|
2022-12-12 23:48:04 +00:00
|
|
|
/* Return the contents of the source file. */
|
|
|
|
virtual std::optional<std::string> getSource() const
|
|
|
|
{ return std::nullopt; };
|
2020-05-09 00:18:28 +00:00
|
|
|
|
2022-12-12 23:48:04 +00:00
|
|
|
virtual void print(std::ostream & out) const = 0;
|
2020-04-08 15:48:21 +00:00
|
|
|
|
2022-12-12 23:48:04 +00:00
|
|
|
std::optional<LinesOfCode> getCodeLines() const;
|
2020-04-08 15:48:21 +00:00
|
|
|
};
|
|
|
|
|
2022-12-12 23:48:04 +00:00
|
|
|
std::ostream & operator << (std::ostream & str, const AbstractPos & pos);
|
2022-04-29 17:27:38 +00:00
|
|
|
|
2021-09-13 17:57:25 +00:00
|
|
|
void printCodeLines(std::ostream & out,
|
2022-04-07 19:42:01 +00:00
|
|
|
const std::string & prefix,
|
2022-12-12 23:48:04 +00:00
|
|
|
const AbstractPos & errPos,
|
2021-09-13 17:57:25 +00:00
|
|
|
const LinesOfCode & loc);
|
|
|
|
|
2020-06-18 21:25:26 +00:00
|
|
|
struct Trace {
|
2022-12-12 23:48:04 +00:00
|
|
|
std::shared_ptr<AbstractPos> pos;
|
2020-06-18 21:25:26 +00:00
|
|
|
hintformat hint;
|
2022-10-17 01:05:02 +00:00
|
|
|
bool frame;
|
2020-03-24 17:21:35 +00:00
|
|
|
};
|
|
|
|
|
2020-04-29 03:06:08 +00:00
|
|
|
struct ErrorInfo {
|
2020-04-26 20:47:41 +00:00
|
|
|
Verbosity level;
|
2021-01-20 23:27:36 +00:00
|
|
|
hintformat msg;
|
2022-12-12 23:48:04 +00:00
|
|
|
std::shared_ptr<AbstractPos> errPos;
|
2020-06-18 21:25:26 +00:00
|
|
|
std::list<Trace> traces;
|
2020-03-24 17:21:35 +00:00
|
|
|
|
2022-03-03 09:50:35 +00:00
|
|
|
Suggestions suggestions;
|
|
|
|
|
2022-02-25 15:00:00 +00:00
|
|
|
static std::optional<std::string> programName;
|
2020-04-07 01:43:22 +00:00
|
|
|
};
|
2020-03-24 17:21:35 +00:00
|
|
|
|
2022-02-25 15:00:00 +00:00
|
|
|
std::ostream & showErrorInfo(std::ostream & out, const ErrorInfo & einfo, bool showTrace);
|
2020-03-24 17:21:35 +00:00
|
|
|
|
2020-04-26 20:47:41 +00:00
|
|
|
/* BaseError should generally not be caught, as it has Interrupted as
|
|
|
|
a subclass. Catch Error instead. */
|
|
|
|
class BaseError : public std::exception
|
2020-04-07 01:43:22 +00:00
|
|
|
{
|
2020-04-26 20:47:41 +00:00
|
|
|
protected:
|
2020-05-07 22:43:36 +00:00
|
|
|
mutable ErrorInfo err;
|
2020-04-30 00:57:05 +00:00
|
|
|
|
2022-02-25 15:00:00 +00:00
|
|
|
mutable std::optional<std::string> what_;
|
|
|
|
const std::string & calcWhat() const;
|
2020-06-15 12:06:58 +00:00
|
|
|
|
2020-04-07 01:43:22 +00:00
|
|
|
public:
|
2020-04-26 20:47:41 +00:00
|
|
|
unsigned int status = 1; // exit status
|
|
|
|
|
2022-03-17 23:58:09 +00:00
|
|
|
BaseError(const BaseError &) = default;
|
|
|
|
|
2020-04-26 20:47:41 +00:00
|
|
|
template<typename... Args>
|
|
|
|
BaseError(unsigned int status, const Args & ... args)
|
2021-01-20 23:27:36 +00:00
|
|
|
: err { .level = lvlError, .msg = hintfmt(args...) }
|
2020-04-26 20:47:41 +00:00
|
|
|
, status(status)
|
2020-04-30 00:57:05 +00:00
|
|
|
{ }
|
2020-04-26 20:47:41 +00:00
|
|
|
|
|
|
|
template<typename... Args>
|
2022-01-17 18:28:42 +00:00
|
|
|
explicit BaseError(const std::string & fs, const Args & ... args)
|
2021-01-20 23:27:36 +00:00
|
|
|
: err { .level = lvlError, .msg = hintfmt(fs, args...) }
|
2020-04-30 00:57:05 +00:00
|
|
|
{ }
|
2020-04-26 20:47:41 +00:00
|
|
|
|
2022-03-03 09:50:35 +00:00
|
|
|
template<typename... Args>
|
|
|
|
BaseError(const Suggestions & sug, const Args & ... args)
|
|
|
|
: err { .level = lvlError, .msg = hintfmt(args...), .suggestions = sug }
|
|
|
|
{ }
|
|
|
|
|
2020-05-03 14:01:25 +00:00
|
|
|
BaseError(hintformat hint)
|
2021-01-20 23:27:36 +00:00
|
|
|
: err { .level = lvlError, .msg = hint }
|
2020-05-03 14:01:25 +00:00
|
|
|
{ }
|
|
|
|
|
2020-06-15 12:06:58 +00:00
|
|
|
BaseError(ErrorInfo && e)
|
|
|
|
: err(std::move(e))
|
|
|
|
{ }
|
|
|
|
|
|
|
|
BaseError(const ErrorInfo & e)
|
2020-04-26 20:47:41 +00:00
|
|
|
: err(e)
|
2020-04-30 00:57:05 +00:00
|
|
|
{ }
|
2020-04-29 03:06:08 +00:00
|
|
|
|
2020-04-26 20:47:41 +00:00
|
|
|
#ifdef EXCEPTION_NEEDS_THROW_SPEC
|
|
|
|
~BaseError() throw () { };
|
2020-05-07 22:43:36 +00:00
|
|
|
const char * what() const throw () { return calcWhat().c_str(); }
|
2020-04-26 20:47:41 +00:00
|
|
|
#else
|
2020-05-07 22:43:36 +00:00
|
|
|
const char * what() const noexcept override { return calcWhat().c_str(); }
|
2020-04-26 20:47:41 +00:00
|
|
|
#endif
|
2020-03-24 20:24:57 +00:00
|
|
|
|
2022-02-25 15:00:00 +00:00
|
|
|
const std::string & msg() const { return calcWhat(); }
|
2020-10-07 14:33:19 +00:00
|
|
|
const ErrorInfo & info() const { calcWhat(); return err; }
|
2020-06-24 19:10:41 +00:00
|
|
|
|
2023-01-02 19:53:39 +00:00
|
|
|
void pushTrace(Trace trace)
|
|
|
|
{
|
2022-03-17 23:58:09 +00:00
|
|
|
err.traces.push_front(trace);
|
|
|
|
}
|
|
|
|
|
2020-06-24 19:46:25 +00:00
|
|
|
template<typename... Args>
|
2023-01-02 19:53:39 +00:00
|
|
|
void addTrace(std::shared_ptr<AbstractPos> && e, std::string_view fs, const Args & ... args)
|
2020-06-24 19:46:25 +00:00
|
|
|
{
|
2023-01-02 19:53:39 +00:00
|
|
|
addTrace(std::move(e), hintfmt(std::string(fs), args...));
|
2020-06-24 19:46:25 +00:00
|
|
|
}
|
|
|
|
|
2023-01-02 19:53:39 +00:00
|
|
|
void addTrace(std::shared_ptr<AbstractPos> && e, hintformat hint, bool frame = false);
|
2020-06-25 00:31:28 +00:00
|
|
|
|
|
|
|
bool hasTrace() const { return !err.traces.empty(); }
|
2022-10-24 23:46:10 +00:00
|
|
|
|
|
|
|
const ErrorInfo & info() { return err; };
|
2020-04-07 01:43:22 +00:00
|
|
|
};
|
|
|
|
|
2020-04-26 20:47:41 +00:00
|
|
|
#define MakeError(newClass, superClass) \
|
|
|
|
class newClass : public superClass \
|
|
|
|
{ \
|
|
|
|
public: \
|
|
|
|
using superClass::superClass; \
|
|
|
|
}
|
2020-04-08 15:07:58 +00:00
|
|
|
|
2020-04-26 20:47:41 +00:00
|
|
|
MakeError(Error, BaseError);
|
2020-06-17 08:26:52 +00:00
|
|
|
MakeError(UsageError, Error);
|
2020-07-20 18:13:37 +00:00
|
|
|
MakeError(UnimplementedError, Error);
|
2020-03-24 20:24:57 +00:00
|
|
|
|
2020-04-26 20:47:41 +00:00
|
|
|
class SysError : public Error
|
2020-04-02 22:02:40 +00:00
|
|
|
{
|
2020-04-26 20:47:41 +00:00
|
|
|
public:
|
|
|
|
int errNo;
|
2020-03-24 15:18:23 +00:00
|
|
|
|
2020-04-26 20:47:41 +00:00
|
|
|
template<typename... Args>
|
2022-07-19 10:49:33 +00:00
|
|
|
SysError(int errNo_, const Args & ... args)
|
2020-06-18 21:25:26 +00:00
|
|
|
: Error("")
|
2020-05-06 20:07:20 +00:00
|
|
|
{
|
2022-07-19 10:49:33 +00:00
|
|
|
errNo = errNo_;
|
2020-05-06 20:07:20 +00:00
|
|
|
auto hf = hintfmt(args...);
|
2021-01-20 23:27:36 +00:00
|
|
|
err.msg = hintfmt("%1%: %2%", normaltxt(hf.str()), strerror(errNo));
|
2020-05-06 20:07:20 +00:00
|
|
|
}
|
2022-07-19 10:49:33 +00:00
|
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
SysError(const Args & ... args)
|
|
|
|
: SysError(errno, args ...)
|
|
|
|
{
|
|
|
|
}
|
2020-04-26 20:47:41 +00:00
|
|
|
};
|
2020-03-27 16:03:02 +00:00
|
|
|
|
2020-03-22 18:25:47 +00:00
|
|
|
}
|