/*
* Copyright 2014 Google Inc. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef FRUIT_META_ERRORS_H
#define FRUIT_META_ERRORS_H
#include <fruit/impl/meta/basics.h>
#include <fruit/impl/meta/logical_operations.h>
namespace fruit {
namespace impl {
namespace meta {
template <typename T>
struct CheckIfError {
using type = T;
};
template <typename ErrorTag, typename... ErrorArgs>
struct CheckIfError<Error<ErrorTag, ErrorArgs...>> {
using type = typename ErrorTag::template apply<ErrorArgs...>;
};
// ConstructError(ErrorTag, Args...) returns Error<ErrorTag, Args...>.
// Never construct an Error<...> directly, using this metafunction makes debugging easier.
struct ConstructError {
template <typename ErrorTag, typename... Args>
struct apply {
#if FRUIT_DEEP_TEMPLATE_INSTANTIATION_STACKTRACES_FOR_ERRORS
static_assert(true || sizeof(typename CheckIfError<Error<ErrorTag, UnwrapType<Args>...>>::type), "");
#endif
using type = Error<ErrorTag, typename TypeUnwrapper<Args>::type...>;
};
};
// Extracts the first error in the given types.
struct ExtractFirstError {
template <typename... Types>
struct apply;
template <typename Type, typename... Types>
struct apply<Type, Types...> : public apply<Types...> {};
template <typename ErrorTag, typename... ErrorParams, typename... Types>
struct apply<Error<ErrorTag, ErrorParams...>, Types...> {
using type = Error<ErrorTag, ErrorParams...>;
};
};
} // namespace meta
} // namespace impl
} // namespace fruit
#endif // FRUIT_META_ERRORS_H