I think this is a v8/v10 difference
[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;
50166086 6use Scalar::Util qw(weaken refaddr);
65b76960 7
8has _store => (is => 'ro', required => 1, init_arg => 'store');
9
9f2b6cc8 10has _class => (is => 'ro', predicate => '_has_class', init_arg => 'class');
65b76960 11
3a2e7c1c 12has _set_over => (is => 'ro', required => 1, init_arg => 'set_over');
13
14## member cache (all members)
15
16has _member_cache => (
50166086 17 is => 'ro', lazy_build => 1,
3a2e7c1c 18 predicate => '_member_cache_built',
50166086 19 writer => '_set_member_cache',
3a2e7c1c 20);
65b76960 21
22method _build__member_cache {
23 my $stream = $self->_new_raw_stream;
24 my @cache;
25 while (my ($raw) = $stream->next) {
3a2e7c1c 26 my $obj = do {
27 if (my ($obj) = $self->_key_cache_get_raw($raw)) {
9f2b6cc8 28 # can't just $self->_merge($obj, $raw) since $obj might have changed
29 $self->_refresh($obj, $raw)
3a2e7c1c 30 } else {
de9534fa 31 $self->_add_to_key_cache($self->_inflate($raw))
3a2e7c1c 32 }
33 };
34 push @cache, $obj;
65b76960 35 }
9f2b6cc8 36 $self->_notify_observers(all_members => \@cache);
de9534fa 37 \@cache
65b76960 38}
39
3a2e7c1c 40method _add_to_member_cache ($to_add) {
c51eabc5 41 return $to_add unless $self->_member_cache_built;
3a2e7c1c 42 push @{$self->_member_cache}, $to_add;
de9534fa 43 $to_add
3a2e7c1c 44}
45
c51eabc5 46method _remove_from_member_cache ($to_remove) {
47 return $to_remove unless $self->_member_cache_built;
48 @{$self->_member_cache} = grep $_ ne $to_remove, @{$self->_member_cache};
49 $to_remove
50}
51
3a2e7c1c 52## key cache - by primary/unique key
53
54has _key_cache => (is => 'ro', default => sub { {} });
55
56method _add_to_key_cache ($to_add) {
57 $self->_key_cache->{$self->_object_to_id($to_add)} = $to_add;
de9534fa 58 $to_add
3a2e7c1c 59}
60
c51eabc5 61method _remove_from_key_cache ($to_remove) {
62 # should return $to_remove
63 delete $self->_key_cache->{$self->_object_to_id($to_remove)}
64}
65
3a2e7c1c 66method _key_cache_has_raw ($raw) {
67 exists $self->_key_cache->{$self->_raw_to_id($raw)}
68}
69
70method _key_cache_has_object ($obj) {
71 exists $self->_key_cache->{$self->_object_to_id($obj)}
72}
73
74method _key_cache_get_raw ($raw) {
e49bd861 75 $self->_key_cache_get_id($self->_raw_to_id($raw))
3a2e7c1c 76}
77
78method _key_cache_get_object ($obj) {
e49bd861 79 $self->_key_cache_get_id($self->_object_to_id($obj))
3a2e7c1c 80}
81
e49bd861 82method _key_cache_get_object_spec ($spec) {
83 # see _object_spec_to_id for doc of what the difference is
84 $self->_key_cache_get_id($self->_object_spec_to_id($spec))
85}
3a2e7c1c 86
e49bd861 87method _key_cache_get_id ($id) {
88 exists $self->_key_cache->{$id}
89 ? ($self->_key_cache->{$id})
90 : ()
65b76960 91}
92
50166086 93method _all_key_cache_members {
94 values %{$self->_key_cache}
95}
96
32746a09 97method _set_key_cache_members ($members) {
98 %{$self->_key_cache} = (map +($self->_object_to_id($_) => $_), @$members);
99 return
100}
101
9f2b6cc8 102## observers
103
104has _observer_callbacks => (
50166086 105 is => 'ro', default => sub { {} },
9f2b6cc8 106);
107
108method _notify_observers ($event, $payload) {
50166086 109 my $oc = $self->_observer_callbacks;
110 foreach my $refaddr (keys %$oc) {
111 my ($obj, $cb) = @{$oc->{$refaddr}};
112 unless (defined $obj) { # weak ref was garbage collected
113 delete $oc->{$refaddr};
114 next;
115 }
116 $obj->$cb($self, $event, $payload);
9f2b6cc8 117 }
50166086 118 $payload
119}
120
121method _register_observer ($obj, $cb) {
122 my $entry = [ $obj, $cb ];
123 weaken($entry->[0]);
124 $self->_observer_callbacks->{refaddr($obj)} = $entry;
125 return
126}
127
128method _setup_observation_of ($other) {
129 $other->_register_observer($self, method ($from, $event, $payload) {
130 if ($event eq 'add' or $event eq 'get') {
131 $self->_add_to_caches($payload);
132 } elsif ($event eq 'remove') {
133 $self->_remove_from_caches($payload);
134 } elsif ($event eq 'all_members') {
135 # separate arrayref since future add will trigger push()
32746a09 136 $self->_set_caches([ @$payload ]);
50166086 137 }
138 });
139 return
9f2b6cc8 140}
141
3a2e7c1c 142## thunking between the store representation and the set representation
143#
144# _inflate is raw data -> final repr
145# _deflate is final repr -> raw data
146# _merge takes final repr + raw data and updates the repr
147# (this is used for pk-generated values and later lazy loading)
e49bd861 148#
149# _deflate_spec is attributes of final repr -> raw data
32746a09 150# _merge_spec is final repr + extra attributes and update repr
3a2e7c1c 151
65b76960 152method _inflate ($raw) {
3347c67e 153 bless($raw, $self->_class) if $self->_has_class;
c51eabc5 154 $raw
65b76960 155}
156
3a2e7c1c 157method _deflate ($obj) {
158 +{ %$obj }
159}
160
161method _merge ($obj, $raw) {
162 @{$obj}{keys %$raw} = values %$raw;
c51eabc5 163 $obj
3a2e7c1c 164}
165
9f2b6cc8 166method _refresh ($obj, $raw) {
167 # if $obj has been changed but not flushed we'd destroy data doing
168 # a blind merge - but if $obj has change tracking of some sort then
169 # we -could- do something safely, so this method exists to be mangled
170 # by subclasses
171 $obj
172}
173
e49bd861 174method _deflate_spec ($spec) {
175 $spec
176}
177
32746a09 178method _merge_spec ($obj, $spec) {
179 @{$obj}{keys %$spec} = values %$spec;
180 $obj
181}
182
3a2e7c1c 183## methods to get ids
184
185method _raw_to_id ($raw) {
186 # XXX must escape this. or do something else.
187 join ';', map $raw->{$_}, @{$self->_set_over}
188}
189
190method _object_to_id ($obj) {
c51eabc5 191 $self->_raw_to_id($self->_deflate($obj))
3a2e7c1c 192}
193
e49bd861 194method _object_spec_to_id ($spec) {
195 # intentionally C&P from _raw_to - this is not the same thing. If a column
196 # were mapped to an attribute of a different name, the raw would have the
197 # column name as a key but an object spec would have the attribute name
198 join ';', map $spec->{$_}, @{$self->_set_over}
199}
200
c51eabc5 201## array-ish operations - i.e. get all members
202
e49bd861 203method _new_raw_stream {
ec6807f8 204 $self->_store->new_select_command({})->execute
e49bd861 205}
206
65b76960 207method flatten {
208 @{$self->_member_cache};
209}
210
211method as_stream {
212 Data::Perl::Stream::Array->new(array => $self->_member_cache);
213}
214
32746a09 215method _set_caches ($members) {
216 $self->_set_member_cache($members);
217 $self->_set_key_cache_members($members);
218 return
219}
220
e49bd861 221## load single row
222
223method get ($spec) {
224 if (my ($got) = $self->_key_cache_get_object_spec($spec)) {
225 return $got
226 }
227 if (my ($raw) = $self->_get_from_store($self->_deflate_spec($spec))) {
50166086 228 return $self->_notify_observers(
229 get => $self->_add_to_key_cache($self->_inflate($raw))
230 );
e49bd861 231 }
232 return undef # we aren't handling cache misses here yet
233}
234
235method _get_from_store ($raw) {
236 $self->_store->new_select_single_command($raw)->execute
237}
238
32746a09 239## add member
c51eabc5 240
3a2e7c1c 241method add ($new) {
242 $self->_add_to_store($new);
243 $self->_add_to_caches($new);
9f2b6cc8 244 $self->_notify_observers(add => $new);
c51eabc5 245 $new
3a2e7c1c 246}
247
248method _add_to_store ($new) {
249 my $new_raw = $self->_deflate($new);
250 $self->_merge($new, $self->_store->new_insert_command($new_raw)->execute);
c51eabc5 251 $new
3a2e7c1c 252}
253
254method _add_to_caches ($new) {
255 $self->_add_to_member_cache($new);
256 $self->_add_to_key_cache($new);
de9534fa 257 $new
3a2e7c1c 258}
259
32746a09 260## remove member
c51eabc5 261
262method remove ($old) {
263 $self->_remove_from_store($old);
264 $self->_remove_from_caches($old);
9f2b6cc8 265 $self->_notify_observers(remove => $old);
c51eabc5 266 $old
267}
268
269method _remove_from_store ($old) {
a1e15ee1 270 $self->_store->new_delete_single_command($self->_deflate($old))->execute
c51eabc5 271}
272
273method _remove_from_caches ($old) {
274 $self->_remove_from_member_cache($old);
275 $self->_remove_from_key_cache($old);
276 $old
277}
278
32746a09 279## update member
48d91d77 280
281method _update_in_store ($obj) {
282 # this is currently a call command but we should think about it
283 # being a row command so that we can have RETURNING or other
284 # mechanisms handle things like set-on-update datetime values
a1e15ee1 285 $self->_store->new_update_single_command($self->_deflate($obj))->execute
48d91d77 286}
287
32746a09 288# I do wonder if we needed _merge_spec or if we'd be better off with
289# just using the raw merge routine ...
290
291method _update_set_in_store ($spec) {
292 $self->_store->new_update_command($self->_deflate_spec($spec))->execute;
293 if ($self->_member_cache_built) {
294 my $cache = $self->_member_cache;
295 foreach my $obj (@{$cache}) {
296 $self->_merge_spec($obj, $spec);
297 }
298 $self->_notify_observers(all_members => $cache);
299 }
300 return
301}
302
303method _remove_set_from_store {
304 $self->_store->new_delete_command->execute;
305 $self->_set_caches([]);
306 $self->_notify_observers(all_members => []);
307 return
308}
309
65b76960 3101;