1 ;;; elmo-pop3.el --- POP3 Interface for ELMO.
3 ;; Copyright (C) 1998,1999,2000 Yuuichi Teranishi <teranisi@gohome.org>
4 ;; Copyright (C) 1999,2000 Kenichi OKADA <okada@opaopa.org>
6 ;; Author: Yuuichi Teranishi <teranisi@gohome.org>
7 ;; Kenichi OKADA <okada@opaopa.org>
8 ;; Keywords: mail, net news
10 ;; This file is part of ELMO (Elisp Library for Message Orchestration).
12 ;; This program is free software; you can redistribute it and/or modify
13 ;; it under the terms of the GNU General Public License as published by
14 ;; the Free Software Foundation; either version 2, or (at your option)
17 ;; This program is distributed in the hope that it will be useful,
18 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
19 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 ;; GNU General Public License for more details.
22 ;; You should have received a copy of the GNU General Public License
23 ;; along with GNU Emacs; see the file COPYING. If not, write to the
24 ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
25 ;; Boston, MA 02111-1307, USA.
42 (autoload 'md5 "md5"))
44 (defcustom elmo-pop3-default-use-uidl t
45 "If non-nil, use UIDL on POP3."
49 (defvar elmo-pop3-use-uidl-internal t
50 "(Internal switch for using UIDL on POP3).")
52 (defvar elmo-pop3-use-cache t
53 "Use cache in pop3 folder.")
55 (defvar elmo-pop3-send-command-synchronously nil
56 "If non-nil, commands are send synchronously.
57 If server doesn't accept asynchronous commands, this variable should be
60 (defcustom elmo-pop3-exists-exactly nil
61 "If non-nil, POP3 folder existence is checked everytime before the session."
65 (defconst elmo-pop3-folder-name-syntax `(([user ".+"])
67 (?: [uidl "^[A-Za-z]+$"])
68 ,@elmo-net-folder-name-syntax))
70 (defvar sasl-mechanism-alist)
72 (defvar elmo-pop3-retrieve-progress-reporter nil)
75 (defvar elmo-pop3-debug nil
76 "Non-nil forces POP3 folder as debug mode.
77 Debug information is inserted in the buffer \"*POP3 DEBUG*\"")
80 (defsubst elmo-pop3-debug (message &rest args)
82 (let ((biff (string-match "BIFF-" (buffer-name)))
84 (with-current-buffer (get-buffer-create (concat "*POP3 DEBUG*"
86 (goto-char (point-max))
88 (insert (apply 'format message args) "\n")))))
92 (luna-define-class elmo-pop3-folder (elmo-net-folder elmo-location-map)
94 (luna-define-internal-accessors 'elmo-pop3-folder))
96 (defsubst elmo-pop3-folder-use-uidl (folder)
97 (if elmo-inhibit-number-mapping
99 (elmo-pop3-folder-use-uidl-internal folder)))
101 (luna-define-method elmo-folder-initialize ((folder elmo-pop3-folder) name)
102 (let ((elmo-network-stream-type-alist
103 (if elmo-pop3-stream-type-alist
104 (append elmo-pop3-stream-type-alist
105 elmo-network-stream-type-alist)
106 elmo-network-stream-type-alist))
108 (setq tokens (car (elmo-parse-separated-tokens
110 elmo-pop3-folder-name-syntax)))
112 (elmo-net-folder-set-user-internal folder
113 (or (cdr (assq 'user tokens))
114 elmo-pop3-default-user))
116 (setq auth (cdr (assq 'auth tokens)))
117 (elmo-net-folder-set-auth-internal folder
119 (intern (downcase auth))
120 elmo-pop3-default-authenticate-type))
122 (setq uidl (cdr (assq 'uidl tokens)))
123 (elmo-pop3-folder-set-use-uidl-internal folder
125 (string= uidl "uidl")
126 elmo-pop3-default-use-uidl))
128 (elmo-net-folder-set-parameters
131 (list :server elmo-pop3-default-server
132 :port elmo-pop3-default-port
134 (elmo-get-network-stream-type elmo-pop3-default-stream-type)))
138 (luna-define-class elmo-pop3-session (elmo-network-session) ())
141 (defvar elmo-pop3-read-point nil)
142 (defvar elmo-pop3-number-uidl-hash nil) ; number -> uidl
143 (defvar elmo-pop3-uidl-number-hash nil) ; uidl -> number
144 (defvar elmo-pop3-size-hash nil) ; number -> size
145 (defvar elmo-pop3-uidl-done nil)
146 (defvar elmo-pop3-list-done nil)
147 (defvar elmo-pop3-lock nil)
149 (defvar elmo-pop3-local-variables '(elmo-pop3-read-point
150 elmo-pop3-uidl-number-hash
151 elmo-pop3-number-uidl-hash
157 (luna-define-method elmo-network-close-session ((session elmo-pop3-session))
158 (when (elmo-network-session-process-internal session)
159 (when (memq (process-status
160 (elmo-network-session-process-internal session))
162 (elmo-pop3-send-command (elmo-network-session-process-internal session)
165 (or (cdr (elmo-pop3-read-response
166 (elmo-network-session-process-internal session)
168 (error "POP error: QUIT failed")))
169 (kill-buffer (process-buffer
170 (elmo-network-session-process-internal session)))
171 (delete-process (elmo-network-session-process-internal session))))
173 (defun elmo-pop3-get-session (folder &optional if-exists)
174 "Get POP3 session for FOLDER.
175 If IF-EXISTS is non-nil, don't get new session.
176 If IF-EXISTS is `any-exists', get BIFF session or normal session if exists."
177 (let ((elmo-pop3-use-uidl-internal (elmo-pop3-folder-use-uidl folder)))
179 (if (eq if-exists 'any-exists)
180 (or (elmo-network-get-session 'elmo-pop3-session
183 (elmo-network-get-session 'elmo-pop3-session
186 (elmo-network-get-session 'elmo-pop3-session
188 (if (elmo-folder-biff-internal folder)
192 ;; For saving existency.
193 (unless (file-exists-p (elmo-folder-msgdb-path folder))
194 (elmo-make-directory (elmo-folder-msgdb-path folder))))))
196 (defun elmo-pop3-send-command (process command &optional no-erase no-log)
197 (with-current-buffer (process-buffer process)
200 (goto-char (point-min))
201 (setq elmo-pop3-read-point (point))
202 (elmo-pop3-debug "SEND: %s\n" (if no-log "<NO LOGGING>" command))
203 (process-send-string process (concat command "\r\n"))))
205 (defun elmo-pop3-read-response (process &optional not-command)
206 "Read response and return a cons cell of \(CODE . BODY\).
207 PROCESS is the process to read response from.
208 If optional NOT-COMMAND is non-nil, read only the first line.
209 CODE is one of the following:
210 'ok ... response is OK.
211 'err ... response is ERROR.
212 'login-delay ... user is not allowed to login until the login delay
214 'in-use ... authentication was successful but the mailbox is in use."
215 ;; buffer is in case for process is dead.
216 (with-current-buffer (process-buffer process)
217 (let ((case-fold-search nil)
218 (response-string nil)
219 (response-continue t)
223 (while response-continue
224 (goto-char elmo-pop3-read-point)
225 (while (not (re-search-forward "\r?\n" nil t))
226 (accept-process-output process 1)
227 (goto-char elmo-pop3-read-point))
228 (setq match-end (point))
229 (setq response-string
230 (buffer-substring elmo-pop3-read-point (- match-end 2)))
231 (goto-char elmo-pop3-read-point)
232 (if (looking-at "\\+.*$")
234 (setq response-continue nil)
235 (setq elmo-pop3-read-point match-end)
238 (concat return-value "\n" response-string)
240 (if (looking-at "\\-.*$")
242 (when (looking-at "[^ ]+ \\[\\([^]]+\\)\\]")
246 (buffer-substring (match-beginning 1)
249 response-continue nil
250 elmo-pop3-read-point match-end
251 return-value (cons (or return-value 'err) nil)))
252 (setq elmo-pop3-read-point match-end)
254 (setq response-continue nil))
257 (concat return-value "\n" response-string)
259 (setq elmo-pop3-read-point match-end)))
262 (cons 'ok return-value)))))
264 (defun elmo-pop3-process-filter (process output)
265 (when (buffer-live-p (process-buffer process))
266 (with-current-buffer (process-buffer process)
267 (goto-char (point-max))
269 (elmo-pop3-debug "RECEIVED: %s\n" output)
270 (when elmo-pop3-retrieve-progress-reporter
271 (elmo-progress-notify 'elmo-retrieve-message :set (buffer-size))))))
273 (defun elmo-pop3-auth-user (session)
274 (let ((process (elmo-network-session-process-internal session))
277 (elmo-pop3-send-command
279 (format "user %s" (elmo-network-session-user-internal session))
281 (setq response (elmo-pop3-read-response process t))
282 (unless (eq (car response) 'ok)
283 (signal 'elmo-open-error '(elmo-pop-auth-user)))
284 (elmo-pop3-send-command process
288 (elmo-network-session-password-key session)))
290 (setq response (elmo-pop3-read-response process t))
294 (error "Maildrop is currently in use"))
296 (error "Not allowed to login until the login delay period has expired"))
298 (signal 'elmo-authenticate-error '(elmo-pop-auth-user))))
301 (defun elmo-pop3-auth-apop (session)
302 (unless (string-match "^\+OK .*\\(<[^\>]+>\\)"
303 (elmo-network-session-greeting-internal session))
304 (signal 'elmo-open-error '(elmo-pop3-auth-apop)))
305 ;; good, APOP ready server
306 (elmo-pop3-send-command
307 (elmo-network-session-process-internal session)
309 (elmo-network-session-user-internal session)
311 (concat (match-string
313 (elmo-network-session-greeting-internal session))
315 (elmo-network-session-password-key session)))))
317 (let ((response (elmo-pop3-read-response
318 (elmo-network-session-process-internal session)
323 (error "Maildrop is currently in use"))
325 (error "Not allowed to login until the login delay period has expired"))
327 (signal 'elmo-authenticate-error '(elmo-pop-auth-apop))))
330 (luna-define-method elmo-network-initialize-session-buffer :after
331 ((session elmo-pop3-session) buffer)
332 (with-current-buffer buffer
333 (mapcar 'make-variable-buffer-local elmo-pop3-local-variables)))
335 (luna-define-method elmo-network-initialize-session ((session
337 (let ((process (elmo-network-session-process-internal session))
339 (with-current-buffer (process-buffer process)
340 (set-process-filter process 'elmo-pop3-process-filter)
341 (setq elmo-pop3-read-point (point-min))
342 ;; Skip garbage output from process before greeting.
343 (while (and (memq (process-status process) '(open run))
344 (goto-char (point-max))
346 (not (looking-at "+OK")))
347 (accept-process-output process 1))
348 (setq elmo-pop3-read-point (point))
349 (or (elmo-network-session-set-greeting-internal
351 (cdr (elmo-pop3-read-response process t))) ; if ok, cdr is non-nil.
352 (signal 'elmo-open-error
353 '(elmo-network-intialize-session)))
354 (when (eq (elmo-network-stream-type-symbol
355 (elmo-network-session-stream-type-internal session))
357 (elmo-pop3-send-command process "stls")
358 (if (eq 'ok (car (elmo-pop3-read-response process)))
359 (starttls-negotiate process)
360 (signal 'elmo-open-error '(elmo-pop3-starttls-error)))))))
362 (luna-define-method elmo-network-authenticate-session ((session
364 (with-current-buffer (process-buffer
365 (elmo-network-session-process-internal session))
366 (let* ((process (elmo-network-session-process-internal session))
367 (auth (elmo-network-session-auth-internal session))
368 (auth (mapcar (lambda (mechanism) (upcase (symbol-name mechanism)))
369 (if (listp auth) auth (list auth)))))
370 (or (and (string= "USER" (car auth))
371 (elmo-pop3-auth-user session))
372 (and (string= "APOP" (car auth))
373 (elmo-pop3-auth-apop session))
374 (let (sasl-mechanisms
375 client name step response mechanism
376 sasl-read-passphrase)
378 (setq sasl-mechanisms (mapcar 'car sasl-mechanism-alist))
379 (setq mechanism (sasl-find-mechanism auth))
381 (signal 'elmo-authenticate-error '(elmo-pop3-auth-no-mechanisms)))
385 (elmo-network-session-user-internal session)
387 (elmo-network-session-server-internal session)))
388 ;;; (if elmo-pop3-auth-user-realm
389 ;;; (sasl-client-set-property client 'realm elmo-pop3-auth-user-realm))
390 (setq name (sasl-mechanism-name mechanism))
391 (elmo-network-session-set-auth-internal session
392 (intern (downcase name)))
393 (setq sasl-read-passphrase
396 (elmo-network-session-password-key session))))
397 (setq step (sasl-next-step client nil))
398 (elmo-pop3-send-command
401 (and (sasl-step-data step)
404 (elmo-base64-encode-string
405 (sasl-step-data step) 'no-line-break))))
409 (setq response (elmo-pop3-read-response process t))
413 (error "Maildrop is currently in use"))
415 (error "Not allowed to login \
416 until the login delay period has expired"))
418 (signal 'elmo-authenticate-error
419 (list (intern (concat "elmo-pop3-auth-"
420 (downcase name)))))))
421 (if (sasl-next-step client step)
423 (signal 'elmo-authenticate-error
425 (concat "elmo-pop3-auth-"
427 ;; The authentication process is finished.
431 (elmo-base64-decode-string
432 (cadr (split-string response " "))))
433 (setq step (sasl-next-step client step))
434 (elmo-pop3-send-command
436 (if (sasl-step-data step)
437 (elmo-base64-encode-string (sasl-step-data step)
439 "") nil 'no-log))))))))
441 (luna-define-method elmo-network-setup-session ((session
443 (let ((process (elmo-network-session-process-internal session))
445 (with-current-buffer (process-buffer process)
446 (setq elmo-pop3-size-hash (elmo-make-hash 31))
447 ;; To get obarray of uidl and size
448 (elmo-pop3-send-command process "list")
449 (if (null (cdr (elmo-pop3-read-response process)))
450 (error "POP LIST command failed"))
451 (if (null (setq response
452 (elmo-pop3-read-contents process)))
453 (error "POP LIST command failed"))
454 ;; POP server always returns a sequence of serial numbers.
455 (setq count (elmo-pop3-parse-list-response response))
457 (when elmo-pop3-use-uidl-internal
458 (setq elmo-pop3-uidl-number-hash (elmo-make-hash (* count 2)))
459 (setq elmo-pop3-number-uidl-hash (elmo-make-hash (* count 2)))
461 (elmo-pop3-send-command process "uidl")
462 (unless (cdr (elmo-pop3-read-response process))
463 (error "POP UIDL failed"))
464 (unless (setq response (elmo-pop3-read-contents process))
465 (error "POP UIDL failed"))
466 (elmo-pop3-parse-uidl-response response)))))
468 (defun elmo-pop3-read-contents (process)
469 (with-current-buffer (process-buffer process)
470 (let ((case-fold-search nil)
472 (goto-char elmo-pop3-read-point)
473 (while (not (re-search-forward "^\\.\r\n" nil t))
474 (accept-process-output process 1)
475 (goto-char elmo-pop3-read-point))
476 (setq match-end (point))
478 (buffer-substring elmo-pop3-read-point
481 (luna-define-method elmo-folder-expand-msgdb-path ((folder elmo-pop3-folder))
482 (convert-standard-filename
484 (elmo-safe-filename (elmo-net-folder-user-internal folder))
485 (expand-file-name (elmo-net-folder-server-internal folder)
488 elmo-msgdb-directory)))))
490 (luna-define-method elmo-folder-exists-p ((folder elmo-pop3-folder))
491 (if (and elmo-pop3-exists-exactly
492 (elmo-folder-plugged-p folder))
494 (let (elmo-auto-change-plugged ; don't change plug status.
495 (elmo-inhibit-number-mapping t) ; No need to use uidl.
498 (setq session (elmo-pop3-get-session folder))
500 (elmo-network-close-session session)))))
501 (or (file-directory-p (elmo-folder-msgdb-path folder))
503 (when (elmo-folder-plugged-p folder)
504 (let ((elmo-pop3-exists-exactly t))
505 (elmo-folder-exists-p folder))))))
507 (defun elmo-pop3-parse-uidl-response (string)
508 (let ((buffer (current-buffer))
511 (let (number uid list)
513 (goto-char (point-min))
514 (while (re-search-forward "^\\([0-9]+\\)[\t ]+\\([^ \n]+\\)$" nil t)
515 (setq number (elmo-match-buffer 1))
516 (setq uid (elmo-match-buffer 2))
517 (with-current-buffer buffer
518 (elmo-set-hash-val uid number elmo-pop3-uidl-number-hash)
519 (elmo-set-hash-val (concat "#" number) uid
520 elmo-pop3-number-uidl-hash))
521 (setq list (cons uid list)))
522 (with-current-buffer buffer (setq elmo-pop3-uidl-done t))
525 (defun elmo-pop3-parse-list-response (string)
526 (let ((buffer (current-buffer))
531 (goto-char (point-min))
532 (while (re-search-forward "^\\([0-9]+\\)[\t ]+\\([0-9]+\\)$" nil t)
535 (cons (elmo-match-buffer 1)
536 (elmo-match-buffer 2))
538 (setq count (1+ count)))
539 (with-current-buffer buffer
540 (setq elmo-pop3-size-hash (elmo-make-hash (* (length alist) 2)))
542 (elmo-set-hash-val (concat "#" (car (car alist)))
545 (setq alist (cdr alist)))
546 (setq elmo-pop3-list-done t))
549 (defun elmo-pop3-list-location (folder)
550 (with-current-buffer (process-buffer
551 (elmo-network-session-process-internal
552 (elmo-pop3-get-session folder)))
554 (if elmo-pop3-uidl-done
558 (setq locations (cons (symbol-name atom) locations)))
559 elmo-pop3-uidl-number-hash)
562 (< (elmo-pop3-uidl-to-number loc1)
563 (elmo-pop3-uidl-to-number loc2)))))
564 (error "POP3: Error in UIDL")))))
566 (defun elmo-pop3-list-folder-by-location (folder locations)
567 (mapcar #'car (elmo-location-map-update folder locations)))
569 (defun elmo-pop3-list-by-uidl-subr (folder &optional nonsort)
570 (let ((flist (elmo-pop3-list-folder-by-location
572 (elmo-pop3-list-location folder))))
574 (cons (elmo-max-of-list flist) (length flist))
577 (defun elmo-pop3-list-by-list (folder)
578 (with-current-buffer (process-buffer
579 (elmo-network-session-process-internal
580 (elmo-pop3-get-session folder)))
582 (if elmo-pop3-list-done
584 (mapatoms (lambda (atom)
585 (setq list (cons (string-to-int
586 (substring (symbol-name atom) 1))
590 (error "POP3: Error in list")))))
592 (defsubst elmo-pop3-folder-list-messages (folder)
593 (if (elmo-pop3-folder-use-uidl folder)
594 (elmo-pop3-list-by-uidl-subr folder)
595 (elmo-pop3-list-by-list folder)))
597 (luna-define-method elmo-folder-list-messages-plugged
598 ((folder elmo-pop3-folder) &optional nohide)
599 (elmo-pop3-folder-list-messages folder))
601 (luna-define-method elmo-folder-status ((folder elmo-pop3-folder))
602 (elmo-folder-open-internal folder)
603 (elmo-folder-check folder)
604 (if (elmo-pop3-folder-use-uidl folder)
606 (elmo-pop3-list-by-uidl-subr folder 'nonsort)
607 (elmo-folder-close-internal folder))
609 (elmo-network-session-process-internal
610 (elmo-pop3-get-session folder)))
613 (with-current-buffer (process-buffer process)
614 (elmo-pop3-send-command process "STAT")
615 (setq response (cdr (elmo-pop3-read-response process)))
616 ;; response: "^\+OK 2 7570$"
617 (if (not (string-match "^\+OK[ \t]*\\([0-9]*\\)" response))
618 (error "POP STAT command failed")
621 (substring response (match-beginning 1)(match-end 1 ))))
622 (elmo-folder-close-internal folder)
623 (cons total total))))))
625 (defvar elmo-pop3-header-fetch-chop-length 200)
627 (defsubst elmo-pop3-next-result-arrived-p ()
629 ((eq (following-char) ?+)
630 (if (re-search-forward "\n\\.\r?\n" nil t)
634 (if (search-forward "\n" nil t)
640 (defun elmo-pop3-retrieve-headers (process tobuffer articles)
642 (set-buffer (process-buffer process))
646 (last-point (point-min)))
647 (elmo-with-progress-display (elmo-retrieve-header (length articles))
649 ;; Send HEAD commands.
651 (elmo-pop3-send-command process
652 (format "top %s 0" (car articles))
654 ;;; (accept-process-output process 1)
655 (setq articles (cdr articles))
656 (setq count (1+ count))
657 ;; Every 200 requests we have to read the stream in
658 ;; order to avoid deadlocks.
659 (when (or elmo-pop3-send-command-synchronously
660 (null articles) ;All requests have been sent.
661 (zerop (% count elmo-pop3-header-fetch-chop-length)))
662 (unless elmo-pop3-send-command-synchronously
663 (accept-process-output process 1))
666 (goto-char last-point)
668 (while (elmo-pop3-next-result-arrived-p)
669 (setq last-point (point))
670 (setq received (1+ received)))
672 (elmo-progress-notify 'elmo-retrieve-header :set received)
673 (accept-process-output process 1)
674 ;;; (accept-process-output process)
676 ;; Replace all CRLF with LF.
677 (elmo-delete-cr-buffer)
678 (copy-to-buffer tobuffer (point-min) (point-max)))))
680 (luna-define-method elmo-folder-msgdb-create ((folder elmo-pop3-folder)
682 (let ((process (elmo-network-session-process-internal
683 (elmo-pop3-get-session folder))))
684 (with-current-buffer (process-buffer process)
685 (elmo-pop3-msgdb-create-by-header
691 (defun elmo-pop3-uidl-to-number (uidl)
692 (string-to-number (elmo-get-hash-val uidl
693 elmo-pop3-uidl-number-hash)))
695 (defun elmo-pop3-number-to-uidl (number)
696 (elmo-get-hash-val (format "#%d" number)
697 elmo-pop3-number-uidl-hash))
699 (defun elmo-pop3-number-to-size (number)
701 (elmo-get-hash-val (format "#%d" number) elmo-pop3-size-hash)))
703 (defun elmo-pop3-msgdb-create-by-header (folder process numlist
705 (let ((tmp-buffer (get-buffer-create " *ELMO Overview TMP*")))
707 (with-current-buffer (process-buffer process)
708 (when (elmo-pop3-folder-use-uidl folder)
714 (elmo-pop3-uidl-to-number
715 (elmo-map-message-location folder number)))
717 (elmo-pop3-retrieve-headers process tmp-buffer numlist)
718 (elmo-pop3-msgdb-create-message
725 (kill-buffer tmp-buffer))))
727 (defun elmo-pop3-msgdb-create-message (folder
734 (let ((new-msgdb (elmo-make-msgdb))
735 beg entity number message-id flags)
737 (set-buffer-multibyte default-enable-multibyte-characters)
738 (goto-char (point-min))
739 (elmo-with-progress-display (elmo-folder-msgdb-create num)
742 (setq beg (save-excursion (forward-line 1) (point)))
743 (elmo-pop3-next-result-arrived-p)
747 (narrow-to-region beg (point))
749 (elmo-msgdb-create-message-entity-from-buffer
750 (elmo-msgdb-message-entity-handler new-msgdb)
752 (setq numlist (cdr numlist))
754 (with-current-buffer (process-buffer process)
755 (elmo-message-entity-set-field
758 (elmo-pop3-number-to-size
759 (elmo-message-entity-number entity)))
761 (elmo-map-message-number
763 (elmo-pop3-number-to-uidl
764 (elmo-message-entity-number entity))))
765 (elmo-message-entity-set-number entity number)))
766 (setq message-id (elmo-message-entity-field entity 'message-id)
767 flags (elmo-flag-table-get flag-table message-id))
768 (elmo-global-flags-set flags folder number message-id)
769 (elmo-msgdb-append-entity new-msgdb entity flags))))
770 (elmo-progress-notify 'elmo-folder-msgdb-create)))
773 (defun elmo-pop3-read-body (process outbuf)
774 (with-current-buffer (process-buffer process)
775 (let ((start elmo-pop3-read-point)
778 (while (not (re-search-forward "^\\.\r?\n" nil t))
779 (accept-process-output process 1)
782 (with-current-buffer outbuf
784 (insert-buffer-substring (process-buffer process) start (- end 3)))
787 (luna-define-method elmo-folder-open-internal ((folder elmo-pop3-folder))
788 (when (elmo-pop3-folder-use-uidl folder)
789 (elmo-location-map-load folder (elmo-folder-msgdb-path folder))))
791 (luna-define-method elmo-folder-commit :after ((folder elmo-pop3-folder))
792 (when (and (not elmo-inhibit-number-mapping)
793 (elmo-folder-persistent-p folder))
794 (elmo-location-map-save folder (elmo-folder-msgdb-path folder))))
796 (luna-define-method elmo-folder-close-internal ((folder elmo-pop3-folder))
797 (elmo-location-map-teardown folder)
798 ;; Just close connection
799 (elmo-folder-check folder))
801 (luna-define-method elmo-message-fetch-plugged ((folder elmo-pop3-folder)
805 (let ((process (elmo-network-session-process-internal
806 (elmo-pop3-get-session folder)))
807 size response errmsg msg)
808 (with-current-buffer (process-buffer process)
809 (when (elmo-pop3-folder-use-uidl folder)
810 (setq number (elmo-pop3-uidl-to-number
811 (elmo-map-message-location folder number))))
812 (setq size (elmo-pop3-number-to-size number))
814 (elmo-with-progress-display
815 (elmo-retrieve-message size elmo-pop3-retrieve-progress-reporter)
817 (elmo-pop3-send-command process (format "retr %s" number))
818 (when (null (setq response (cdr (elmo-pop3-read-response
820 (error "Fetching message failed"))
821 (setq response (elmo-pop3-read-body process outbuf)))
823 (goto-char (point-min))
824 (while (re-search-forward "^\\." nil t)
827 (elmo-delete-cr-buffer)
830 (defun elmo-pop3-delete-msg (process number)
832 (error "Deleting message failed"))
833 (elmo-pop3-send-command process (format "dele %s" number))
834 (when (null (cdr (elmo-pop3-read-response process t)))
835 (error "Deleting message failed")))
837 (luna-define-method elmo-folder-delete-messages-plugged ((folder
840 (let ((process (elmo-network-session-process-internal
841 (elmo-pop3-get-session folder))))
842 (with-current-buffer (process-buffer process)
843 (dolist (number (if (elmo-pop3-folder-use-uidl folder)
846 (elmo-pop3-uidl-to-number
847 (elmo-map-message-location folder number)))
850 (elmo-pop3-delete-msg process number))
853 (luna-define-method elmo-message-use-cache-p ((folder elmo-pop3-folder) number)
856 (luna-define-method elmo-folder-persistent-p ((folder elmo-pop3-folder))
857 (and (elmo-folder-persistent-internal folder)
858 (elmo-pop3-folder-use-uidl-internal folder)))
860 (luna-define-method elmo-folder-clear :around ((folder elmo-pop3-folder)
861 &optional keep-killed)
863 (elmo-location-map-setup folder))
864 (luna-call-next-method))
866 (luna-define-method elmo-folder-check ((folder elmo-pop3-folder))
867 (if (elmo-folder-plugged-p folder)
868 (let ((session (elmo-pop3-get-session folder 'if-exists)))
870 (elmo-network-close-session session)))))
873 (product-provide (provide 'elmo-pop3) (require 'elmo-version))
875 ;;; elmo-pop3.el ends here