/* * 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_OFFSET_H #define SQLPP_OFFSET_H #include #include #include namespace sqlpp { // OFFSET DATA template struct offset_data_t { offset_data_t(Offset value): _value(value) {} offset_data_t(const offset_data_t&) = default; offset_data_t(offset_data_t&&) = default; offset_data_t& operator=(const offset_data_t&) = default; offset_data_t& operator=(offset_data_t&&) = default; ~offset_data_t() = default; Offset _value; }; // OFFSET template struct offset_t { using _traits = make_traits; using _recursive_traits = make_recursive_traits; static_assert(is_integral_t::value, "offset requires an integral value or integral parameter"); // Data using _data_t = offset_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 = offset_data_t; _impl_t offset; _impl_t& operator()() { return offset; } const _impl_t& operator()() const { return offset; } template static auto _get_member(T t) -> decltype(t.offset) { return t.offset; } }; template struct _methods_t { static void _check_consistency() {} }; }; // DYNAMIC OFFSET DATA template struct dynamic_offset_data_t { dynamic_offset_data_t(): _value(noop()) { } template dynamic_offset_data_t(Offset value): _initialized(true), _value(typename wrap_operand::type(value)) { } dynamic_offset_data_t(const dynamic_offset_data_t&) = default; dynamic_offset_data_t(dynamic_offset_data_t&&) = default; dynamic_offset_data_t& operator=(const dynamic_offset_data_t&) = default; dynamic_offset_data_t& operator=(dynamic_offset_data_t&&) = default; ~dynamic_offset_data_t() = default; bool _initialized = false; interpretable_t _value; }; // DYNAMIC OFFSET template struct dynamic_offset_t { using _traits = make_traits; using _recursive_traits = make_recursive_traits<>; // Data using _data_t = dynamic_offset_data_t; // Member implementation with data and methods template struct _impl_t { template void set(Offset value) { // FIXME: Make sure that Offset does not require external tables? Need to read up on SQL using arg_t = typename wrap_operand::type; _data._value = arg_t{value}; _data._initialized = true; } public: _data_t _data; }; // Member template for adding the named member to a statement template struct _member_t { using _data_t = dynamic_offset_data_t; _impl_t offset; _impl_t& operator()() { return offset; } const _impl_t& operator()() const { return offset; } template static auto _get_member(T t) -> decltype(t.offset) { return t.offset; } }; template struct _methods_t { static void _check_consistency() {} template void set_offset(Offset value) { // FIXME: Make sure that Offset does not require external tables? Need to read up on SQL using arg_t = typename wrap_operand::type; static_cast(this)->_offset()._value = arg_t{value}; static_cast(this)->_offset()._initialized = true; } }; bool _initialized = false; interpretable_t _value; }; struct no_offset_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_offset; _impl_t& operator()() { return no_offset; } const _impl_t& operator()() const { return no_offset; } template static auto _get_member(T t) -> decltype(t.no_offset) { return t.no_offset; } }; template struct _methods_t { using _database_t = typename Policies::_database_t; template using _new_statement_t = typename Policies::template _new_statement_t; static void _check_consistency() {} template auto offset(Arg arg) -> _new_statement_t::type>> { return { *static_cast(this), offset_data_t::type>{{arg}} }; } auto dynamic_offset() -> _new_statement_t> { static_assert(not std::is_same<_database_t, void>::value, "dynamic_offset must not be called in a static statement"); return { *static_cast(this), dynamic_offset_data_t<_database_t>{} }; } }; }; // Interpreters template struct serializer_t> { using T = offset_data_t; static Context& _(const T& t, Context& context) { context << " OFFSET "; serialize(t._value, context); return context; } }; template struct serializer_t> { using T = dynamic_offset_data_t; static Context& _(const T& t, Context& context) { if (t._initialized) { context << " OFFSET "; serialize(t._value, context); } return context; } }; } #endif