1 /**
2 * Copyright 2005-2014 The Kuali Foundation
3 *
4 * Licensed under the Educational Community License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.opensource.org/licenses/ecl2.php
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16 package org.kuali.rice.krad.document;
17
18 import org.kuali.rice.core.api.mo.common.GloballyUnique;
19 import org.kuali.rice.kew.api.action.ActionType;
20 import org.kuali.rice.kew.framework.postprocessor.ActionTakenEvent;
21 import org.kuali.rice.kew.framework.postprocessor.DocumentRouteLevelChange;
22 import org.kuali.rice.kew.framework.postprocessor.DocumentRouteStatusChange;
23 import org.kuali.rice.kim.api.identity.Person;
24 import org.kuali.rice.krad.bo.AdHocRoutePerson;
25 import org.kuali.rice.krad.bo.AdHocRouteWorkgroup;
26 import org.kuali.rice.krad.bo.DocumentHeader;
27 import org.kuali.rice.krad.bo.Note;
28 import org.kuali.rice.krad.document.authorization.PessimisticLock;
29 import org.kuali.rice.krad.rules.rule.event.DocumentEvent;
30 import org.kuali.rice.krad.util.NoteType;
31 import org.kuali.rice.krad.util.documentserializer.PropertySerializabilityEvaluator;
32
33 import java.util.List;
34
35
36
37 /**
38 * This is the Document interface. All entities that are regarded as "eDocs" in the system, including Maintenance documents and
39 * Transaction Processing documents should implement this interface as it defines methods that are necessary to interact with the
40 * underlying frameworks and components (i.e. notes, attachments, workflow, etc).
41 *
42 * @author Kuali Rice Team (rice.collab@kuali.org)
43 */
44 public interface Document extends GloballyUnique {
45
46 /**
47 * This retrieves the standard {@code DocumentHeader} object, which contains standard meta-data about a document.
48 *
49 * @return document header since all docs will have a document header
50 */
51 DocumentHeader getDocumentHeader();
52
53 /**
54 * Sets the associated {@code DocumentHeader} for this document.
55 *
56 * @param documentHeader
57 */
58 void setDocumentHeader(DocumentHeader documentHeader);
59
60 /**
61 * All documents have a document header id. This is the quick accessor to that unique identifier and should return the same
62 * value as documentHeader.getDocumentHeaderId().
63 *
64 * @return doc header id
65 */
66 String getDocumentNumber();
67
68 /**
69 * setter for document header id
70 *
71 * @param documentHeaderId
72 */
73 void setDocumentNumber(String documentHeaderId);
74
75 /**
76
77 * This is the method to integrate with workflow, where we will actually populate the workflow defined data structure(s) so that
78 * workflow can routed based on this data. This method is responsible for passing over the proper Kuali (client system) data
79 * that will be used by workflow to determine how the document is actually routed.
80 */
81 void populateDocumentForRouting();
82
83 /**
84 * This is a method where we can get the xml of a document that the workflow system will use to base it's routing and search
85 * attributes on.
86 *
87 * @return the document serialized to an xml string
88 */
89 String serializeDocumentToXml();
90
91 /**
92 * This method is used to get the xml that should be used in a Route Report. In it's default implementation this will call the
93 * methods prepareForSave() and populateDocumentForRouting().
94 *
95 * @return XML data for routing
96 */
97 String getXmlForRouteReport();
98
99 /**
100 * method to integrate with workflow, where we will actually handle the transitions of levels for documents
101 *
102 * @param levelChangeEvent route level change event
103 */
104 void doRouteLevelChange(DocumentRouteLevelChange levelChangeEvent);
105
106 /**
107 * method to integrate with workflow where we will be able to perform logic for an action taken being performed on a document
108 *
109 * @param event action taken event
110 */
111 void doActionTaken(ActionTakenEvent event);
112
113 /**
114 * method to integrate with workflow where we will be able to perform logic after an action taken being performed on a document
115 *
116 * @param performed action type performed
117 * @param event action taken event
118 * @since 2.1
119 */
120 public void afterActionTaken(ActionType performed, ActionTakenEvent event);
121
122 /**
123 * This method will be called after the Workflow engine has completely finished processing a document.
124 *
125 * @param successfullyProcessed - true if the document was processed successfully, false otherwise
126 */
127 void afterWorkflowEngineProcess(boolean successfullyProcessed);
128
129 /**
130 * This method will be called before the Workflow engine has begun processing a document.
131 */
132 void beforeWorkflowEngineProcess();
133
134 /**
135 * This method will be called before the Workflow engine has begun processing a document.
136 * @return additional document IDs to lock prior to processing
137 */
138 List<String> getWorkflowEngineDocumentIdsToLock();
139
140 /**
141 * Getter method to get the document title as it will appear in and be searchable in workflow.
142 *
143 * @return document title
144 */
145 String getDocumentTitle();
146
147 /**
148 * getter method to get the list of ad hoc route persons associated with a document at a point in time, this list is only valid
149 * for a given users version of a document as this state is only persisted in workflow itself when someone takes an action on a
150 * document
151 *
152 * @return list of ad hoc route persons
153 */
154 List<AdHocRoutePerson> getAdHocRoutePersons();
155
156 /**
157 * getter method to get the list of ad hoc route workgroups associated with a document at a point in time, this list is only
158 * valid for a given users version of a document as this state is only persisted in workflow itself when someone takes an action
159 * on a document
160 *
161 * @return list of ad hoc route workgroups
162 */
163 List<AdHocRouteWorkgroup> getAdHocRouteWorkgroups();
164
165 /**
166 * setter method to set the list of ad hoc route persons associated with a document at a point in time, this list is only valid
167 * for a given users version of a document as this state is only persisted in workflow itself when someone takes an action on a
168 * document
169 *
170 * @param adHocRoutePersons
171 */
172 void setAdHocRoutePersons(List<AdHocRoutePerson> adHocRoutePersons);
173
174 /**
175 * setter method to set the list of ad hoc route workgroups associated with a document at a point in time, this list is only
176 * valid for a given users version of a document as this state is only persisted in workflow itself when someone takes an action
177 * on a document
178 *
179 * @param adHocRouteWorkgroups
180 */
181 void setAdHocRouteWorkgroups(List<AdHocRouteWorkgroup> adHocRouteWorkgroups);
182
183 /**
184 * This method provides a hook that will be called before the document is saved. This method is useful for applying document
185 * level data to children. For example, if someone changes data at the document level, and that data needs to be propagated to
186 * child objects or child lists of objects, you can use this method to update the child object or iterate through the list of
187 * child objects and apply the document level data to them. Any document that follows this paradigm will need to make use of
188 * this method to apply all of those changes.
189 */
190 void prepareForSave();
191
192 /**
193 * Sends document off to the rules engine to verify business rules.
194 *
195 * @param event - indicates which document event was requested
196 * @throws org.kuali.rice.krad.exception.ValidationException - containing the MessageMap from the validation session.
197 */
198 void validateBusinessRules(DocumentEvent event);
199
200 /**
201 * Do any work on the document that requires the DocumentEvent before the save.
202 *
203 * @param event - indicates which document event was requested
204 */
205 void prepareForSave(DocumentEvent event);
206
207 /**
208 * Do any work on the document after the save.
209 *
210 * @param event - indicates which document event was requested
211 */
212 void postProcessSave(DocumentEvent event);
213
214 /**
215 * This method provides a hook that will be called after a document is retrieved, but before it is returned from the
216 * DocumentService.
217 */
218 void processAfterRetrieve();
219
220 /**
221 * This method returns whether or not this document can be copied.
222 *
223 * @return True if it can be copied, false if not.
224 */
225 boolean getAllowsCopy();
226
227 /**
228 * Generate any necessary events required during the save event generation
229 *
230 * @return a list of document events that were triggered by the save event
231 */
232 List<DocumentEvent> generateSaveEvents();
233
234 /**
235 * Handle the doRouteStatusChange event from the post processor
236 * @param statusChangeEvent status change event
237 */
238 void doRouteStatusChange(DocumentRouteStatusChange statusChangeEvent);
239
240 /**
241 * Returns the note type which should be used for notes associated with this document.
242 * This method should never return null.
243 *
244 * @return the note type supported by this document, this value should never be null
245 */
246 NoteType getNoteType();
247
248 /**
249 * Return the target PersistableBusinessObject that notes associated with this document should be attached to.
250 * In general, this method should never return null. However, it is permissible that it will return a
251 * business object which has not been persisted yet (and therefore does not have it's unique object id
252 * established). This is only valid in cases where the note type is {@link NoteType#BUSINESS_OBJECT}.
253 *
254 * In these cases it's the responsibility for implementers of the Document interface to handle storing transient
255 * copies of the document notes (in XML or otherwise) until the underlying note target has been persisted and can be attached
256 * to the document's notes via it's object id.
257 *
258 * @return the PersistableBusinessObject with which notes on this document should be associated
259 */
260 GloballyUnique getNoteTarget();
261
262 /**
263 * Adds the given Note to the document's list of Notes.
264 *
265 * @param note the Note to add, must be non-null
266 */
267 void addNote(Note note);
268
269 /**
270 * Returns a mutable list of all notes on the document.
271 *
272 * @return the list of notes associated with this document, if this document has no notes then an empty list will be returned
273 */
274 List<Note> getNotes();
275
276 /**
277 * Sets the document's list of notes to the given list.
278 *
279 * @param notes the list of notes to set on the document, must be non-null
280 */
281 void setNotes(List<Note> notes);
282
283 /**
284 * Retrieves the note at the given index.
285 *
286 * @param index the zero-based index of the note to retrieve
287 * @return the note located at the given index
288 * @throws IndexOutOfBoundsException if the index is out of range
289 */
290 Note getNote(int index);
291
292 /**
293 * Removes the given note from the document's list of notes.
294 *
295 * @param note the note to remove from the document's list of notes, must be non-null
296 * @return true if the note was successfully removed, false if the list did not contain the given note
297 */
298 boolean removeNote(Note note);
299
300 /**
301 * This method gets a list of the {@link PessimisticLock} objects associated with this document
302 *
303 * @return list of pessimistic locks
304 */
305 List<PessimisticLock> getPessimisticLocks();
306
307 /**
308 * This method updates the list of {@link PessimisticLock} objects on the document if changes could
309 * have been made
310 */
311 void refreshPessimisticLocks();
312
313 /**
314 * This method adds a new {@link PessimisticLock} to the document
315 *
316 * NOTE: LOCKS ADDED VIA THIS METHOD WILL NOT BE SAVED WITH THE DOCUMENT
317 *
318 * @param lock - the lock to add to the document
319 */
320 void addPessimisticLock(PessimisticLock lock);
321
322 /**
323 * Renamed:
324 * @see Document#getLockClearingMethodNames()
325 * @return the list of method names of an action that should clear locks for the current user
326 */
327 @Deprecated
328 List<String> getLockClearningMethodNames();
329
330 /**
331 * This is a method that is used by Kuali Pessimistic Locking to get the names (method to call values)
332 * of the KNS KualiDocumentActionBase methods that should release locks
333 *
334 * @return the list of method names of an action that should clear locks for the current user
335 */
336 List<String> getLockClearingMethodNames();
337
338 /**
339 * Returns an evaluator object that determines whether a given property relative to the root object ({@link #wrapDocumentWithMetadataForXmlSerialization()}
340 * is serializable during the document serialization process.
341 *
342 * @return a fully initialized evaluator object, ready to be used for workflow routing
343 *
344 * @see org.kuali.rice.krad.service.DocumentSerializerService
345 * @see #wrapDocumentWithMetadataForXmlSerialization()
346 */
347
348 String getBasePathToDocumentDuringSerialization();
349
350 /**
351 * Returns an evaluator object that determines whether a given property relative to the root object ({@link #wrapDocumentWithMetadataForXmlSerialization()}
352 * is serializable during the document serialization process.
353 *
354 * @return a fully initialized evaluator object, ready to be used for workflow routing
355 *
356 * @see org.kuali.rice.krad.service.DocumentSerializerService
357 * @see #wrapDocumentWithMetadataForXmlSerialization()
358 */
359 PropertySerializabilityEvaluator getDocumentPropertySerizabilityEvaluator();
360
361 /**
362 * This method will return the root object to be serialized for workflow routing. If necessary, this method will wrap this document object with a wrapper (i.e. contains a reference back to this document). This
363 * wrapper may also contain references to additional objects that provide metadata useful to the workflow engine.
364 *
365 * If no wrappers are necessary, then this object may return "this"
366 *
367 * @return a wrapper object (most likely containing a reference to "this"), or "this" itself.
368 * @see org.kuali.rice.krad.workflow.KualiDocumentXmlMaterializer
369 */
370 Object wrapDocumentWithMetadataForXmlSerialization();
371
372 /**
373 * This method returns whether or not this document supports custom lock descriptors for pessimistic locking.
374 *
375 * @return True if the document can generate custom lock descriptors, false otherwise.
376 * @see #getCustomLockDescriptor(Person)
377 */
378 boolean useCustomLockDescriptors();
379
380 /**
381 * Generates a custom lock descriptor for pessimistic locking. This method should not be called unless {@link #useCustomLockDescriptors()} returns true.
382 *
383 * @param user The user trying to establish the lock.
384 * @return A String representing the lock descriptor.
385 * @see #useCustomLockDescriptors()
386 * @see org.kuali.rice.krad.service.PessimisticLockService
387 */
388 String getCustomLockDescriptor(Person user);
389 }