Mercurial > hg > xemacs-beta
comparison lisp/utils/mail-utils.el @ 0:376386a54a3c r19-14
Import from CVS: tag r19-14
author | cvs |
---|---|
date | Mon, 13 Aug 2007 08:45:50 +0200 |
parents | |
children | 0293115a14e9 |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:376386a54a3c |
---|---|
1 ;;; mail-utils.el --- utility functions used both by rmail and rnews | |
2 | |
3 ;; Copyright (C) 1985 Free Software Foundation, Inc. | |
4 | |
5 ;; Maintainer: FSF | |
6 ;; Keywords: mail, news | |
7 | |
8 ;; This file is part of XEmacs. | |
9 | |
10 ;; XEmacs is free software; you can redistribute it and/or modify it | |
11 ;; under the terms of the GNU General Public License as published by | |
12 ;; the Free Software Foundation; either version 2, or (at your option) | |
13 ;; any later version. | |
14 | |
15 ;; XEmacs is distributed in the hope that it will be useful, but | |
16 ;; WITHOUT ANY WARRANTY; without even the implied warranty of | |
17 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
18 ;; General Public License for more details. | |
19 | |
20 ;; You should have received a copy of the GNU General Public License | |
21 ;; along with XEmacs; see the file COPYING. If not, write to the Free | |
22 ;; Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. | |
23 | |
24 ;;; Synched up with: FSF 19.30. | |
25 | |
26 ;;; Commentary: | |
27 | |
28 ;; Utility functions for mail and netnews handling. These handle fine | |
29 ;; points of header parsing. | |
30 | |
31 ;;; Code: | |
32 | |
33 ;;; We require lisp-mode to make sure that lisp-mode-syntax-table has | |
34 ;;; been initialized. | |
35 (require 'lisp-mode) | |
36 | |
37 ;;;###autoload | |
38 (defvar mail-use-rfc822 nil "\ | |
39 *If non-nil, use a full, hairy RFC822 parser on mail addresses. | |
40 Otherwise, (the default) use a smaller, somewhat faster, and | |
41 often correct parser.") | |
42 | |
43 ;; Returns t if file FILE is an Rmail file. | |
44 ;;;###autoload | |
45 (defun mail-file-babyl-p (file) | |
46 (let ((buf (generate-new-buffer " *rmail-file-p*"))) | |
47 (unwind-protect | |
48 (save-excursion | |
49 (set-buffer buf) | |
50 (insert-file-contents file nil 0 100) | |
51 (looking-at "BABYL OPTIONS:")) | |
52 (kill-buffer buf)))) | |
53 | |
54 (defun mail-string-delete (string start end) | |
55 "Returns a string containing all of STRING except the part | |
56 from START (inclusive) to END (exclusive)." | |
57 (if (null end) (substring string 0 start) | |
58 (concat (substring string 0 start) | |
59 (substring string end nil)))) | |
60 | |
61 (defun mail-strip-quoted-names (address) | |
62 "Delete comments and quoted strings in an address list ADDRESS. | |
63 Also delete leading/trailing whitespace and replace FOO <BAR> with just BAR. | |
64 Return a modified address list." | |
65 (if (null address) | |
66 nil | |
67 (if mail-use-rfc822 | |
68 (progn (require 'rfc822) | |
69 (mapconcat 'identity (rfc822-addresses address) ", ")) | |
70 (let (pos) | |
71 (string-match "\\`[ \t\n]*" address) | |
72 ;; strip surrounding whitespace | |
73 (setq address (substring address | |
74 (match-end 0) | |
75 (string-match "[ \t\n]*\\'" address | |
76 (match-end 0)))) | |
77 | |
78 ;; Detect nested comments. | |
79 (if (string-match "[ \t]*(\\([^)\"\\]\\|\\\\.\\|\\\\\n\\)*(" address) | |
80 ;; Strip nested comments. | |
81 (save-excursion | |
82 (set-buffer (get-buffer-create " *temp*")) | |
83 (erase-buffer) | |
84 (insert address) | |
85 (set-syntax-table lisp-mode-syntax-table) | |
86 (goto-char 1) | |
87 (while (search-forward "(" nil t) | |
88 (forward-char -1) | |
89 (skip-chars-backward " \t") | |
90 (delete-region (point) | |
91 (save-excursion | |
92 (condition-case () | |
93 (forward-sexp 1) | |
94 (error (goto-char (point-max)))) | |
95 (point)))) | |
96 (setq address (buffer-string)) | |
97 (erase-buffer)) | |
98 ;; Strip non-nested comments an easier way. | |
99 (while (setq pos (string-match | |
100 ;; This doesn't hack rfc822 nested comments | |
101 ;; `(xyzzy (foo) whinge)' properly. Big deal. | |
102 "[ \t]*(\\([^)\"\\]\\|\\\\.\\|\\\\\n\\)*)" | |
103 address)) | |
104 (setq address | |
105 (mail-string-delete address | |
106 pos (match-end 0))))) | |
107 | |
108 ;; strip `quoted' names (This is supposed to hack `"Foo Bar" <bar@host>') | |
109 (setq pos 0) | |
110 (while (setq pos (string-match | |
111 "[ \t]*\"\\([^\"\\]\\|\\\\.\\|\\\\\n\\)*\"[ \t\n]*" | |
112 address pos)) | |
113 ;; If the next thing is "@", we have "foo bar"@host. Leave it. | |
114 (if (and (> (length address) (match-end 0)) | |
115 (= (aref address (match-end 0)) ?@)) | |
116 (setq pos (match-end 0)) | |
117 (setq address | |
118 (mail-string-delete address | |
119 pos (match-end 0))))) | |
120 ;; Retain only part of address in <> delims, if there is such a thing. | |
121 (while (setq pos (string-match "\\(,\\s-*\\|\\`\\)[^,]*<\\([^>,]*>\\)" | |
122 address)) | |
123 (let ((junk-beg (match-end 1)) | |
124 (junk-end (match-beginning 2)) | |
125 (close (match-end 0))) | |
126 (setq address (mail-string-delete address (1- close) close)) | |
127 (setq address (mail-string-delete address junk-beg junk-end)))) | |
128 address)))) | |
129 | |
130 (or (and (boundp 'rmail-default-dont-reply-to-names) | |
131 (not (null rmail-default-dont-reply-to-names))) | |
132 (setq rmail-default-dont-reply-to-names "info-")) | |
133 | |
134 ; rmail-dont-reply-to-names is defined in loaddefs | |
135 (defun rmail-dont-reply-to (userids) | |
136 "Returns string of mail addresses USERIDS sans any recipients | |
137 that start with matches for `rmail-dont-reply-to-names'. | |
138 Usenet paths ending in an element that matches are removed also." | |
139 (if (null rmail-dont-reply-to-names) | |
140 (setq rmail-dont-reply-to-names | |
141 (concat (if rmail-default-dont-reply-to-names | |
142 (concat rmail-default-dont-reply-to-names "\\|") | |
143 "") | |
144 (concat (regexp-quote (user-login-name)) | |
145 "\\>")))) | |
146 (let ((match (concat "\\(^\\|,\\)[ \t\n]*\\([^,\n]*!\\|\\)\\(" | |
147 rmail-dont-reply-to-names | |
148 "\\)")) | |
149 (case-fold-search t) | |
150 pos epos) | |
151 (while (setq pos (string-match match userids)) | |
152 (if (> pos 0) (setq pos (match-beginning 2))) | |
153 (setq epos | |
154 ;; Delete thru the next comma, plus whitespace after. | |
155 (if (string-match ",[ \t\n]+" userids (match-end 0)) | |
156 (match-end 0) | |
157 (length userids))) | |
158 (setq userids | |
159 (mail-string-delete | |
160 userids pos epos))) | |
161 ;; get rid of any trailing commas | |
162 (if (setq pos (string-match "[ ,\t\n]*\\'" userids)) | |
163 (setq userids (substring userids 0 pos))) | |
164 ;; remove leading spaces. they bother me. | |
165 (if (string-match "\\s *" userids) | |
166 (substring userids (match-end 0)) | |
167 userids))) | |
168 | |
169 ;;;###autoload | |
170 (defun mail-fetch-field (field-name &optional last all) | |
171 "Return the value of the header field FIELD-NAME. | |
172 The buffer is expected to be narrowed to just the headers of the message. | |
173 If second arg LAST is non-nil, use the last such field if there are several. | |
174 If third arg ALL is non-nil, concatenate all such fields with commas between." | |
175 (save-excursion | |
176 (goto-char (point-min)) | |
177 (let ((case-fold-search t) | |
178 (name (concat "^" (regexp-quote field-name) "[ \t]*:[ \t]*"))) | |
179 (if all | |
180 (let ((value "")) | |
181 (while (re-search-forward name nil t) | |
182 (let ((opoint (point))) | |
183 (while (progn (forward-line 1) | |
184 (looking-at "[ \t]"))) | |
185 ;; Back up over newline, then trailing spaces or tabs | |
186 (forward-char -1) | |
187 (while (member (preceding-char) '(? ?\t)) | |
188 (forward-char -1)) | |
189 (setq value (concat value | |
190 (if (string= value "") "" ", ") | |
191 (buffer-substring-no-properties | |
192 opoint (point)))))) | |
193 (and (not (string= value "")) value)) | |
194 (if (re-search-forward name nil t) | |
195 (progn | |
196 (if last (while (re-search-forward name nil t))) | |
197 (let ((opoint (point))) | |
198 (while (progn (forward-line 1) | |
199 (looking-at "[ \t]"))) | |
200 ;; Back up over newline, then trailing spaces or tabs | |
201 (forward-char -1) | |
202 (while (member (preceding-char) '(? ?\t)) | |
203 (forward-char -1)) | |
204 (buffer-substring-no-properties opoint (point))))))))) | |
205 | |
206 ;; Parse a list of tokens separated by commas. | |
207 ;; It runs from point to the end of the visible part of the buffer. | |
208 ;; Whitespace before or after tokens is ignored, | |
209 ;; but whitespace within tokens is kept. | |
210 (defun mail-parse-comma-list () | |
211 (let (accumulated | |
212 beg) | |
213 (skip-chars-forward " ") | |
214 (while (not (eobp)) | |
215 (setq beg (point)) | |
216 (skip-chars-forward "^,") | |
217 (skip-chars-backward " ") | |
218 (setq accumulated | |
219 (cons (buffer-substring beg (point)) | |
220 accumulated)) | |
221 (skip-chars-forward "^,") | |
222 (skip-chars-forward ", ")) | |
223 accumulated)) | |
224 | |
225 (defun mail-comma-list-regexp (labels) | |
226 (let (pos) | |
227 (setq pos (or (string-match "[^ \t]" labels) 0)) | |
228 ;; Remove leading and trailing whitespace. | |
229 (setq labels (substring labels pos (string-match "[ \t]*$" labels pos))) | |
230 ;; Change each comma to \|, and flush surrounding whitespace. | |
231 (while (setq pos (string-match "[ \t]*,[ \t]*" labels)) | |
232 (setq labels | |
233 (concat (substring labels 0 pos) | |
234 "\\|" | |
235 (substring labels (match-end 0)))))) | |
236 labels) | |
237 | |
238 (defun mail-rfc822-time-zone (time) | |
239 (let* ((sec (or (car (current-time-zone time)) 0)) | |
240 (absmin (/ (abs sec) 60))) | |
241 (format "%c%02d%02d" (if (< sec 0) ?- ?+) (/ absmin 60) (% absmin 60)))) | |
242 | |
243 (defun mail-rfc822-date () | |
244 (let* ((time (current-time)) | |
245 (s (current-time-string time))) | |
246 (string-match "[^ ]+ +\\([^ ]+\\) +\\([^ ]+\\) \\([^ ]+\\) \\([^ ]+\\)" s) | |
247 (concat (substring s (match-beginning 2) (match-end 2)) " " | |
248 (substring s (match-beginning 1) (match-end 1)) " " | |
249 (substring s (match-beginning 4) (match-end 4)) " " | |
250 (substring s (match-beginning 3) (match-end 3)) " " | |
251 (mail-rfc822-time-zone time)))) | |
252 | |
253 (provide 'mail-utils) | |
254 | |
255 ;;; mail-utils.el ends here |