Classes in this File | Line Coverage | Branch Coverage | Complexity | ||||
CollectionProxyDefaultImpl |
|
| 1.9;1.9 |
1 | package org.apache.ojb.broker.core.proxy; | |
2 | ||
3 | /* Copyright 2002-2005 The Apache Software Foundation | |
4 | * | |
5 | * Licensed under the Apache License, Version 2.0 (the "License"); | |
6 | * you may not use this file except in compliance with the License. | |
7 | * You may obtain a copy of the License at | |
8 | * | |
9 | * http://www.apache.org/licenses/LICENSE-2.0 | |
10 | * | |
11 | * Unless required by applicable law or agreed to in writing, software | |
12 | * distributed under the License is distributed on an "AS IS" BASIS, | |
13 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
14 | * See the License for the specific language governing permissions and | |
15 | * limitations under the License. | |
16 | */ | |
17 | ||
18 | import java.util.ArrayList; | |
19 | import java.util.Collection; | |
20 | import java.util.Iterator; | |
21 | ||
22 | import org.apache.ojb.broker.ManageableCollection; | |
23 | import org.apache.ojb.broker.OJBRuntimeException; | |
24 | import org.apache.ojb.broker.PBFactoryException; | |
25 | import org.apache.ojb.broker.PBKey; | |
26 | import org.apache.ojb.broker.PersistenceBroker; | |
27 | import org.apache.ojb.broker.PersistenceBrokerException; | |
28 | import org.apache.ojb.broker.PersistenceBrokerFactory; | |
29 | import org.apache.ojb.broker.metadata.MetadataManager; | |
30 | import org.apache.ojb.broker.metadata.MetadataException; | |
31 | import org.apache.ojb.broker.core.PersistenceBrokerThreadMapping; | |
32 | import org.apache.ojb.broker.query.Query; | |
33 | import org.apache.ojb.broker.util.collections.IRemovalAwareCollection; | |
34 | import org.apache.ojb.broker.util.collections.RemovalAwareCollection; | |
35 | ||
36 | /** | |
37 | * A place holder for a whole collection to support deferred loading of relationships. | |
38 | * The complete collection is loaded on access to the data. | |
39 | * | |
40 | * @author <a href="mailto:jbraeuchi@hotmail.com">Jakob Braeuchi</a> | |
41 | * @version $Id: CollectionProxyDefaultImpl.java,v 1.1 2007-08-24 22:17:32 ewestfal Exp $ | |
42 | */ | |
43 | public class CollectionProxyDefaultImpl implements Collection, ManageableCollection, CollectionProxy | |
44 | { | |
45 | /** The key for acquiring the above broker */ | |
46 | private PBKey _brokerKey; | |
47 | /** Flag set when per-thread metadata profiles are in use. */ | |
48 | private boolean _perThreadDescriptorsEnabled; | |
49 | /** Profile key used when lazy-loading with per-thread metadata profiles. */ | |
50 | private Object _profileKey; | |
51 | /** The query that defines the values in the collection */ | |
52 | private Query _query; | |
53 | /** The actual data (if already loaded) */ | |
54 | private Collection _data; | |
55 | /** The collection type */ | |
56 | private Class _collectionClass; | |
57 | /** The number of objects */ | |
58 | private int _size = -1; | |
59 | /* | |
60 | arminw | |
61 | fix a bug, caused by closing PB instances | |
62 | obtained from PersistenceBrokerThreadMapping. | |
63 | TODO: Could we find a better solution for this? | |
64 | */ | |
65 | private boolean _needsClose; | |
66 | /** Objects that listen on this proxy for loading events */ | |
67 | private transient ArrayList _listeners; | |
68 | ||
69 | /** | |
70 | * Creates a new collection proxy (uses | |
71 | * {@link org.apache.ojb.broker.util.collections.RemovalAwareCollection} | |
72 | * as the collection class). | |
73 | * | |
74 | * @param brokerKey The key of the persistence broker | |
75 | * @param query The defining query | |
76 | */ | |
77 | public CollectionProxyDefaultImpl(PBKey brokerKey, Query query) | |
78 | { | |
79 | this(brokerKey, RemovalAwareCollection.class, query); | |
80 | } | |
81 | ||
82 | /** | |
83 | * Creates a new collection proxy that uses the given collection type. | |
84 | * | |
85 | * @param brokerKey The key of the persistence broker | |
86 | * @param collClass The collection type | |
87 | * @param query The defining query | |
88 | */ | |
89 | public CollectionProxyDefaultImpl(PBKey brokerKey, Class collClass, Query query) | |
90 | { | |
91 | MetadataManager mm = MetadataManager.getInstance(); | |
92 | _perThreadDescriptorsEnabled = mm.isEnablePerThreadChanges(); | |
93 | if (_perThreadDescriptorsEnabled) | |
94 | { | |
95 | // mkalen: To minimize memory footprint we remember only the OJB profile key | |
96 | // (instead of all active class-mappings). | |
97 | final Object key = mm.getCurrentProfileKey(); | |
98 | if (key == null) | |
99 | { | |
100 | // mkalen: Unsupported: using proxies with per-thread metadata changes without profile keys. | |
101 | throw new MetadataException("Trying to create a Collection proxy with per-thread metadata changes enabled, but no profile key."); | |
102 | } | |
103 | setProfileKey(key); | |
104 | } | |
105 | setBrokerKey(brokerKey); | |
106 | setCollectionClass(collClass); | |
107 | setQuery(query); | |
108 | } | |
109 | ||
110 | /** | |
111 | * Reactivates metadata profile used when creating proxy, if needed. | |
112 | * Calls to this method should be guarded by checking | |
113 | * {@link #_perThreadDescriptorsEnabled} since the profile never | |
114 | * needs to be reloaded if not using pre-thread metadata changes. | |
115 | */ | |
116 | protected void loadProfileIfNeeded() | |
117 | { | |
118 | final Object key = getProfileKey(); | |
119 | if (key != null) | |
120 | { | |
121 | final MetadataManager mm = MetadataManager.getInstance(); | |
122 | if (!key.equals(mm.getCurrentProfileKey())) | |
123 | { | |
124 | mm.loadProfile(key); | |
125 | } | |
126 | } | |
127 | } | |
128 | ||
129 | /** | |
130 | * Determines whether the collection data already has been loaded from the database. | |
131 | * | |
132 | * @return <code>true</code> if the data is already loaded | |
133 | */ | |
134 | public boolean isLoaded() | |
135 | { | |
136 | return _data != null; | |
137 | } | |
138 | ||
139 | /** | |
140 | * Determines the number of elements that the query would return. Override this | |
141 | * method if the size shall be determined in a specific way. | |
142 | * | |
143 | * @return The number of elements | |
144 | */ | |
145 | protected synchronized int loadSize() throws PersistenceBrokerException | |
146 | { | |
147 | PersistenceBroker broker = getBroker(); | |
148 | try | |
149 | { | |
150 | return broker.getCount(getQuery()); | |
151 | } | |
152 | catch (Exception ex) | |
153 | { | |
154 | throw new PersistenceBrokerException(ex); | |
155 | } | |
156 | finally | |
157 | { | |
158 | releaseBroker(broker); | |
159 | } | |
160 | } | |
161 | ||
162 | /** | |
163 | * Sets the size internally. | |
164 | * | |
165 | * @param size The new size | |
166 | */ | |
167 | protected synchronized void setSize(int size) | |
168 | { | |
169 | _size = size; | |
170 | } | |
171 | ||
172 | /** | |
173 | * Loads the data from the database. Override this method if the objects | |
174 | * shall be loaded in a specific way. | |
175 | * | |
176 | * @return The loaded data | |
177 | */ | |
178 | protected Collection loadData() throws PersistenceBrokerException | |
179 | { | |
180 | PersistenceBroker broker = getBroker(); | |
181 | try | |
182 | { | |
183 | Collection result; | |
184 | ||
185 | if (_data != null) // could be set by listener | |
186 | { | |
187 | result = _data; | |
188 | } | |
189 | else if (_size != 0) | |
190 | { | |
191 | // TODO: returned ManageableCollection should extend Collection to avoid | |
192 | // this cast | |
193 | result = (Collection) broker.getCollectionByQuery(getCollectionClass(), getQuery()); | |
194 | } | |
195 | else | |
196 | { | |
197 | result = (Collection)getCollectionClass().newInstance(); | |
198 | } | |
199 | return result; | |
200 | } | |
201 | catch (Exception ex) | |
202 | { | |
203 | throw new PersistenceBrokerException(ex); | |
204 | } | |
205 | finally | |
206 | { | |
207 | releaseBroker(broker); | |
208 | } | |
209 | } | |
210 | ||
211 | /** | |
212 | * Notifies all listeners that the data is about to be loaded. | |
213 | */ | |
214 | protected void beforeLoading() | |
215 | { | |
216 | if (_listeners != null) | |
217 | { | |
218 | CollectionProxyListener listener; | |
219 | ||
220 | if (_perThreadDescriptorsEnabled) { | |
221 | loadProfileIfNeeded(); | |
222 | } | |
223 | for (int idx = _listeners.size() - 1; idx >= 0; idx--) | |
224 | { | |
225 | listener = (CollectionProxyListener)_listeners.get(idx); | |
226 | listener.beforeLoading(this); | |
227 | } | |
228 | } | |
229 | } | |
230 | ||
231 | /** | |
232 | * Notifies all listeners that the data has been loaded. | |
233 | */ | |
234 | protected void afterLoading() | |
235 | { | |
236 | if (_listeners != null) | |
237 | { | |
238 | CollectionProxyListener listener; | |
239 | ||
240 | if (_perThreadDescriptorsEnabled) { | |
241 | loadProfileIfNeeded(); | |
242 | } | |
243 | for (int idx = _listeners.size() - 1; idx >= 0; idx--) | |
244 | { | |
245 | listener = (CollectionProxyListener)_listeners.get(idx); | |
246 | listener.afterLoading(this); | |
247 | } | |
248 | } | |
249 | } | |
250 | ||
251 | /** | |
252 | * @see Collection#size() | |
253 | */ | |
254 | public int size() | |
255 | { | |
256 | if (isLoaded()) | |
257 | { | |
258 | return getData().size(); | |
259 | } | |
260 | else | |
261 | { | |
262 | if (_size < 0) | |
263 | { | |
264 | _size = loadSize(); | |
265 | } | |
266 | return _size; | |
267 | } | |
268 | } | |
269 | ||
270 | /** | |
271 | * @see Collection#isEmpty() | |
272 | */ | |
273 | public boolean isEmpty() | |
274 | { | |
275 | return size() == 0; | |
276 | } | |
277 | ||
278 | /** | |
279 | * @see Collection#contains(Object) | |
280 | */ | |
281 | public boolean contains(Object o) | |
282 | { | |
283 | return getData().contains(o); | |
284 | } | |
285 | ||
286 | /** | |
287 | * @see Collection#iterator() | |
288 | */ | |
289 | public Iterator iterator() | |
290 | { | |
291 | return getData().iterator(); | |
292 | } | |
293 | ||
294 | /** | |
295 | * @see Collection#toArray() | |
296 | */ | |
297 | public Object[] toArray() | |
298 | { | |
299 | return getData().toArray(); | |
300 | } | |
301 | ||
302 | /** | |
303 | * @see Collection#toArray(Object[]) | |
304 | */ | |
305 | public Object[] toArray(Object[] a) | |
306 | { | |
307 | return getData().toArray(a); | |
308 | } | |
309 | ||
310 | /** | |
311 | * @see Collection#add(Object) | |
312 | */ | |
313 | public boolean add(Object o) | |
314 | { | |
315 | return getData().add(o); | |
316 | } | |
317 | ||
318 | /** | |
319 | * @see Collection#remove(Object) | |
320 | */ | |
321 | public boolean remove(Object o) | |
322 | { | |
323 | return getData().remove(o); | |
324 | } | |
325 | ||
326 | /** | |
327 | * @see Collection#containsAll(Collection) | |
328 | */ | |
329 | public boolean containsAll(Collection c) | |
330 | { | |
331 | return getData().containsAll(c); | |
332 | } | |
333 | ||
334 | /** | |
335 | * @see Collection#addAll(Collection) | |
336 | */ | |
337 | public boolean addAll(Collection c) | |
338 | { | |
339 | return getData().addAll(c); | |
340 | } | |
341 | ||
342 | /** | |
343 | * @see Collection#removeAll(Collection) | |
344 | */ | |
345 | public boolean removeAll(Collection c) | |
346 | { | |
347 | return getData().removeAll(c); | |
348 | } | |
349 | ||
350 | /** | |
351 | * @see Collection#retainAll(Collection) | |
352 | */ | |
353 | public boolean retainAll(Collection c) | |
354 | { | |
355 | return getData().retainAll(c); | |
356 | } | |
357 | ||
358 | /** | |
359 | * Clears the proxy. A cleared proxy is defined as loaded | |
360 | * | |
361 | * @see Collection#clear() | |
362 | */ | |
363 | public void clear() | |
364 | { | |
365 | Class collClass = getCollectionClass(); | |
366 | ||
367 | // ECER: assure we notify all objects being removed, | |
368 | // necessary for RemovalAwareCollections... | |
369 | if (IRemovalAwareCollection.class.isAssignableFrom(collClass)) | |
370 | { | |
371 | getData().clear(); | |
372 | } | |
373 | else | |
374 | { | |
375 | Collection coll; | |
376 | // BRJ: use an empty collection so isLoaded will return true | |
377 | // for non RemovalAwareCollections only !! | |
378 | try | |
379 | { | |
380 | coll = (Collection) collClass.newInstance(); | |
381 | } | |
382 | catch (Exception e) | |
383 | { | |
384 | coll = new ArrayList(); | |
385 | } | |
386 | ||
387 | setData(coll); | |
388 | } | |
389 | _size = 0; | |
390 | } | |
391 | ||
392 | /** | |
393 | * Returns the defining query. | |
394 | * | |
395 | * @return The query | |
396 | */ | |
397 | public Query getQuery() | |
398 | { | |
399 | return _query; | |
400 | } | |
401 | ||
402 | /** | |
403 | * Sets the defining query. | |
404 | * | |
405 | * @param query The query | |
406 | */ | |
407 | protected void setQuery(Query query) | |
408 | { | |
409 | _query = query; | |
410 | } | |
411 | ||
412 | /** | |
413 | * Release the broker instance. | |
414 | */ | |
415 | protected synchronized void releaseBroker(PersistenceBroker broker) | |
416 | { | |
417 | /* | |
418 | arminw: | |
419 | only close the broker instance if we get | |
420 | it from the PBF, do nothing if we obtain it from | |
421 | PBThreadMapping | |
422 | */ | |
423 | if (broker != null && _needsClose) | |
424 | { | |
425 | _needsClose = false; | |
426 | broker.close(); | |
427 | } | |
428 | } | |
429 | ||
430 | /** | |
431 | * Acquires a broker instance. If no PBKey is available a runtime exception will be thrown. | |
432 | * | |
433 | * @return A broker instance | |
434 | */ | |
435 | protected synchronized PersistenceBroker getBroker() throws PBFactoryException | |
436 | { | |
437 | /* | |
438 | mkalen: | |
439 | NB! The loadProfileIfNeeded must be called _before_ acquiring a broker below, | |
440 | since some methods in PersistenceBrokerImpl will keep a local reference to | |
441 | the descriptor repository that was active during broker construction/refresh | |
442 | (not checking the repository beeing used on method invocation). | |
443 | ||
444 | PersistenceBrokerImpl#getClassDescriptor(Class clazz) is such a method, | |
445 | that will throw ClassNotPersistenceCapableException on the following scenario: | |
446 | ||
447 | (All happens in one thread only): | |
448 | t0: activate per-thread metadata changes | |
449 | t1: load, register and activate profile A | |
450 | t2: load object O1 witch collection proxy C to objects {O2} (C stores profile key K(A)) | |
451 | t3: close broker from t2 | |
452 | t4: load, register and activate profile B | |
453 | t5: reference O1.getO2Collection, causing C loadData() to be invoked | |
454 | t6: C calls getBroker | |
455 | broker B is created and descriptorRepository is set to descriptors from profile B | |
456 | t7: C calls loadProfileIfNeeded, re-activating profile A | |
457 | t8: C calls B.getCollectionByQuery | |
458 | t9: B gets callback (via QueryReferenceBroker) to getClassDescriptor | |
459 | the local descriptorRepository from t6 is used! | |
460 | => We will now try to query for {O2} with profile B | |
461 | (even though we re-activated profile A in t7) | |
462 | => ClassNotPersistenceCapableException | |
463 | ||
464 | Keeping loadProfileIfNeeded() at the start of this method changes everything from t6: | |
465 | t6: C calls loadProfileIfNeeded, re-activating profile A | |
466 | t7: C calls getBroker, | |
467 | broker B is created and descriptorRepository is set to descriptors from profile A | |
468 | t8: C calls B.getCollectionByQuery | |
469 | t9: B gets callback to getClassDescriptor, | |
470 | the local descriptorRepository from t6 is used | |
471 | => We query for {O2} with profile A | |
472 | => All good :-) | |
473 | */ | |
474 | if (_perThreadDescriptorsEnabled) | |
475 | { | |
476 | loadProfileIfNeeded(); | |
477 | } | |
478 | ||
479 | PersistenceBroker broker; | |
480 | if (getBrokerKey() == null) | |
481 | { | |
482 | /* | |
483 | arminw: | |
484 | if no PBKey is set we throw an exception, because we don't | |
485 | know which PB (connection) should be used. | |
486 | */ | |
487 | throw new OJBRuntimeException("Can't find associated PBKey. Need PBKey to obtain a valid" + | |
488 | "PersistenceBroker instance from intern resources."); | |
489 | } | |
490 | // first try to use the current threaded broker to avoid blocking | |
491 | broker = PersistenceBrokerThreadMapping.currentPersistenceBroker(getBrokerKey()); | |
492 | // current broker not found or was closed, create a intern new one | |
493 | if (broker == null || broker.isClosed()) | |
494 | { | |
495 | broker = PersistenceBrokerFactory.createPersistenceBroker(getBrokerKey()); | |
496 | // signal that we use a new internal obtained PB instance to read the | |
497 | // data and that this instance have to be closed after use | |
498 | _needsClose = true; | |
499 | } | |
500 | return broker; | |
501 | } | |
502 | ||
503 | /** | |
504 | * Returns the collection data, load it if not already done so. | |
505 | * | |
506 | * @return The data | |
507 | */ | |
508 | public synchronized Collection getData() | |
509 | { | |
510 | if (!isLoaded()) | |
511 | { | |
512 | beforeLoading(); | |
513 | setData(loadData()); | |
514 | afterLoading(); | |
515 | } | |
516 | return _data; | |
517 | } | |
518 | ||
519 | /** | |
520 | * Sets the collection data. | |
521 | * | |
522 | * @param data The data | |
523 | */ | |
524 | public void setData(Collection data) | |
525 | { | |
526 | _data = data; | |
527 | } | |
528 | ||
529 | /** | |
530 | * Returns the collection type. | |
531 | * | |
532 | * @return The collection type | |
533 | */ | |
534 | public Class getCollectionClass() | |
535 | { | |
536 | return _collectionClass; | |
537 | } | |
538 | ||
539 | /** | |
540 | * Sets the collection type. | |
541 | * | |
542 | * @param collClass The collection type | |
543 | */ | |
544 | protected void setCollectionClass(Class collClass) | |
545 | { | |
546 | _collectionClass = collClass; | |
547 | } | |
548 | ||
549 | /** | |
550 | * @see org.apache.ojb.broker.ManageableCollection#ojbAdd(Object) | |
551 | */ | |
552 | public void ojbAdd(Object anObject) | |
553 | { | |
554 | add(anObject); | |
555 | } | |
556 | ||
557 | /** | |
558 | * @see org.apache.ojb.broker.ManageableCollection#ojbAddAll(ManageableCollection) | |
559 | */ | |
560 | public void ojbAddAll(ManageableCollection otherCollection) | |
561 | { | |
562 | addAll((CollectionProxyDefaultImpl)otherCollection); | |
563 | } | |
564 | ||
565 | /** | |
566 | * @see org.apache.ojb.broker.ManageableCollection#ojbIterator() | |
567 | */ | |
568 | public Iterator ojbIterator() | |
569 | { | |
570 | return iterator(); | |
571 | } | |
572 | ||
573 | /** | |
574 | * @see org.apache.ojb.broker.ManageableCollection#afterStore(PersistenceBroker broker) | |
575 | */ | |
576 | public void afterStore(PersistenceBroker broker) throws PersistenceBrokerException | |
577 | { | |
578 | // If the real subject is a ManageableCollection | |
579 | // the afterStore() callback must be invoked ! | |
580 | Collection c = getData(); | |
581 | ||
582 | if (c instanceof ManageableCollection) | |
583 | { | |
584 | ((ManageableCollection)c).afterStore(broker); | |
585 | } | |
586 | } | |
587 | ||
588 | /** | |
589 | * Returns the key of the persistence broker used by this collection. | |
590 | * | |
591 | * @return The broker key | |
592 | */ | |
593 | public PBKey getBrokerKey() | |
594 | { | |
595 | return _brokerKey; | |
596 | } | |
597 | ||
598 | /** | |
599 | * Sets the key of the persistence broker used by this collection. | |
600 | * | |
601 | * @param brokerKey The key of the broker | |
602 | */ | |
603 | protected void setBrokerKey(PBKey brokerKey) | |
604 | { | |
605 | _brokerKey = brokerKey; | |
606 | } | |
607 | ||
608 | /** | |
609 | * Returns the metadata profile key used when creating this proxy. | |
610 | * | |
611 | * @return brokerKey The key of the broker | |
612 | */ | |
613 | protected Object getProfileKey() | |
614 | { | |
615 | return _profileKey; | |
616 | } | |
617 | ||
618 | /** | |
619 | * Sets the metadata profile key used when creating this proxy. | |
620 | * | |
621 | * @param profileKey the metadata profile key | |
622 | */ | |
623 | public void setProfileKey(Object profileKey) | |
624 | { | |
625 | _profileKey = profileKey; | |
626 | } | |
627 | ||
628 | /** | |
629 | * Adds a listener to this collection. | |
630 | * | |
631 | * @param listener The listener to add | |
632 | */ | |
633 | public synchronized void addListener(CollectionProxyListener listener) | |
634 | { | |
635 | if (_listeners == null) | |
636 | { | |
637 | _listeners = new ArrayList(); | |
638 | } | |
639 | // to avoid multi-add of same listener, do check | |
640 | if(!_listeners.contains(listener)) | |
641 | { | |
642 | _listeners.add(listener); | |
643 | } | |
644 | } | |
645 | ||
646 | /** | |
647 | * Removes the given listener from this collecton. | |
648 | * | |
649 | * @param listener The listener to remove | |
650 | */ | |
651 | public synchronized void removeListener(CollectionProxyListener listener) | |
652 | { | |
653 | if (_listeners != null) | |
654 | { | |
655 | _listeners.remove(listener); | |
656 | } | |
657 | } | |
658 | ||
659 | } |