aboutsummaryrefslogtreecommitdiff
path: root/odb/mysql/result.txx
blob: 4b18abc4a8f5f5cde461c3c3596660f2564cdd05 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
// file      : odb/mysql/result.txx
// author    : Boris Kolpackov <boris@codesynthesis.com>
// copyright : Copyright (c) 2009-2010 Code Synthesis Tools CC
// license   : GNU GPL v2; see accompanying LICENSE file

#include <odb/cache-traits.hxx>

namespace odb
{
  namespace mysql
  {
    template <typename T>
    result_impl<T>::
    ~result_impl ()
    {
    }

    template <typename T>
    result_impl<T>::
    result_impl (details::shared_ptr<select_statement> statement,
                 object_statements<object_type>& statements)
        : odb::result_impl<T> (statements.connection ().database ()),
          statement_ (statement),
          statements_ (statements)
    {
      next ();
    }

    template <typename T>
    void result_impl<T>::
    load (object_type& obj)
    {
      // This is a top-level call so the statements cannot be locked.
      //
      assert (!statements_.locked ());
      typename object_statements<object_type>::auto_lock l (statements_);

      object_traits::init (obj, statements_.image (), this->database ());

      statements_.load_delayed ();
      l.unlock ();
    }

    template <typename T>
    typename result_impl<T>::id_type result_impl<T>::
    load_id ()
    {
      return object_traits::id (statements_.image ());
    }

    template <typename T>
    void result_impl<T>::
    next ()
    {
      this->current (pointer_type ());

      // If the result was cached the image can grow between calls
      // to next() as a result of other statements execution.
      //
      if (statement_->cached ())
      {
        typename object_traits::image_type& im (statements_.image ());

        if (im.version != statements_.out_image_version ())
        {
          binding& b (statements_.out_image_binding ());
          object_traits::bind (b.bind, im, true);
          statements_.out_image_version (im.version);
          b.version++;
        }
      }

      select_statement::result r (statement_->fetch ());

      switch (r)
      {
      case select_statement::truncated:
        {
          typename object_traits::image_type& im (statements_.image ());
          object_traits::grow (im, statements_.out_image_error ());

          if (im.version != statements_.out_image_version ())
          {
            binding& b (statements_.out_image_binding ());
            object_traits::bind (b.bind, im, true);
            statements_.out_image_version (im.version);
            b.version++;
            statement_->refetch ();
          }
          // Fall throught.
        }
      case select_statement::success:
        {
          break;
        }
      case select_statement::no_data:
        {
          this->end_ = true;
          break;
        }
      }
    }

    template <typename T>
    void result_impl<T>::
    cache ()
    {
      statement_->cache ();
    }

    template <typename T>
    std::size_t result_impl<T>::
    size ()
    {
      return statement_->result_size ();
    }
  }
}