/* * Copyright (c) 2013-2014, Roland Bock * All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, * are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, this * list of conditions and the following disclaimer in the documentation and/or * other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef SQLPP_ORDER_BY_H #define SQLPP_ORDER_BY_H #include #include #include #include #include #include #include namespace sqlpp { // ORDER BY DATA template struct order_by_data_t { order_by_data_t(Expressions... expressions): _expressions(expressions...) {} order_by_data_t(const order_by_data_t&) = default; order_by_data_t(order_by_data_t&&) = default; order_by_data_t& operator=(const order_by_data_t&) = default; order_by_data_t& operator=(order_by_data_t&&) = default; ~order_by_data_t() = default; std::tuple _expressions; interpretable_list_t _dynamic_expressions; }; // ORDER BY template struct order_by_t { using _traits = make_traits; using _recursive_traits = make_recursive_traits; using _is_dynamic = is_database; static_assert(_is_dynamic::value or sizeof...(Expressions), "at least one expression (e.g. a column) required in order_by()"); static_assert(not detail::has_duplicates::value, "at least one duplicate argument detected in order_by()"); static_assert(detail::all_t::value...>::value, "at least one argument is not an expression in order_by()"); // Data using _data_t = order_by_data_t; // Member implementation with data and methods template struct _impl_t { template void add_ntc(Expression expression) { add(expression); } template void add(Expression expression) { static_assert(_is_dynamic::value, "add() must not be called for static order_by"); static_assert(is_expression_t::value, "invalid expression argument in order_by::add()"); static_assert(TableCheckRequired::value or Policies::template _no_unknown_tables::value, "expression uses tables unknown to this statement in order_by::add()"); using ok = detail::all_t<_is_dynamic::value, is_expression_t::value>; _add_impl(expression, ok()); // dispatch to prevent compile messages after the static_assert } private: template void _add_impl(Expression expression, const std::true_type&) { return _data._dynamic_expressions.emplace_back(expression); } template void _add_impl(Expression expression, const std::false_type&); public: _data_t _data; }; // Member template for adding the named member to a statement template struct _member_t { using _data_t = order_by_data_t; _impl_t order_by; _impl_t& operator()() { return order_by; } const _impl_t& operator()() const { return order_by; } template static auto _get_member(T t) -> decltype(t.order_by) { return t.order_by; } }; template struct _methods_t { static void _check_consistency() {} }; }; // NO ORDER BY YET struct no_order_by_t { using _traits = make_traits; using _recursive_traits = make_recursive_traits<>; // Data using _data_t = no_data_t; // Member implementation with data and methods template struct _impl_t { _data_t _data; }; // Member template for adding the named member to a statement template struct _member_t { using _data_t = no_data_t; _impl_t no_order_by; _impl_t& operator()() { return no_order_by; } const _impl_t& operator()() const { return no_order_by; } template static auto _get_member(T t) -> decltype(t.no_order_by) { return t.no_order_by; } }; template struct _methods_t { using _database_t = typename Policies::_database_t; template using _new_statement_t = new_statement; static void _check_consistency() {} template auto order_by(Args... args) const -> _new_statement_t> { return { static_cast&>(*this), order_by_data_t{args...} }; } template auto dynamic_order_by(Args... args) const -> _new_statement_t> { static_assert(not std::is_same<_database_t, void>::value, "dynamic_order_by must not be called in a static statement"); return { static_cast&>(*this), order_by_data_t<_database_t, Args...>{args...} }; } }; }; // Interpreters template struct serializer_t> { using T = order_by_data_t; static Context& _(const T& t, Context& context) { if (sizeof...(Expressions) == 0 and t._dynamic_expressions.empty()) return context; context << " ORDER BY "; interpret_tuple(t._expressions, ',', context); if (sizeof...(Expressions) and not t._dynamic_expressions.empty()) context << ','; interpret_list(t._dynamic_expressions, ',', context); return context; } }; } #endif