// file : odb/cache-traits.hxx // copyright : Copyright (c) 2009-2012 Code Synthesis Tools CC // license : GNU GPL v2; see accompanying LICENSE file #ifndef ODB_CACHE_TRAITS_HXX #define ODB_CACHE_TRAITS_HXX #include #include #include #include #include namespace odb { // Caching traits for objects passed by pointer. // template struct pointer_cache_traits_impl; template struct pointer_cache_traits: pointer_cache_traits_impl< P, typename object_traits::element_type>::id_type, pointer_traits

::kind> { }; template struct pointer_cache_traits_impl { typedef P pointer_type; typedef odb::pointer_traits pointer_traits; typedef typename pointer_traits::element_type element_type; // element_type can be const while object_type is always non-const. // typedef typename object_traits::object_type object_type; typedef typename object_traits::id_type id_type; typedef session::object_position position_type; struct insert_guard { insert_guard (const position_type& pos): pos_ (pos) {} ~insert_guard () {erase (pos_);} position_type position () const {return pos_;} void release () {pos_.map_ = 0;} private: position_type pos_; }; // We need the insert() overload with explicit id to handle self- // references. In such cases the object is not yet loaded and the // id member does not contain the correct id. // // Qualify the database type to resolve a phony ambiguity in VC 10. // static position_type insert (odb::database& db, const id_type& id, const pointer_type& p) { if (session::has_current ()) { // Cast away constness if any. // return session::current ().insert ( db, id, pointer_traits::cast (p)); } else return position_type (); } static position_type insert (odb::database& db, const pointer_type& p) { const id_type& id ( object_traits::id ( pointer_traits::get_ref (p))); return insert (db, id, p); } static pointer_type find (odb::database& db, const id_type& id) { if (session::has_current ()) return session::current ().find (db, id); else return pointer_type (); } static void erase (odb::database& db, const id_type& id) { if (session::has_current ()) session::current ().erase (db, id); } static void erase (const position_type& p) { if (p.map_ != 0) session::current ().erase (p); } }; template struct pointer_cache_traits_impl { typedef P pointer_type; struct position_type {}; static position_type insert (odb::database&, const pointer_type&) { return position_type (); } }; // Unique pointers don't work with the object cache. // template struct pointer_cache_traits_impl { typedef P pointer_type; typedef typename pointer_traits::element_type element_type; typedef typename object_traits::id_type id_type; struct position_type {}; struct insert_guard { insert_guard (const position_type&) {} position_type position () const {return position_type ();} void release () {} }; static position_type insert (odb::database&, const id_type&, const pointer_type&) { return position_type (); } static position_type insert (odb::database&, const pointer_type&) { return position_type (); } static pointer_type find (odb::database&, const id_type&) { return pointer_type (); } static void erase (odb::database&, const id_type&) {} static void erase (const position_type&) {} }; template struct pointer_cache_traits_impl { typedef P pointer_type; struct position_type {}; static position_type insert (odb::database&, const pointer_type&) { return position_type (); } }; // Caching traits for objects passed by reference. Only if the object // pointer kind is raw do we add the object to the session. // template struct reference_cache_traits_impl; template struct reference_cache_traits: reference_cache_traits_impl< T, typename object_traits::id_type, pointer_traits::pointer_type>::kind> { }; template struct reference_cache_traits_impl { typedef T element_type; typedef typename object_traits::pointer_type pointer_type; typedef typename object_traits::id_type id_type; typedef typename pointer_cache_traits::position_type position_type; struct insert_guard { insert_guard (const position_type&) {} position_type position () const {return position_type ();} void release () {} }; static position_type insert (odb::database&, const id_type&, element_type&) { return position_type (); } static position_type insert (odb::database&, element_type&) { return position_type (); } }; template struct reference_cache_traits_impl { typedef T element_type; struct position_type {}; static position_type insert (odb::database&, element_type&) { return position_type (); } }; template struct reference_cache_traits_impl { typedef T element_type; typedef typename object_traits::pointer_type pointer_type; typedef typename object_traits::id_type id_type; typedef typename pointer_cache_traits::position_type position_type; typedef typename pointer_cache_traits::insert_guard insert_guard; static position_type insert (odb::database& db, const id_type& id, element_type& obj) { pointer_type p (&obj); return pointer_cache_traits::insert (db, id, p); } static position_type insert (odb::database& db, element_type& obj) { pointer_type p (&obj); return pointer_cache_traits::insert (db, p); } }; template struct reference_cache_traits_impl { typedef T element_type; struct position_type {}; static position_type insert (odb::database&, element_type&) { return position_type (); } }; } #include #endif // ODB_CACHE_TRAITS_HXX