// Copyright 2014 The Chromium OS Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include <brillo/errors/error.h> #include <base/logging.h> #include <base/strings/stringprintf.h> using brillo::Error; using brillo::ErrorPtr; namespace { inline void LogError(const base::Location& location, const std::string& domain, const std::string& code, const std::string& message) { // Use logging::LogMessage() directly instead of LOG(ERROR) to substitute // the current error location with the location passed in to the Error object. // This way the log will contain the actual location of the error, and not // as if it always comes from brillo/errors/error.cc(22). if (location.function_name() == nullptr) { logging::LogMessage(location.file_name(), location.line_number(), logging::LOG_ERROR) .stream() << "Domain=" << domain << ", Code=" << code << ", Message=" << message; return; } logging::LogMessage( location.file_name(), location.line_number(), logging::LOG_ERROR).stream() << location.function_name() << "(...): " << "Domain=" << domain << ", Code=" << code << ", Message=" << message; } } // anonymous namespace ErrorPtr Error::Create(const base::Location& location, const std::string& domain, const std::string& code, const std::string& message) { return Create(location, domain, code, message, ErrorPtr()); } ErrorPtr Error::Create(const base::Location& location, const std::string& domain, const std::string& code, const std::string& message, ErrorPtr inner_error) { LogError(location, domain, code, message); return ErrorPtr( new Error(location, domain, code, message, std::move(inner_error))); } void Error::AddTo(ErrorPtr* error, const base::Location& location, const std::string& domain, const std::string& code, const std::string& message) { if (error) { *error = Create(location, domain, code, message, std::move(*error)); } else { // Create already logs the error, but if |error| is nullptr, // we still want to log the error... LogError(location, domain, code, message); } } void Error::AddToPrintf(ErrorPtr* error, const base::Location& location, const std::string& domain, const std::string& code, const char* format, ...) { va_list ap; va_start(ap, format); std::string message = base::StringPrintV(format, ap); va_end(ap); AddTo(error, location, domain, code, message); } ErrorPtr Error::Clone() const { ErrorPtr inner_error = inner_error_ ? inner_error_->Clone() : nullptr; return ErrorPtr( new Error(location_, domain_, code_, message_, std::move(inner_error))); } bool Error::HasDomain(const std::string& domain) const { return FindErrorOfDomain(this, domain) != nullptr; } bool Error::HasError(const std::string& domain, const std::string& code) const { return FindError(this, domain, code) != nullptr; } const Error* Error::GetFirstError() const { const Error* err = this; while (err->GetInnerError()) err = err->GetInnerError(); return err; } Error::Error(const base::Location& location, const std::string& domain, const std::string& code, const std::string& message, ErrorPtr inner_error) : domain_(domain), code_(code), message_(message), location_(location), inner_error_(std::move(inner_error)) { } const Error* Error::FindErrorOfDomain(const Error* error_chain_start, const std::string& domain) { while (error_chain_start) { if (error_chain_start->GetDomain() == domain) break; error_chain_start = error_chain_start->GetInnerError(); } return error_chain_start; } const Error* Error::FindError(const Error* error_chain_start, const std::string& domain, const std::string& code) { while (error_chain_start) { if (error_chain_start->GetDomain() == domain && error_chain_start->GetCode() == code) break; error_chain_start = error_chain_start->GetInnerError(); } return error_chain_start; }