2015-10-05 01:26:43 +00:00
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
#include <cerrno>
|
|
|
|
|
#include <vector>
|
2016-05-28 14:14:18 +00:00
|
|
|
|
#include <memory>
|
2015-10-05 01:26:43 +00:00
|
|
|
|
|
|
|
|
|
#include <Poco/Exception.h>
|
|
|
|
|
|
|
|
|
|
#include <DB/Common/StackTrace.h>
|
|
|
|
|
|
2015-10-05 01:31:57 +00:00
|
|
|
|
namespace Poco { class Logger; }
|
|
|
|
|
|
2015-10-05 01:26:43 +00:00
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
class Exception : public Poco::Exception
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
Exception(int code = 0) : Poco::Exception(code) {}
|
|
|
|
|
Exception(const std::string & msg, int code = 0) : Poco::Exception(msg, code) {}
|
|
|
|
|
Exception(const std::string & msg, const std::string & arg, int code = 0) : Poco::Exception(msg, arg, code) {}
|
|
|
|
|
Exception(const std::string & msg, const Exception & exc, int code = 0) : Poco::Exception(msg, exc, code), trace(exc.trace) {}
|
|
|
|
|
Exception(const Exception & exc) : Poco::Exception(exc), trace(exc.trace) {}
|
|
|
|
|
explicit Exception(const Poco::Exception & exc) : Poco::Exception(exc.displayText()) {}
|
|
|
|
|
~Exception() throw() override {}
|
|
|
|
|
Exception & operator = (const Exception & exc)
|
|
|
|
|
{
|
|
|
|
|
Poco::Exception::operator=(exc);
|
|
|
|
|
trace = exc.trace;
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
const char * name() const throw() override { return "DB::Exception"; }
|
|
|
|
|
const char * className() const throw() override { return "DB::Exception"; }
|
|
|
|
|
DB::Exception * clone() const override { return new DB::Exception(*this); }
|
|
|
|
|
void rethrow() const override { throw *this; }
|
|
|
|
|
|
|
|
|
|
/// Дописать к существующему сообщению что-нибудь ещё.
|
|
|
|
|
void addMessage(const std::string & arg) { extendedMessage(arg); }
|
|
|
|
|
|
|
|
|
|
const StackTrace & getStackTrace() const { return trace; }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
StackTrace trace;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// Содержит дополнительный член saved_errno. См. функцию throwFromErrno.
|
|
|
|
|
class ErrnoException : public Exception
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
ErrnoException(int code = 0, int saved_errno_ = 0)
|
|
|
|
|
: Exception(code), saved_errno(saved_errno_) {}
|
|
|
|
|
ErrnoException(const std::string & msg, int code = 0, int saved_errno_ = 0)
|
|
|
|
|
: Exception(msg, code), saved_errno(saved_errno_) {}
|
|
|
|
|
ErrnoException(const std::string & msg, const std::string & arg, int code = 0, int saved_errno_ = 0)
|
|
|
|
|
: Exception(msg, arg, code), saved_errno(saved_errno_) {}
|
|
|
|
|
ErrnoException(const std::string & msg, const Exception & exc, int code = 0, int saved_errno_ = 0)
|
|
|
|
|
: Exception(msg, exc, code), saved_errno(saved_errno_) {}
|
|
|
|
|
ErrnoException(const ErrnoException & exc)
|
|
|
|
|
: Exception(exc), saved_errno(exc.saved_errno) {}
|
|
|
|
|
|
|
|
|
|
int getErrno() const { return saved_errno; }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
int saved_errno;
|
|
|
|
|
};
|
|
|
|
|
|
2015-10-05 01:31:57 +00:00
|
|
|
|
|
2016-05-28 10:35:44 +00:00
|
|
|
|
using Exceptions = std::vector<std::exception_ptr>;
|
2015-10-05 01:31:57 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void throwFromErrno(const std::string & s, int code = 0, int the_errno = errno);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** Попробовать записать исключение в лог (и забыть про него).
|
|
|
|
|
* Можно использовать в деструкторах в блоке catch (...).
|
|
|
|
|
*/
|
|
|
|
|
void tryLogCurrentException(const char * log_name, const std::string & start_of_message = "");
|
|
|
|
|
void tryLogCurrentException(Poco::Logger * logger, const std::string & start_of_message = "");
|
|
|
|
|
|
2017-02-07 16:36:12 +00:00
|
|
|
|
/** Prints current exception in canonical format.
|
|
|
|
|
* with_stacktrace - prints stack trace for DB::Exception.
|
|
|
|
|
* check_embedded_stacktrace - if DB::Exception has embedded stacktrace then
|
|
|
|
|
* only this stack trace will be printed.
|
|
|
|
|
*/
|
|
|
|
|
std::string getCurrentExceptionMessage(bool with_stacktrace, bool check_embedded_stacktrace = false);
|
|
|
|
|
|
|
|
|
|
/// Returns error code from ErrorCodes
|
2017-02-01 20:30:46 +00:00
|
|
|
|
int getCurrentExceptionCode();
|
2015-10-05 01:31:57 +00:00
|
|
|
|
|
2015-10-05 05:40:27 +00:00
|
|
|
|
void tryLogException(std::exception_ptr e, const char * log_name, const std::string & start_of_message = "");
|
|
|
|
|
void tryLogException(std::exception_ptr e, Poco::Logger * logger, const std::string & start_of_message = "");
|
|
|
|
|
|
|
|
|
|
std::string getExceptionMessage(std::exception_ptr e, bool with_stacktrace);
|
|
|
|
|
|
|
|
|
|
|
2016-06-08 14:39:30 +00:00
|
|
|
|
void rethrowFirstException(const Exceptions & exceptions);
|
2015-10-05 01:31:57 +00:00
|
|
|
|
|
2016-05-28 14:14:18 +00:00
|
|
|
|
std::unique_ptr<Poco::Exception> convertCurrentException();
|
2015-10-05 05:40:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
typename std::enable_if<std::is_pointer<T>::value, T>::type exception_cast(std::exception_ptr e)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
std::rethrow_exception(e);
|
|
|
|
|
}
|
|
|
|
|
catch (typename std::remove_pointer<T>::type & concrete)
|
|
|
|
|
{
|
|
|
|
|
return &concrete;
|
|
|
|
|
}
|
|
|
|
|
catch (...)
|
|
|
|
|
{
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-05 01:26:43 +00:00
|
|
|
|
}
|