2018-02-03 18:22:01 -08:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
|
|
|
|
//
|
|
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
|
|
// copyright notice and this permission notice is present in all copies. The
|
|
|
|
// full license for this software is available in the LICENSE file.
|
2017-08-23 15:10:28 -06:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_M_ERROR_H
|
|
|
|
|
2017-09-07 04:10:15 -07:00
|
|
|
namespace ircd::m
|
|
|
|
{
|
|
|
|
struct error;
|
|
|
|
}
|
2017-08-23 15:10:28 -06:00
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
/// This hierarchy is designed to allow developers to throw an exception with
|
|
|
|
/// matrix protocol specific information which has the potential to become a
|
|
|
|
/// proper matrix JSON error object sent over HTTP to clients. Many exceptions
|
|
|
|
/// aren't intended to reach clients, but just in case, those can utilize an
|
|
|
|
/// http::INTERNAL_SERVER_ERROR. A regular IRCD_EXCEPTION won't be rooted in
|
|
|
|
/// m::error; if thrown during a client's request and not first caught it
|
|
|
|
/// may result in an internal server error sent to the client and may be
|
|
|
|
/// further rethrown.
|
|
|
|
///
|
|
|
|
/// The IRCD_M_EXCEPTION macro is provided and should be used to declare the
|
|
|
|
/// error type first rather than throwing an m::error directly if possible.
|
|
|
|
///
|
2018-03-04 09:20:24 -08:00
|
|
|
class ircd::m::error
|
|
|
|
:public http::error
|
2017-08-23 15:10:28 -06:00
|
|
|
{
|
2018-12-19 15:57:04 -08:00
|
|
|
static thread_local char fmtbuf[4_KiB];
|
2018-03-04 09:20:24 -08:00
|
|
|
|
2018-12-30 17:34:27 -08:00
|
|
|
IRCD_OVERLOAD(internal)
|
2019-07-11 17:22:16 -07:00
|
|
|
error(internal_t, const http::code &, std::string object);
|
2018-12-30 17:34:27 -08:00
|
|
|
|
|
|
|
protected:
|
|
|
|
IRCD_OVERLOAD(child)
|
|
|
|
template<class... args> error(child_t, args&&... a)
|
|
|
|
:error{std::forward<args>(a)...}
|
|
|
|
{}
|
|
|
|
|
2018-03-04 09:20:24 -08:00
|
|
|
public:
|
2019-04-25 23:35:46 -07:00
|
|
|
string_view errcode() const noexcept;
|
|
|
|
string_view errstr() const noexcept;
|
|
|
|
|
2018-12-10 13:08:35 -08:00
|
|
|
template<class... args> error(const http::code &, const string_view &errcode, const string_view &fmt, args&&...);
|
|
|
|
template<class... args> error(const string_view &errcode, const string_view &fmt, args&&...);
|
2018-01-28 09:09:58 -08:00
|
|
|
error(const http::code &, const json::object &object);
|
2017-09-14 11:30:06 -07:00
|
|
|
error(const http::code &, const json::members &);
|
2017-09-12 10:02:27 -07:00
|
|
|
error(const http::code &, const json::iov &);
|
2017-09-07 04:10:15 -07:00
|
|
|
error(const http::code &);
|
2018-03-05 01:34:03 -08:00
|
|
|
error(std::string);
|
|
|
|
error();
|
2020-02-28 12:12:51 -08:00
|
|
|
~error() noexcept;
|
2017-09-07 04:10:15 -07:00
|
|
|
};
|
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
/// Macro for all matrix exceptions; all errors rooted from m::error
|
|
|
|
///
|
|
|
|
/// Unfortunately we use another macro here instead of IRCD_EXCEPTION but what
|
|
|
|
/// we gain is a more suitable exception hierarchy for the matrix protocol
|
|
|
|
/// for fecundity of m::error, which is still the progeny of ircd::error. This
|
|
|
|
/// macro takes 3 arguments:
|
|
|
|
///
|
|
|
|
/// - parent: A parent exception class type similar to the rest of ircd's
|
|
|
|
/// system. For this macro, the parent can never be above m::error.
|
|
|
|
///
|
|
|
|
/// - name: The name of the exception is also what will be seen in the matrix
|
|
|
|
/// protocol JSON's errcode. The matrix protocol error codes are UPPER_CASE and
|
|
|
|
/// will appear as defined. This is also the name of this class itself too.
|
|
|
|
///
|
|
|
|
/// - httpcode: An HTTP code which will be used if this exception ever makes
|
|
|
|
/// it out to a client.
|
|
|
|
///
|
2018-01-28 09:09:58 -08:00
|
|
|
#define IRCD_M_EXCEPTION(_parent_, _name_, _httpcode_) \
|
|
|
|
struct _name_ \
|
|
|
|
: _parent_ \
|
|
|
|
{ \
|
|
|
|
_name_() \
|
|
|
|
: _parent_ \
|
|
|
|
{ \
|
|
|
|
child, _httpcode_, "M_"#_name_, "%s", http::status(_httpcode_) \
|
|
|
|
}{} \
|
|
|
|
\
|
2018-12-10 13:08:35 -08:00
|
|
|
template<class... args> _name_(const string_view &fmt, args&&... a) \
|
2018-01-28 09:09:58 -08:00
|
|
|
: _parent_ \
|
|
|
|
{ \
|
2018-03-05 01:34:03 -08:00
|
|
|
child, _httpcode_, "M_"#_name_, fmt, std::forward<args>(a)... \
|
2018-01-28 09:09:58 -08:00
|
|
|
}{} \
|
|
|
|
\
|
|
|
|
template<class... args> _name_(child_t, args&&... a) \
|
|
|
|
: _parent_ \
|
|
|
|
{ \
|
|
|
|
child, std::forward<args>(a)... \
|
|
|
|
}{} \
|
2017-08-23 15:10:28 -06:00
|
|
|
};
|
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
// These are some common m::error, but not all of them; other declarations
|
|
|
|
// may be dispersed throughout ircd::m.
|
2017-09-07 04:10:15 -07:00
|
|
|
namespace ircd::m
|
|
|
|
{
|
|
|
|
IRCD_M_EXCEPTION(error, UNKNOWN, http::INTERNAL_SERVER_ERROR);
|
2017-11-15 17:48:25 -08:00
|
|
|
IRCD_M_EXCEPTION(error, BAD_REQUEST, http::BAD_REQUEST);
|
2018-02-08 15:00:53 -08:00
|
|
|
IRCD_M_EXCEPTION(error, BAD_JSON, http::BAD_REQUEST);
|
|
|
|
IRCD_M_EXCEPTION(error, NOT_JSON, http::BAD_REQUEST);
|
2017-10-15 22:00:22 -07:00
|
|
|
IRCD_M_EXCEPTION(error, BAD_SIGNATURE, http::UNAUTHORIZED);
|
2019-07-26 19:09:28 -07:00
|
|
|
IRCD_M_EXCEPTION(error, ACCESS_DENIED, http::FORBIDDEN);
|
2018-02-08 15:00:53 -08:00
|
|
|
IRCD_M_EXCEPTION(error, FORBIDDEN, http::FORBIDDEN);
|
|
|
|
IRCD_M_EXCEPTION(error, NOT_FOUND, http::NOT_FOUND);
|
|
|
|
IRCD_M_EXCEPTION(error, UNSUPPORTED, http::NOT_IMPLEMENTED);
|
2018-02-14 21:52:19 -08:00
|
|
|
IRCD_M_EXCEPTION(error, NEED_MORE_PARAMS, http::MULTIPLE_CHOICES);
|
2018-02-22 19:31:15 -08:00
|
|
|
IRCD_M_EXCEPTION(error, UNAVAILABLE, http::SERVICE_UNAVAILABLE);
|
2018-02-23 21:17:47 -08:00
|
|
|
IRCD_M_EXCEPTION(error, BAD_PAGINATION, http::BAD_REQUEST);
|
2017-09-07 04:10:15 -07:00
|
|
|
}
|
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
template<class... args>
|
|
|
|
ircd::m::error::error(const http::code &status,
|
|
|
|
const string_view &errcode,
|
2018-12-10 13:08:35 -08:00
|
|
|
const string_view &fmt,
|
2017-11-15 17:48:25 -08:00
|
|
|
args&&... a)
|
2018-12-30 17:34:27 -08:00
|
|
|
:error
|
2017-11-15 17:48:25 -08:00
|
|
|
{
|
2018-12-30 17:34:27 -08:00
|
|
|
internal, status, [&errcode, &fmt, &a...]() -> json::strung
|
2017-11-15 17:48:25 -08:00
|
|
|
{
|
2018-03-04 09:20:24 -08:00
|
|
|
const string_view str
|
2017-11-15 17:48:25 -08:00
|
|
|
{
|
2018-03-04 09:20:24 -08:00
|
|
|
fmt::sprintf{fmtbuf, fmt, std::forward<args>(a)...}
|
2017-11-15 17:48:25 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
return json::members
|
|
|
|
{
|
2018-01-28 09:09:58 -08:00
|
|
|
{ "errcode", errcode },
|
|
|
|
{ "error", str },
|
2017-11-15 17:48:25 -08:00
|
|
|
};
|
|
|
|
}()
|
|
|
|
}{}
|
|
|
|
|
|
|
|
template<class... args>
|
|
|
|
ircd::m::error::error(const string_view &errcode,
|
2018-12-10 13:08:35 -08:00
|
|
|
const string_view &fmt,
|
2017-11-15 17:48:25 -08:00
|
|
|
args&&... a)
|
|
|
|
:error
|
|
|
|
{
|
2018-03-05 01:34:03 -08:00
|
|
|
http::INTERNAL_SERVER_ERROR, errcode, fmt, std::forward<args>(a)...
|
2018-01-28 09:09:58 -08:00
|
|
|
}
|
|
|
|
{}
|