summaryrefslogtreecommitdiffstats
path: root/Lib
diff options
context:
space:
mode:
Diffstat (limited to 'Lib')
-rw-r--r--Lib/rfc822.py145
1 files changed, 92 insertions, 53 deletions
diff --git a/Lib/rfc822.py b/Lib/rfc822.py
index b42eaec..6e584eb 100644
--- a/Lib/rfc822.py
+++ b/Lib/rfc822.py
@@ -307,90 +307,129 @@ def unquote(str):
# Parse an address into (name, address) tuple
+# (By Sjoerd Mullender)
+
+error = 'parseaddr.error'
+
+specials = regex.compile('[][()<>,.;:@\\" \000-\037\177-\377]')
+
+def quote(str):
+ return '"%s"' % string.join(
+ string.split(
+ string.join(
+ string.split(str, '\\'),
+ '\\\\'),
+ '"'),
+ '\\"')
def parseaddr(address):
- import string
- str = ''
- email = ''
- comment = ''
+ token = [] # the current token
+ tokens = [] # the list of tokens
backslash = 0
dquote = 0
+ was_quoted = 0
space = 0
paren = 0
- bracket = 0
- seen_bracket = 0
for c in address:
if backslash:
- str = str + c
+ token.append(c)
backslash = 0
- continue
if c == '\\':
backslash = 1
+ was_quoted = 1
continue
if dquote:
if c == '"':
dquote = 0
else:
- str = str + c
+ token.append(c)
continue
if c == '"':
dquote = 1
+ was_quoted = 1
continue
- if c in string.whitespace:
- space = 1
- continue
- if space:
- str = str + ' '
- space = 0
if paren:
if c == '(':
paren = paren + 1
- str = str + c
- continue
- if c == ')':
+ elif c == ')':
paren = paren - 1
if paren == 0:
- comment = comment + str
- str = ''
+ token = string.join(token, '')
+ tokens.append((2, token))
+ token = []
continue
+ token.append(c)
+ continue
if c == '(':
- paren = paren + 1
- if bracket:
- email = email + str
- str = ''
- elif not seen_bracket:
- email = email + str
- str = ''
+ paren = 1
+ token = string.join(token, '')
+ tokens.append((was_quoted, token))
+ was_quoted = 0
+ token = []
continue
- if bracket:
- if c == '>':
- bracket = 0
- email = email + str
- str = ''
- continue
- if c == '<':
- bracket = 1
- seen_bracket = 1
- comment = comment + str
- str = ''
- email = ''
+ if c in string.whitespace:
+ space = 1
+ continue
+ if c in '<>@,;:.[]':
+ token = string.join(token, '')
+ tokens.append((was_quoted, token))
+ was_quoted = 0
+ token = []
+ tokens.append((0, c))
+ space = 0
continue
- if c == '#' and not bracket and not paren:
- # rest is comment
- break
- str = str + c
- if str:
- if seen_bracket:
- if bracket:
- email = str
+ if space:
+ token = string.join(token, '')
+ tokens.append((was_quoted, token))
+ was_quoted = 0
+ token = []
+ space = 0
+ token.append(c)
+ token = string.join(token, '')
+ tokens.append((was_quoted, token))
+ if (0, '<') in tokens:
+ name = []
+ addr = []
+ cur = name
+ for token in tokens:
+ if token[1] == '':
+ continue
+ if token == (0, '<'):
+ if addr:
+ raise error, 'syntax error'
+ cur = addr
+ elif token == (0, '>'):
+ if cur is not addr:
+ raise error, 'syntax error'
+ cur = name
+ elif token[0] == 2:
+ if cur is name:
+ name.append('(' + token[1] + ')')
+ else:
+ name.append(token[1])
+ elif token[0] == 1 and cur is addr:
+ if specials.search(token[1]) >= 0:
+ cur.append(quote(token[1]))
+ else:
+ cur.append(token[1])
else:
- comment = comment + str
- else:
- if paren:
- comment = comment + str
+ cur.append(token[1])
+ else:
+ name = []
+ addr = []
+ for token in tokens:
+ if token[1] == '':
+ continue
+ if token[0] == 2:
+ name.append(token[1])
+ elif token[0] == 1:
+ if specials.search(token[1]) >= 0:
+ addr.append(quote(token[1]))
+ else:
+ addr.append(token[1])
else:
- email = email + str
- return string.strip(comment), string.strip(email)
+ addr.append(token[1])
+ return string.join(name, ' '), string.join(addr, '')
# Parse a date field