Я использую std :: error_code и имею кучу определенных ошибок (с использованием класса enum) и зарегистрировано.std :: error_code, my_error :: check_block == my_error :: validate && my_error :: accept_block == my_error :: validate
У меня есть очень общая ошибка, теперь называемая my_error :: validate, но хочу предоставить более конкретные версии в моей библиотеке. Как правило, люди хотят использовать:
if (ec == bc::error::validate)
// ...
Однако иногда они могут пожелать, чтобы увидеть конкретную ошибку, связанную с этим станд :: Error_Code или распечатать сообщение об ошибке.
// ec.message() says "check_block() failed to do XYZ"
assert(ec == bc::error::check_block);
Я хочу, чтобы иметь возможность включить что-то вроде:
if (ec == bc::error::validate)
{
if (ec == bc::error::check_block)
// bc::error::check_block is a more specific case of bc::error::validate
}
Кажется, я могу каким-то образом использовать категории или условия? Как я могу это сделать без необходимости определения целого ряда новых перечислений ошибок? Это для библиотеки, поэтому для пользователя этой библиотеки будет больно использовать bc :: generic_error :: validate и bc :: error :: check_block.
код ниже:
#include <system_error>
namespace bc {
enum class error
{
// storage errors
missing_object = 1,
object_already_exists,
unspent_output,
// transaction_pool errors
bad_transaction,
// network errors
resolve_failed,
network_unreachable,
address_in_use,
listen_failed,
accept_failed,
bad_stream,
channel_stopped,
channel_timeout,
// validate
validate_failed,
check_block,
accept_block,
connect_block
};
class error_category_impl
: public std::error_category
{
public:
virtual const char* name() const;
virtual std::string message(int ev) const;
virtual std::error_condition default_error_condition(int ev) const;
};
const std::error_category& error_category();
std::error_code make_error_code(error e);
std::error_condition make_error_condition(error e);
} // bc
namespace std
{
template <>
struct is_error_code_enum<libbitcoin::error>
: public true_type {};
}
И исходный файл TU:
#include <bc/error.hpp>
namespace bc {
const char* error_category_impl::name() const
{
return "bitcoin";
}
std::string error_category_impl::message(int ev) const
{
error ec = static_cast<error>(ev);
switch (ec)
{
case error::missing_object:
return "Object does not exist";
case error::object_already_exists:
return "Matching previous object found";
case error::unspent_output:
return "Unspent output";
case error::bad_transaction:
return "Transaction failed to validate";
case error::resolve_failed:
return "Resolving hostname failed";
case error::network_unreachable:
return "Unable to reach remote network";
case error::address_in_use:
return "Address already in use";
case error::listen_failed:
return "Listen incoming connections failed";
case error::accept_failed:
return "Accept connection failed";
case error::bad_stream:
return "Bad stream";
case error::channel_stopped:
return "Channel stopped";
case error::channel_timeout:
return "Channel timed out";
default:
return "Unknown error";
}
}
std::error_condition
error_category_impl::default_error_condition(int ev) const
{
error ec = static_cast<error>(ev);
switch (ec)
{
case error::check_block:
case error::accept_block:
case error::connect_block:
//return error::validate_failed;
return std::errc::permission_denied;
default:
return std::error_condition(ev, *this);
}
}
const std::error_category& error_category()
{
static error_category_impl instance;
return instance;
}
std::error_code make_error_code(error e)
{
return std::error_code(static_cast<int>(e), error_category());
}
std::error_condition make_error_condition(error e)
{
return std::error_condition(static_cast<int>(e), error_category());
}
} // bc
Очень интересно! Сейчас не так много примеров того, как реализовать пользовательское сообщение об ошибке на C++ 11, поэтому ваш ответ является очень ценным ресурсом. Для записи я немного изучил ваш вопрос, понял, что вам нужно сделать валидацию error_condition и сопоставить его с accept_block/check_block и т. Д., Но не мог понять, как это сделать. Очень забавно видеть, что в конечном итоге в этом случае простые перечисления лучше, чем C++ 11 enum class именно потому, что они имеют глобальную видимость в их пространстве имен! –
Также неплохо видеть, что достаточно хорошо разработан для поддержки умеренно сложной инфраструктуры ошибок, такой как ваш, с этим разделом разметки кода ошибки. Я просто хочу, чтобы они нашли способ во время стандартизации, чтобы немного упростить систему, по-прежнему трудно понять, как это работает с первого взгляда. –
Да, я думаю, что это довольно классная система :) Исходный код доступен здесь для всех заинтересованных: http://gitorious.org/libbitcoin/libbitcoin/trees/master (см. Include/error.hpp и src/error.cpp) – genjix