1 /**
2 *
3 * Copyright 2003-2004 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 package javax.mail;
19
20 import java.util.ArrayList;
21 import java.util.List;
22 import javax.mail.Flags.Flag;
23 import javax.mail.event.ConnectionEvent;
24 import javax.mail.event.ConnectionListener;
25 import javax.mail.event.FolderEvent;
26 import javax.mail.event.FolderListener;
27 import javax.mail.event.MessageChangedEvent;
28 import javax.mail.event.MessageChangedListener;
29 import javax.mail.event.MessageCountEvent;
30 import javax.mail.event.MessageCountListener;
31 import javax.mail.search.SearchTerm;
32
33 /**
34 * An abstract representation of a folder in a mail system; subclasses would
35 * implement Folders for each supported protocol.
36 * <p/>
37 * Depending on protocol and implementation, folders may contain other folders, messages,
38 * or both as indicated by the {@link Folder#HOLDS_FOLDERS} and {@link Folder#HOLDS_MESSAGES} flags.
39 * If the immplementation supports hierarchical folders, the format of folder names is
40 * implementation dependent; however, components of the name are separated by the
41 * delimiter character returned by {@link Folder#getSeparator()}.
42 * <p/>
43 * The case-insensitive folder name "INBOX" is reserved to refer to the primary folder
44 * for the current user on the current server; not all stores will provide an INBOX
45 * and it may not be available at all times.
46 *
47 * @version $Rev: 126350 $ $Date: 2005-01-24 22:35:47 -0800 (Mon, 24 Jan 2005) $
48 */
49 public abstract class Folder {
50 /**
51 * Flag that indicates that a folder can contain messages.
52 */
53 public static final int HOLDS_MESSAGES = 1;
54 /**
55 * Flag that indicates that a folder can contain other folders.
56 */
57 public static final int HOLDS_FOLDERS = 2;
58
59 /**
60 * Flag indicating that this folder cannot be modified.
61 */
62 public static final int READ_ONLY = 1;
63 /**
64 * Flag indictaing that this folder can be modified.
65 * Question: what does it mean if both are set?
66 */
67 public static final int READ_WRITE = 2;
68
69 /**
70 * The store that this folder is part of.
71 */
72 protected Store store;
73 /**
74 * The current mode of this folder.
75 * When open, this can be {@link #READ_ONLY} or {@link #READ_WRITE};
76 * otherwise is set to -1.
77 */
78 protected int mode = -1;
79
80 private final List connectionListeners = new ArrayList(2);
81 private final List folderListeners = new ArrayList(2);
82 private final List messageChangedListeners = new ArrayList(2);
83 private final List messageCountListeners = new ArrayList(2);
84 private final EventQueue queue = new EventQueue();
85
86 /**
87 * Constructor that initializes the Store.
88 *
89 * @param store the store that this folder is part of
90 */
91 protected Folder(Store store) {
92 this.store = store;
93 }
94
95 /**
96 * Return the name of this folder.
97 * This can be invoked when the folder is closed.
98 *
99 * @return this folder's name
100 */
101 public abstract String getName();
102
103 /**
104 * Return the full absolute name of this folder.
105 * This can be invoked when the folder is closed.
106 *
107 * @return the full name of this folder
108 */
109 public abstract String getFullName();
110
111 /**
112 * Return the URLName for this folder, which includes the location of the store.
113 *
114 * @return the URLName for this folder
115 * @throws MessagingException
116 */
117 public URLName getURLName() throws MessagingException {
118
119 return store.getURLName();
120 }
121
122 /**
123 * Return the store that this folder is part of.
124 *
125 * @return the store this folder is part of
126 */
127 public Store getStore() {
128 return store;
129 }
130
131 /**
132 * Return the parent for this folder; if the folder is at the root of a heirarchy
133 * this returns null.
134 * This can be invoked when the folder is closed.
135 *
136 * @return this folder's parent
137 * @throws MessagingException
138 */
139 public abstract Folder getParent() throws MessagingException;
140
141 /**
142 * Check to see if this folder physically exists in the store.
143 * This can be invoked when the folder is closed.
144 *
145 * @return true if the folder really exists
146 * @throws MessagingException if there was a problem accessing the store
147 */
148 public abstract boolean exists() throws MessagingException;
149
150 /**
151 * Return a list of folders from this Folder's namespace that match the supplied pattern.
152 * Patterns may contain the following wildcards:
153 * <ul><li>'%' which matches any characater except hierarchy delimiters</li>
154 * <li>'*' which matches any character including hierarchy delimiters</li>
155 * </ul>
156 * This can be invoked when the folder is closed.
157 *
158 * @param pattern the pattern to search for
159 * @return a, possibly empty, array containing Folders that matched the pattern
160 * @throws MessagingException if there was a problem accessing the store
161 */
162 public abstract Folder[] list(String pattern) throws MessagingException;
163
164 /**
165 * Return a list of folders to which the user is subscribed and which match the supplied pattern.
166 * If the store does not support the concept of subscription then this should match against
167 * all folders; the default implementation of this method achieves this by defaulting to the
168 * {@link #list(String)} method.
169 *
170 * @param pattern the pattern to search for
171 * @return a, possibly empty, array containing subscribed Folders that matched the pattern
172 * @throws MessagingException if there was a problem accessing the store
173 */
174 public Folder[] listSubscribed(String pattern) throws MessagingException {
175 return list(pattern);
176 }
177
178 /**
179 * Convenience method that invokes {@link #list(String)} with the pattern "%".
180 *
181 * @return a, possibly empty, array of subfolders
182 * @throws MessagingException if there was a problem accessing the store
183 */
184 public Folder[] list() throws MessagingException {
185 return list("%");
186 }
187
188 /**
189 * Convenience method that invokes {@link #listSubscribed(String)} with the pattern "%".
190 *
191 * @return a, possibly empty, array of subscribed subfolders
192 * @throws MessagingException if there was a problem accessing the store
193 */
194 public Folder[] listSubscribed() throws MessagingException {
195 return listSubscribed("%");
196 }
197
198 /**
199 * Return the character used by this folder's Store to separate path components.
200 *
201 * @return the name separater character
202 * @throws MessagingException if there was a problem accessing the store
203 */
204 public abstract char getSeparator() throws MessagingException;
205
206 /**
207 * Return the type of this folder, indicating whether it can contain subfolders,
208 * messages, or both. The value returned is a bitmask with the appropriate bits set.
209 *
210 * @return the type of this folder
211 * @throws MessagingException if there was a problem accessing the store
212 * @see #HOLDS_FOLDERS
213 * @see #HOLDS_MESSAGES
214 */
215 public abstract int getType() throws MessagingException;
216
217 /**
218 * Create a new folder capable of containing subfoldera and/or messages as
219 * determined by the type parameter. Any hierarchy defined by the folder
220 * name will be recursively created.
221 * If the folder was sucessfully created, a {@link FolderEvent#CREATED CREATED FolderEvent}
222 * is sent to all FolderListeners registered with this Folder or with the Store.
223 *
224 * @param type the type, indicating if this folder should contain subfolders, messages or both
225 * @return true if the folder was sucessfully created
226 * @throws MessagingException if there was a problem accessing the store
227 */
228 public abstract boolean create(int type) throws MessagingException;
229
230 /**
231 * Determine if the user is subscribed to this Folder. The default implementation in
232 * this class always returns true.
233 *
234 * @return true is the user is subscribed to this Folder
235 */
236 public boolean isSubscribed() {
237 return true;
238 }
239
240 /**
241 * Set the user's subscription to this folder.
242 * Not all Stores support subscription; the default implementation in this class
243 * always throws a MethodNotSupportedException
244 *
245 * @param subscribed whether to subscribe to this Folder
246 * @throws MessagingException if there was a problem accessing the store
247 * @throws MethodNotSupportedException if the Store does not support subscription
248 */
249 public void setSubscribed(boolean subscribed) throws MessagingException {
250 throw new MethodNotSupportedException();
251 }
252
253 /**
254 * Check to see if this Folder conatins messages with the {@link Flag.RECENT} flag set.
255 * This can be used when the folder is closed to perform a light-weight check for new mail;
256 * to perform an incremental check for new mail the folder must be opened.
257 *
258 * @return true if the Store has recent messages
259 * @throws MessagingException if there was a problem accessing the store
260 */
261 public abstract boolean hasNewMessages() throws MessagingException;
262
263 /**
264 * Get the Folder determined by the supplied name; if the name is relative
265 * then it is interpreted relative to this folder. This does not check that
266 * the named folder actually exists.
267 *
268 * @param name the name of the folder to return
269 * @return the named folder
270 * @throws MessagingException if there was a problem accessing the store
271 */
272 public abstract Folder getFolder(String name) throws MessagingException;
273
274 /**
275 * Delete this folder and possibly any subfolders. This operation can only be
276 * performed on a closed folder.
277 * If recurse is true, then all subfolders are deleted first, then any messages in
278 * this folder are removed and it is finally deleted; {@link FolderEvent#DELETED}
279 * events are sent as appropriate.
280 * If recurse is false, then the behaviour depends on the folder type and store
281 * implementation as followd:
282 * <ul>
283 * <li>If the folder can only conrain messages, then all messages are removed and
284 * then the folder is deleted; a {@link FolderEvent#DELETED} event is sent.</li>
285 * <li>If the folder can onlu contain subfolders, then if it is empty it will be
286 * deleted and a {@link FolderEvent#DELETED} event is sent; if the folder is not
287 * empty then the delete fails and this method returns false.</li>
288 * <li>If the folder can contain both subfolders and messages, then if the folder
289 * does not contain any subfolders, any messages are deleted, the folder itself
290 * is deleted and a {@link FolderEvent#DELETED} event is sent; if the folder does
291 * contain subfolders then the implementation may choose from the following three
292 * behaviors:
293 * <ol>
294 * <li>it may return false indicting the operation failed</li>
295 * <li>it may remove all messages within the folder, send a {@link FolderEvent#DELETED}
296 * event, and then return true to indicate the delete was performed. Note this does
297 * not delete the folder itself and the {@link #exists()} operation for this folder
298 * will return true</li>
299 * <li>it may remove all messages within the folder as per the previous option; in
300 * addition it may change the type of the Folder to only HOLDS_FOLDERS indictaing
301 * that messages may no longer be added</li>
302 * </li>
303 * </ul>
304 * FolderEvents are sent to all listeners registered with this folder or
305 * with the Store.
306 *
307 * @param recurse whether subfolders should be recursively deleted as well
308 * @return true if the delete operation succeeds
309 * @throws MessagingException if there was a problem accessing the store
310 */
311 public abstract boolean delete(boolean recurse) throws MessagingException;
312
313 /**
314 * Rename this folder; the folder must be closed.
315 * If the rename is successfull, a {@link FolderEvent#RENAMED} event is sent to
316 * all listeners registered with this folder or with the store.
317 *
318 * @param newName the new name for this folder
319 * @return true if the rename succeeded
320 * @throws MessagingException if there was a problem accessing the store
321 */
322 public abstract boolean renameTo(Folder newName) throws MessagingException;
323
324 /**
325 * Open this folder; the folder must be able to contain messages and
326 * must currently be closed. If the folder is opened successfully then
327 * a {@link ConnectionEvent#OPENED} event is sent to listeners registered
328 * with this Folder.
329 * <p/>
330 * Whether the Store allows multiple connections or if it allows multiple
331 * writers is implementation defined.
332 *
333 * @param mode READ_ONLY or READ_WRITE
334 * @throws MessagingException if there was a problem accessing the store
335 */
336 public abstract void open(int mode) throws MessagingException;
337
338 /**
339 * Close this folder; it must already be open.
340 * A {@link ConnectionEvent#CLOSED} event is sent to all listeners registered
341 * with this folder.
342 *
343 * @param expunge whether to expunge all deleted messages
344 * @throws MessagingException if there was a problem accessing the store; the folder is still closed
345 */
346 public abstract void close(boolean expunge) throws MessagingException;
347
348 /**
349 * Indicates that the folder has been opened.
350 *
351 * @return true if the folder is open
352 */
353 public abstract boolean isOpen();
354
355 /**
356 * Return the mode of this folder ass passed to {@link #open(int)}, or -1 if
357 * the folder is closed.
358 *
359 * @return the mode this folder was opened with
360 */
361 public int getMode() {
362 return mode;
363 }
364
365 /**
366 * Get the flags supported by this folder.
367 *
368 * @return the flags supported by this folder, or null if unknown
369 * @see Flags
370 */
371 public abstract Flags getPermanentFlags();
372
373 /**
374 * Return the number of messages this folder contains.
375 * If this operation is invoked on a closed folder, the implementation
376 * may choose to return -1 to avoid the expense of opening the folder.
377 *
378 * @return the number of messages, or -1 if unknown
379 * @throws MessagingException if there was a problem accessing the store
380 */
381 public abstract int getMessageCount() throws MessagingException;
382
383 /**
384 * Return the numbew of messages in this folder that have the {@link Flag.RECENT} flag set.
385 * If this operation is invoked on a closed folder, the implementation
386 * may choose to return -1 to avoid the expense of opening the folder.
387 * The default implmentation of this method iterates over all messages
388 * in the folder; subclasses should override if possible to provide a more
389 * efficient implementation.
390 *
391 * @return the number of new messages, or -1 if unknown
392 * @throws MessagingException if there was a problem accessing the store
393 */
394 public int getNewMessageCount() throws MessagingException {
395 return getCount(Flags.Flag.RECENT, true);
396 }
397
398 /**
399 * Return the numbew of messages in this folder that do not have the {@link Flag.SEEN} flag set.
400 * If this operation is invoked on a closed folder, the implementation
401 * may choose to return -1 to avoid the expense of opening the folder.
402 * The default implmentation of this method iterates over all messages
403 * in the folder; subclasses should override if possible to provide a more
404 * efficient implementation.
405 *
406 * @return the number of new messages, or -1 if unknown
407 * @throws MessagingException if there was a problem accessing the store
408 */
409 public int getUnreadMessageCount() throws MessagingException {
410 return getCount(Flags.Flag.SEEN, false);
411 }
412
413 /**
414 * Return the numbew of messages in this folder that have the {@link Flag.DELETED} flag set.
415 * If this operation is invoked on a closed folder, the implementation
416 * may choose to return -1 to avoid the expense of opening the folder.
417 * The default implmentation of this method iterates over all messages
418 * in the folder; subclasses should override if possible to provide a more
419 * efficient implementation.
420 *
421 * @return the number of new messages, or -1 if unknown
422 * @throws MessagingException if there was a problem accessing the store
423 */
424 public int getDeletedMessageCount() throws MessagingException {
425 return getCount(Flags.Flag.DELETED, true);
426 }
427
428 private int getCount(Flag flag, boolean value) throws MessagingException {
429 if (!isOpen()) {
430 return -1;
431 }
432 Message[] messages = getMessages();
433 int total = 0;
434 for (int i = 0; i < messages.length; i++) {
435 if (messages[i].getFlags().contains(flag) == value) {
436 total++;
437 }
438 }
439 return total;
440 }
441
442 /**
443 * Retrieve the message with the specified index in this Folder;
444 * messages indices start at 1 not zero.
445 * Clients should note that the index for a specific message may change
446 * if the folder is expunged; {@link Message} objects should be used as
447 * references instead.
448 *
449 * @param index the index of the message to fetch
450 * @return the message
451 * @throws MessagingException if there was a problem accessing the store
452 */
453 public abstract Message getMessage(int index) throws MessagingException;
454
455 /**
456 * Retrieve messages with index between start and end inclusive
457 *
458 * @param start index of first message
459 * @param end index of last message
460 * @return an array of messages from start to end inclusive
461 * @throws MessagingException if there was a problem accessing the store
462 */
463 public Message[] getMessages(int start, int end) throws MessagingException {
464 Message[] result = new Message[end - start + 1];
465 for (int i = 0; i < result.length; i++) {
466 result[i] = getMessage(start++);
467 }
468 return result;
469 }
470
471 /**
472 * Retrieve messages with the specified indices.
473 *
474 * @param ids the indices of the messages to fetch
475 * @return the specified messages
476 * @throws MessagingException if there was a problem accessing the store
477 */
478 public Message[] getMessages(int ids[]) throws MessagingException {
479 Message[] result = new Message[ids.length];
480 for (int i = 0; i < ids.length; i++) {
481 result[i] = getMessage(ids[i]);
482 }
483 return result;
484 }
485
486 /**
487 * Retrieve all messages.
488 *
489 * @return all messages in this folder
490 * @throws MessagingException if there was a problem accessing the store
491 */
492 public Message[] getMessages() throws MessagingException {
493 return getMessages(1, getMessageCount());
494 }
495
496 /**
497 * Append the supplied messages to this folder. A {@link MessageCountEvent} is sent
498 * to all listeners registered with this folder when all messages have been appended.
499 * If the array contains a previously expunged message, it must be re-appended to the Store
500 * and implementations must not abort this operation.
501 *
502 * @param messages the messages to append
503 * @throws MessagingException if there was a problem accessing the store
504 */
505 public abstract void appendMessages(Message[] messages) throws MessagingException;
506
507 /**
508 * Hint to the store to prefetch information on the supplied messaged.
509 * Subclasses should override this method to provide an efficient implementation;
510 * the default implementation in this class simply returns.
511 *
512 * @param messages messages for which information should be fetched
513 * @param profile the information to fetch
514 * @throws MessagingException if there was a problem accessing the store
515 * @see FetchProfile
516 */
517 public void fetch(Message[] messages, FetchProfile profile) throws MessagingException {
518 return;
519 }
520
521 /**
522 * Set flags on the messages to the supplied value; all messages must belong to this folder.
523 * This method may be overridden by subclasses that can optimize the setting
524 * of flags on multiple messages at once; the default implementation simply calls
525 * {@link Message#setFlags(Flags, boolean)} for each supplied messages.
526 *
527 * @param messages whose flags should be set
528 * @param flags the set of flags to modify
529 * @param value the value the flags should be set to
530 * @throws MessagingException if there was a problem accessing the store
531 */
532 public void setFlags(Message[] messages, Flags flags, boolean value) throws MessagingException {
533 for (int i = 0; i < messages.length; i++) {
534 Message message = messages[i];
535 message.setFlags(flags, value);
536 }
537 }
538
539 /**
540 * Set flags on a range of messages to the supplied value.
541 * This method may be overridden by subclasses that can optimize the setting
542 * of flags on multiple messages at once; the default implementation simply
543 * gets each message and then calls {@link Message#setFlags(Flags, boolean)}.
544 *
545 * @param start first message end set
546 * @param end last message end set
547 * @param flags the set of flags end modify
548 * @param value the value the flags should be set end
549 * @throws MessagingException if there was a problem accessing the store
550 */
551 public void setFlags(int start, int end, Flags flags, boolean value) throws MessagingException {
552 for (int i = start; i <= end; i++) {
553 Message message = getMessage(i);
554 message.setFlags(flags, value);
555 }
556 }
557
558 /**
559 * Set flags on a set of messages to the supplied value.
560 * This method may be overridden by subclasses that can optimize the setting
561 * of flags on multiple messages at once; the default implementation simply
562 * gets each message and then calls {@link Message#setFlags(Flags, boolean)}.
563 *
564 * @param ids the indexes of the messages to set
565 * @param flags the set of flags end modify
566 * @param value the value the flags should be set end
567 * @throws MessagingException if there was a problem accessing the store
568 */
569 public void setFlags(int ids[], Flags flags, boolean value) throws MessagingException {
570 for (int i = 0; i < ids.length; i++) {
571 Message message = getMessage(ids[i]);
572 message.setFlags(flags, value);
573 }
574 }
575
576 /**
577 * Copy the specified messages to another folder.
578 * The default implementation simply appends the supplied messages to the
579 * target folder using {@link #appendMessages(Message[])}.
580 * @param messages the messages to copy
581 * @param folder the folder to copy to
582 * @throws MessagingException if there was a problem accessing the store
583 */
584 public void copyMessages(Message[] messages, Folder folder) throws MessagingException {
585 folder.appendMessages(messages);
586 }
587
588 /**
589 * Permanently delete all supplied messages that have the DELETED flag set from the Store.
590 * The original message indices of all messages actually deleted are returned and a
591 * {@link MessageCountEvent} event is sent to all listeners with this folder. The expunge
592 * may cause the indices of all messaged that remain in the folder to change.
593 *
594 * @return the original indices of messages that were actually deleted
595 * @throws MessagingException if there was a problem accessing the store
596 */
597 public abstract Message[] expunge() throws MessagingException;
598
599 /**
600 * Search this folder for messages matching the supplied search criteria.
601 * The default implementation simply invoke <code>search(term, getMessages())
602 * applying the search over all messages in the folder; subclasses may provide
603 * a more efficient mechanism.
604 *
605 * @param term the search criteria
606 * @return an array containing messages that match the criteria
607 * @throws MessagingException if there was a problem accessing the store
608 */
609 public Message[] search(SearchTerm term) throws MessagingException {
610 return search(term, getMessages());
611 }
612
613 /**
614 * Search the supplied messages for those that match the supplied criteria;
615 * messages must belong to this folder.
616 * The default implementation iterates through the messages, returning those
617 * whose {@link Message#match(javax.mail.search.SearchTerm)} method returns true;
618 * subclasses may provide a more efficient implementation.
619 *
620 * @param term the search criteria
621 * @param messages the messages to search
622 * @return an array containing messages that match the criteria
623 * @throws MessagingException if there was a problem accessing the store
624 */
625 public Message[] search(SearchTerm term, Message[] messages) throws MessagingException {
626 List result = new ArrayList(messages.length);
627 for (int i = 0; i < messages.length; i++) {
628 Message message = messages[i];
629 if (message.match(term)) {
630 result.add(message);
631 }
632 }
633 return (Message[]) result.toArray(new Message[result.size()]);
634 }
635
636 public void addConnectionListener(ConnectionListener listener) {
637 connectionListeners.add(listener);
638 }
639
640 public void removeConnectionListener(ConnectionListener listener) {
641 connectionListeners.remove(listener);
642 }
643
644 protected void notifyConnectionListeners(int type) {
645 queue.queueEvent(new ConnectionEvent(this, type), connectionListeners);
646 }
647
648 public void addFolderListener(FolderListener listener) {
649 folderListeners.add(listener);
650 }
651
652 public void removeFolderListener(FolderListener listener) {
653 folderListeners.remove(listener);
654 }
655
656 protected void notifyFolderListeners(int type) {
657 queue.queueEvent(new FolderEvent(this, this, type), folderListeners);
658 }
659
660 protected void notifyFolderRenamedListeners(Folder newFolder) {
661 queue.queueEvent(new FolderEvent(this, this, newFolder, FolderEvent.RENAMED), folderListeners);
662 }
663
664 public void addMessageCountListener(MessageCountListener listener) {
665 messageCountListeners.add(listener);
666 }
667
668 public void removeMessageCountListener(MessageCountListener listener) {
669 messageCountListeners.remove(listener);
670 }
671
672 protected void notifyMessageAddedListeners(Message[] messages) {
673 queue.queueEvent(new MessageCountEvent(this, MessageCountEvent.ADDED, false, messages), messageChangedListeners);
674 }
675
676 protected void notifyMessageRemovedListeners(boolean removed, Message[] messages) {
677 queue.queueEvent(new MessageCountEvent(this, MessageCountEvent.REMOVED, removed, messages), messageChangedListeners);
678 }
679
680 public void addMessageChangedListener(MessageChangedListener listener) {
681 messageChangedListeners.add(listener);
682 }
683
684 public void removeMessageChangedListener(MessageChangedListener listener) {
685 messageChangedListeners.remove(listener);
686 }
687
688 protected void notifyMessageChangedListeners(int type, Message message) {
689 queue.queueEvent(new MessageChangedEvent(this, type, message), messageChangedListeners);
690 }
691
692 /**
693 * Unregisters all listeners.
694 */
695 protected void finalize() throws Throwable {
696 queue.stop();
697 connectionListeners.clear();
698 folderListeners.clear();
699 messageChangedListeners.clear();
700 messageCountListeners.clear();
701 store = null;
702 super.finalize();
703 }
704
705 /**
706 * Returns the full name of this folder; if null, returns the value from the superclass.
707 * @return a string form of this folder
708 */
709 public String toString() {
710 String name = getFullName();
711 return name == null ? super.toString() : name;
712 }
713 }