// file : odb/sqlite/traits.hxx // copyright : Copyright (c) 2005-2015 Code Synthesis Tools CC // license : GNU GPL v2; see accompanying LICENSE file #ifndef ODB_SQLITE_TRAITS_HXX #define ODB_SQLITE_TRAITS_HXX #include #include // ODB_CXX11 #include #include #include // std::numeric_limits #include // std::size_t #include // std::memcpy, std::memset, std::strlen #ifdef ODB_CXX11 # include #endif #include #include #include #include #include #include #include #include #include #include namespace odb { namespace sqlite { // // image_traits // template struct image_traits; template struct image_traits {typedef long long image_type;}; template struct image_traits {typedef double image_type;}; template struct image_traits { typedef details::buffer image_type; // By default the text is in UTF-8. // static const bind::buffer_type bind_value = bind::text; }; template struct image_traits {typedef details::buffer image_type;}; template struct image_traits { typedef stream_buffers image_type; }; template struct image_traits { typedef stream_buffers image_type; }; // // value_traits // template struct wrapped_value_traits; template struct default_value_traits; template ::r> struct select_traits; template struct select_traits { typedef default_value_traits type; }; template struct select_traits { typedef wrapped_value_traits::null_handler> type; }; template class value_traits: public select_traits::type { }; // The wrapped_value_traits specializations should be able to handle // any value type which means we have to have every possible signature // of the set_value() and set_image() functions. // template struct wrapped_value_traits { typedef wrapper_traits wtraits; typedef typename wtraits::unrestricted_wrapped_type wrapped_type; typedef W value_type; typedef wrapped_type query_type; typedef typename image_traits::image_type image_type; typedef value_traits vtraits; static void set_value (W& v, const image_type& i, bool is_null) { vtraits::set_value (wtraits::set_ref (v), i, is_null); } static void set_image (image_type& i, bool& is_null, const W& v) { vtraits::set_image (i, is_null, wtraits::get_ref (v)); } // TEXT and BLOB. // static void set_value (W& v, const details::buffer& b, std::size_t n, bool is_null) { vtraits::set_value (wtraits::set_ref (v), b, n, is_null); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const W& v) { vtraits::set_image (b, n, is_null, wtraits::get_ref (v)); } // TEXT and BLOB STREAM. // static void set_value (W& v, const stream_buffers& b, std::size_t n, bool is_null) { vtraits::set_value (wtraits::set_ref (v), b, n, is_null); } static void set_image (stream_buffers& b, std::size_t& n, bool& is_null, const W& v) { vtraits::set_image (b, n, is_null, wtraits::get_ref (v)); } }; template struct wrapped_value_traits { typedef wrapper_traits wtraits; typedef typename wtraits::unrestricted_wrapped_type wrapped_type; typedef W value_type; typedef wrapped_type query_type; typedef typename image_traits::image_type image_type; typedef value_traits vtraits; static void set_value (W& v, const image_type& i, bool is_null) { if (is_null) wtraits::set_null (v); else vtraits::set_value (wtraits::set_ref (v), i, is_null); } static void set_image (image_type& i, bool& is_null, const W& v) { is_null = wtraits::get_null (v); if (!is_null) vtraits::set_image (i, is_null, wtraits::get_ref (v)); } // TEXT and BLOB. // static void set_value (W& v, const details::buffer& b, std::size_t n, bool is_null) { if (is_null) wtraits::set_null (v); else vtraits::set_value (wtraits::set_ref (v), b, n, is_null); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const W& v) { is_null = wtraits::get_null (v); if (!is_null) vtraits::set_image (b, n, is_null, wtraits::get_ref (v)); } // TEXT and BLOB STREAM. // static void set_value (W& v, const stream_buffers& b, std::size_t n, bool is_null) { if (is_null) wtraits::set_null (v); else vtraits::set_value (wtraits::set_ref (v), b, n, is_null); } static void set_image (stream_buffers& b, std::size_t& n, bool& is_null, const W& v) { is_null = wtraits::get_null (v); if (!is_null) vtraits::set_image (b, n, is_null, wtraits::get_ref (v)); } }; template struct default_value_traits { typedef T value_type; typedef T query_type; typedef typename image_traits::image_type image_type; static void set_value (T& v, const image_type& i, bool is_null) { if (!is_null) v = T (i); else v = T (); } static void set_image (image_type& i, bool& is_null, T v) { is_null = false; i = image_type (v); } }; // Float & double specialization. SQLite converts NaNs to NULLs so // we convert NULLs to NaNs for consistency. // template struct real_value_traits { typedef T value_type; typedef T query_type; typedef double image_type; static void set_value (T& v, double i, bool is_null) { if (!is_null) v = T (i); else v = std::numeric_limits::quiet_NaN (); } static void set_image (double& i, bool& is_null, T v) { is_null = false; i = image_type (v); } }; template <> struct LIBODB_SQLITE_EXPORT default_value_traits: real_value_traits { }; template <> struct LIBODB_SQLITE_EXPORT default_value_traits: real_value_traits { }; // std::string specialization. // template <> struct LIBODB_SQLITE_EXPORT default_value_traits { typedef std::string value_type; typedef std::string query_type; typedef details::buffer image_type; static void set_value (std::string& v, const details::buffer& b, std::size_t n, bool is_null) { if (!is_null) v.assign (b.data (), n); else v.erase (); } static void set_image (details::buffer&, std::size_t& n, bool& is_null, const std::string&); }; // char*/const char* specialization // // Specialization for const char* which only supports initialization // of an image from the value but not the other way around. This way // we can pass such values to the queries. // struct LIBODB_SQLITE_EXPORT c_string_value_traits { typedef const char* value_type; typedef details::buffer image_type; static void set_image (details::buffer&, std::size_t& n, bool& is_null, const char*); }; template <> struct LIBODB_SQLITE_EXPORT default_value_traits: c_string_value_traits {}; template <> struct LIBODB_SQLITE_EXPORT default_value_traits: c_string_value_traits {}; // char[N] specialization. // struct LIBODB_SQLITE_EXPORT c_array_value_traits_base { static void set_value (char* const& v, const details::buffer& b, std::size_t n, bool is_null, std::size_t N); static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const char* v, std::size_t N); }; template struct default_value_traits { typedef char* value_type; typedef char query_type[N]; typedef details::buffer image_type; static void set_value (char* const& v, const details::buffer& b, std::size_t n, bool is_null) { c_array_value_traits_base::set_value (v, b, n, is_null, N); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const char* v) { c_array_value_traits_base::set_image (b, n, is_null, v, N); } }; // std::array (string) specialization. // #ifdef ODB_CXX11 template struct default_value_traits, id_text> { typedef std::array value_type; typedef std::array query_type; typedef details::buffer image_type; static void set_value (value_type& v, const details::buffer& b, std::size_t n, bool is_null) { c_array_value_traits_base::set_value (v.data (), b, n, is_null, N); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const value_type& v) { c_array_value_traits_base::set_image (b, n, is_null, v.data (), N); } }; #endif // char specialization. // template <> struct LIBODB_SQLITE_EXPORT default_value_traits { typedef char value_type; typedef char query_type; typedef details::buffer image_type; static void set_value (char& v, const details::buffer& b, std::size_t n, bool is_null) { c_array_value_traits_base::set_value (&v, b, n, is_null, 1); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, char v) { c_array_value_traits_base::set_image (b, n, is_null, &v, 1); } }; #ifdef _WIN32 // std::wstring specialization. Using UTF-16 binding. // struct wstring_image_traits { typedef details::buffer image_type; static const bind::buffer_type bind_value = bind::text16; }; template <> struct image_traits: wstring_image_traits {}; template <> struct LIBODB_SQLITE_EXPORT default_value_traits { typedef std::wstring value_type; typedef std::wstring query_type; typedef details::buffer image_type; static void set_value (std::wstring& v, const details::buffer& b, std::size_t n, bool is_null) { if (!is_null) v.assign (reinterpret_cast (b.data ()), n / 2); else v.erase (); } static void set_image (details::buffer&, std::size_t& n, bool& is_null, const std::wstring&); }; // wchar_t*/const wchar_t* specialization. // struct LIBODB_SQLITE_EXPORT c_wstring_value_traits { typedef const wchar_t* value_type; typedef details::buffer image_type; static void set_image (details::buffer&, std::size_t& n, bool& is_null, const wchar_t*); }; template <> struct image_traits: wstring_image_traits {}; template <> struct LIBODB_SQLITE_EXPORT default_value_traits: c_wstring_value_traits {}; template <> struct image_traits: wstring_image_traits {}; template <> struct LIBODB_SQLITE_EXPORT default_value_traits: c_wstring_value_traits {}; // wchar_t[N] specialization. // struct LIBODB_SQLITE_EXPORT c_warray_value_traits_base { static void set_value (wchar_t* const& v, const details::buffer& b, std::size_t n, bool is_null, std::size_t N); static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const wchar_t* v, std::size_t N); }; template struct image_traits: wstring_image_traits {}; template struct default_value_traits { typedef wchar_t* value_type; typedef wchar_t query_type[N]; typedef details::buffer image_type; static void set_value (wchar_t* const& v, const details::buffer& b, std::size_t n, bool is_null) { c_warray_value_traits_base::set_value (v, b, n, is_null, N); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const wchar_t* v) { c_warray_value_traits_base::set_image (b, n, is_null, v, N); } }; // std::array (string) specialization. // #ifdef ODB_CXX11 template struct image_traits, id_text>: wstring_image_traits {}; template struct default_value_traits, id_text> { typedef std::array value_type; typedef std::array query_type; typedef details::buffer image_type; static void set_value (value_type& v, const details::buffer& b, std::size_t n, bool is_null) { c_warray_value_traits_base::set_value (v.data (), b, n, is_null, N); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const value_type& v) { c_warray_value_traits_base::set_image (b, n, is_null, v.data (), N); } }; #endif // wchar_t specialization. // template <> struct image_traits: wstring_image_traits {}; template <> struct LIBODB_SQLITE_EXPORT default_value_traits { typedef wchar_t value_type; typedef wchar_t query_type; typedef details::buffer image_type; static void set_value (wchar_t& v, const details::buffer& b, std::size_t n, bool is_null) { c_warray_value_traits_base::set_value (&v, b, n, is_null, 1); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, wchar_t v) { c_warray_value_traits_base::set_image (b, n, is_null, &v, 1); } }; #endif // _WIN32 // std::vector (buffer) specialization. // template <> struct LIBODB_SQLITE_EXPORT default_value_traits< std::vector, id_blob> { public: typedef std::vector value_type; typedef std::vector query_type; typedef details::buffer image_type; static void set_value (value_type& v, const details::buffer& b, std::size_t n, bool is_null) { if (!is_null) v.assign (b.data (), b.data () + n); else v.clear (); } static void set_image (details::buffer&, std::size_t& n, bool& is_null, const value_type&); }; // std::vector (buffer) specialization. // template <> struct LIBODB_SQLITE_EXPORT default_value_traits< std::vector, id_blob> { public: typedef std::vector value_type; typedef std::vector query_type; typedef details::buffer image_type; static void set_value (value_type& v, const details::buffer& b, std::size_t n, bool is_null) { if (!is_null) { const unsigned char* d ( reinterpret_cast (b.data ())); v.assign (d, d + n); } else v.clear (); } static void set_image (details::buffer&, std::size_t& n, bool& is_null, const value_type&); }; // char[N] (buffer) specialization. // template struct default_value_traits { public: typedef char* value_type; typedef char query_type[N]; typedef details::buffer image_type; static void set_value (char* const& v, const details::buffer& b, std::size_t n, bool is_null) { if (!is_null) std::memcpy (v, b.data (), (n < N ? n : N)); else std::memset (v, 0, N); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const char* v) { is_null = false; n = N; if (n > b.capacity ()) b.capacity (n); std::memcpy (b.data (), v, n); } }; // unsigned char[N] (buffer) specialization. // template struct default_value_traits { public: typedef unsigned char* value_type; typedef unsigned char query_type[N]; typedef details::buffer image_type; static void set_value (unsigned char* const& v, const details::buffer& b, std::size_t n, bool is_null) { if (!is_null) std::memcpy (v, b.data (), (n < N ? n : N)); else std::memset (v, 0, N); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const unsigned char* v) { is_null = false; n = N; if (n > b.capacity ()) b.capacity (n); std::memcpy (b.data (), v, n); } }; #ifdef ODB_CXX11 // std::array (buffer) specialization. // template struct default_value_traits, id_blob> { public: typedef std::array value_type; typedef value_type query_type; typedef details::buffer image_type; static void set_value (value_type& v, const details::buffer& b, std::size_t n, bool is_null) { if (!is_null) std::memcpy (v.data (), b.data (), (n < N ? n : N)); else std::memset (v.data (), 0, N); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const value_type& v) { is_null = false; n = N; if (n > b.capacity ()) b.capacity (n); std::memcpy (b.data (), v.data (), n); } }; // std::array (buffer) specialization. // template struct default_value_traits, id_blob> { public: typedef std::array value_type; typedef value_type query_type; typedef details::buffer image_type; static void set_value (value_type& v, const details::buffer& b, std::size_t n, bool is_null) { if (!is_null) std::memcpy (v.data (), b.data (), (n < N ? n : N)); else std::memset (v.data (), 0, N); } static void set_image (details::buffer& b, std::size_t& n, bool& is_null, const value_type& v) { is_null = false; n = N; if (n > b.capacity ()) b.capacity (n); std::memcpy (b.data (), v.data (), n); } }; #endif // text (stream) specialization. // template <> struct LIBODB_SQLITE_EXPORT default_value_traits { public: typedef text value_type; typedef std::string query_type; typedef stream_buffers image_type; static void set_value (text& v, const stream_buffers& b, std::size_t, bool is_null) { if (!is_null) { v.db_ = b.db.in; v.table_ = b.table.in; v.column_ = b.column.in; v.rowid_ = b.rowid.in; } } static void set_image (stream_buffers& b, std::size_t& n, bool& is_null, const text& v) { is_null = false; n = v.size_; b.db.out = &v.db_; b.table.out = &v.table_; b.column.out = &v.column_; b.rowid.out = &v.rowid_; } }; // blob (stream) specialization. // template <> struct LIBODB_SQLITE_EXPORT default_value_traits { public: typedef blob value_type; typedef std::vector query_type; typedef stream_buffers image_type; static void set_value (blob& v, const stream_buffers& b, std::size_t, bool is_null) { if (!is_null) { v.db_ = b.db.in; v.table_ = b.table.in; v.column_ = b.column.in; v.rowid_ = b.rowid.in; } } static void set_image (stream_buffers& b, std::size_t& n, bool& is_null, const blob& v) { is_null = false; n = v.size_; b.db.out = &v.db_; b.table.out = &v.table_; b.column.out = &v.column_; b.rowid.out = &v.rowid_; } }; // // type_traits // template struct default_type_traits; template class type_traits: public default_type_traits {}; // Integral types. // template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_integer; }; // Float types. // template <> struct default_type_traits { static const database_type_id db_type_id = id_real; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_real; }; // String types. // template <> struct default_type_traits { static const database_type_id db_type_id = id_text; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_text; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_text; }; template struct default_type_traits { static const database_type_id db_type_id = id_text; }; #ifdef ODB_CXX11 template struct default_type_traits > { static const database_type_id db_type_id = id_text; }; #endif template <> struct default_type_traits { static const database_type_id db_type_id = id_text; }; // Binary types. // template struct default_type_traits { static const database_type_id db_type_id = id_blob; }; template <> struct default_type_traits > { static const database_type_id db_type_id = id_blob; }; template <> struct default_type_traits > { static const database_type_id db_type_id = id_blob; }; #ifdef ODB_CXX11 template struct default_type_traits > { static const database_type_id db_type_id = id_blob; }; #endif template <> struct default_type_traits { static const database_type_id db_type_id = id_text_stream; }; template <> struct default_type_traits { static const database_type_id db_type_id = id_blob_stream; }; } } #include #endif // ODB_SQLITE_TRAITS_HXX