don't overwrite cached object fields on fetch all since we can't guarantee that it...
[dbsrgits/DBIx-Data-Store-old.git] / lib / DBIx / Data / Collection / Set.pm
CommitLineData
65b76960 1package DBIx::Data::Collection::Set;
2
3use Moose;
4use Method::Signatures::Simple;
5use Data::Perl::Stream::Array;
6
7has _store => (is => 'ro', required => 1, init_arg => 'store');
8
65b76960 9has _class => (is => 'ro', predicate => '_has_class');
10
3a2e7c1c 11has _set_over => (is => 'ro', required => 1, init_arg => 'set_over');
12
13## member cache (all members)
14
15has _member_cache => (
16 is => 'rw', lazy_build => 1,
17 predicate => '_member_cache_built',
18);
65b76960 19
20method _build__member_cache {
21 my $stream = $self->_new_raw_stream;
22 my @cache;
23 while (my ($raw) = $stream->next) {
3a2e7c1c 24 my $obj = do {
25 if (my ($obj) = $self->_key_cache_get_raw($raw)) {
b1c3fd5d 26 $obj # can't $self->_merge($obj, $raw) since $obj might have changed
3a2e7c1c 27 } else {
de9534fa 28 $self->_add_to_key_cache($self->_inflate($raw))
3a2e7c1c 29 }
30 };
31 push @cache, $obj;
65b76960 32 }
de9534fa 33 \@cache
65b76960 34}
35
3a2e7c1c 36method _add_to_member_cache ($to_add) {
c51eabc5 37 return $to_add unless $self->_member_cache_built;
3a2e7c1c 38 push @{$self->_member_cache}, $to_add;
de9534fa 39 $to_add
3a2e7c1c 40}
41
c51eabc5 42method _remove_from_member_cache ($to_remove) {
43 return $to_remove unless $self->_member_cache_built;
44 @{$self->_member_cache} = grep $_ ne $to_remove, @{$self->_member_cache};
45 $to_remove
46}
47
3a2e7c1c 48## key cache - by primary/unique key
49
50has _key_cache => (is => 'ro', default => sub { {} });
51
52method _add_to_key_cache ($to_add) {
53 $self->_key_cache->{$self->_object_to_id($to_add)} = $to_add;
de9534fa 54 $to_add
3a2e7c1c 55}
56
c51eabc5 57method _remove_from_key_cache ($to_remove) {
58 # should return $to_remove
59 delete $self->_key_cache->{$self->_object_to_id($to_remove)}
60}
61
3a2e7c1c 62method _key_cache_has_raw ($raw) {
63 exists $self->_key_cache->{$self->_raw_to_id($raw)}
64}
65
66method _key_cache_has_object ($obj) {
67 exists $self->_key_cache->{$self->_object_to_id($obj)}
68}
69
70method _key_cache_get_raw ($raw) {
e49bd861 71 $self->_key_cache_get_id($self->_raw_to_id($raw))
3a2e7c1c 72}
73
74method _key_cache_get_object ($obj) {
e49bd861 75 $self->_key_cache_get_id($self->_object_to_id($obj))
3a2e7c1c 76}
77
e49bd861 78method _key_cache_get_object_spec ($spec) {
79 # see _object_spec_to_id for doc of what the difference is
80 $self->_key_cache_get_id($self->_object_spec_to_id($spec))
81}
3a2e7c1c 82
e49bd861 83method _key_cache_get_id ($id) {
84 exists $self->_key_cache->{$id}
85 ? ($self->_key_cache->{$id})
86 : ()
65b76960 87}
88
3a2e7c1c 89## thunking between the store representation and the set representation
90#
91# _inflate is raw data -> final repr
92# _deflate is final repr -> raw data
93# _merge takes final repr + raw data and updates the repr
94# (this is used for pk-generated values and later lazy loading)
e49bd861 95#
96# _deflate_spec is attributes of final repr -> raw data
3a2e7c1c 97
65b76960 98method _inflate ($raw) {
3347c67e 99 bless($raw, $self->_class) if $self->_has_class;
c51eabc5 100 $raw
65b76960 101}
102
3a2e7c1c 103method _deflate ($obj) {
104 +{ %$obj }
105}
106
107method _merge ($obj, $raw) {
108 @{$obj}{keys %$raw} = values %$raw;
c51eabc5 109 $obj
3a2e7c1c 110}
111
e49bd861 112method _deflate_spec ($spec) {
113 $spec
114}
115
3a2e7c1c 116## methods to get ids
117
118method _raw_to_id ($raw) {
119 # XXX must escape this. or do something else.
120 join ';', map $raw->{$_}, @{$self->_set_over}
121}
122
123method _object_to_id ($obj) {
c51eabc5 124 $self->_raw_to_id($self->_deflate($obj))
3a2e7c1c 125}
126
e49bd861 127method _object_spec_to_id ($spec) {
128 # intentionally C&P from _raw_to - this is not the same thing. If a column
129 # were mapped to an attribute of a different name, the raw would have the
130 # column name as a key but an object spec would have the attribute name
131 join ';', map $spec->{$_}, @{$self->_set_over}
132}
133
c51eabc5 134## array-ish operations - i.e. get all members
135
e49bd861 136method _new_raw_stream {
137 $self->_store->new_select_command([])->execute
138}
139
65b76960 140method flatten {
141 @{$self->_member_cache};
142}
143
144method as_stream {
145 Data::Perl::Stream::Array->new(array => $self->_member_cache);
146}
147
e49bd861 148## load single row
149
150method get ($spec) {
151 if (my ($got) = $self->_key_cache_get_object_spec($spec)) {
152 return $got
153 }
154 if (my ($raw) = $self->_get_from_store($self->_deflate_spec($spec))) {
155 return $self->_add_to_key_cache($self->_inflate($raw))
156 }
157 return undef # we aren't handling cache misses here yet
158}
159
160method _get_from_store ($raw) {
161 $self->_store->new_select_single_command($raw)->execute
162}
163
c51eabc5 164## add to set
165
3a2e7c1c 166method add ($new) {
167 $self->_add_to_store($new);
168 $self->_add_to_caches($new);
c51eabc5 169 $new
3a2e7c1c 170}
171
172method _add_to_store ($new) {
173 my $new_raw = $self->_deflate($new);
174 $self->_merge($new, $self->_store->new_insert_command($new_raw)->execute);
c51eabc5 175 $new
3a2e7c1c 176}
177
178method _add_to_caches ($new) {
179 $self->_add_to_member_cache($new);
180 $self->_add_to_key_cache($new);
de9534fa 181 $new
3a2e7c1c 182}
183
c51eabc5 184## remove from set
185
186method remove ($old) {
187 $self->_remove_from_store($old);
188 $self->_remove_from_caches($old);
189 $old
190}
191
192method _remove_from_store ($old) {
48d91d77 193 $self->_store->new_delete_command($self->_deflate($old))->execute
c51eabc5 194}
195
196method _remove_from_caches ($old) {
197 $self->_remove_from_member_cache($old);
198 $self->_remove_from_key_cache($old);
199 $old
200}
201
48d91d77 202## update
203
204method _update_in_store ($obj) {
205 # this is currently a call command but we should think about it
206 # being a row command so that we can have RETURNING or other
207 # mechanisms handle things like set-on-update datetime values
208 $self->_store->new_update_command($self->_deflate($obj))->execute
209}
210
65b76960 2111;