summaryrefslogtreecommitdiffstats
path: root/Lib
diff options
context:
space:
mode:
authorSerhiy Storchaka <storchaka@gmail.com>2015-05-19 07:10:15 (GMT)
committerSerhiy Storchaka <storchaka@gmail.com>2015-05-19 07:10:15 (GMT)
commit6baa0a53b05d580d21d260c5d7d6ee2d3ff005ee (patch)
tree064b703fb4ba97f7ee79880b1b6e5b4fb6f9a6a1 /Lib
parent8687275358a9267c5ea7dcd4ce6a4cbd0f91f541 (diff)
parentae760c0a2ccd5557bfd915c947dfbcd3ed64fd60 (diff)
downloadcpython-6baa0a53b05d580d21d260c5d7d6ee2d3ff005ee.zip
cpython-6baa0a53b05d580d21d260c5d7d6ee2d3ff005ee.tar.gz
cpython-6baa0a53b05d580d21d260c5d7d6ee2d3ff005ee.tar.bz2
Issue #6598: Increased time precision and random number range in
email.utils.make_msgid() to strengthen the uniqueness of the message ID.
Diffstat (limited to 'Lib')
-rw-r--r--Lib/email/utils.py9
-rw-r--r--Lib/test/test_email/test_email.py25
2 files changed, 28 insertions, 6 deletions
diff --git a/Lib/email/utils.py b/Lib/email/utils.py
index cacb9b1..317fdfa 100644
--- a/Lib/email/utils.py
+++ b/Lib/email/utils.py
@@ -202,24 +202,23 @@ def format_datetime(dt, usegmt=False):
def make_msgid(idstring=None, domain=None):
"""Returns a string suitable for RFC 2822 compliant Message-ID, e.g:
- <20020201195627.33539.96671@nightshade.la.mastaler.com>
+ <142480216486.20800.16526388040877946887@nightshade.la.mastaler.com>
Optional idstring if given is a string used to strengthen the
uniqueness of the message id. Optional domain if given provides the
portion of the message id after the '@'. It defaults to the locally
defined hostname.
"""
- timeval = time.time()
- utcdate = time.strftime('%Y%m%d%H%M%S', time.gmtime(timeval))
+ timeval = int(time.time()*100)
pid = os.getpid()
- randint = random.randrange(100000)
+ randint = random.getrandbits(64)
if idstring is None:
idstring = ''
else:
idstring = '.' + idstring
if domain is None:
domain = socket.getfqdn()
- msgid = '<%s.%s.%s%s@%s>' % (utcdate, pid, randint, idstring, domain)
+ msgid = '<%d.%d.%d%s@%s>' % (timeval, pid, randint, idstring, domain)
return msgid
diff --git a/Lib/test/test_email/test_email.py b/Lib/test/test_email/test_email.py
index b627760..d7e3dca 100644
--- a/Lib/test/test_email/test_email.py
+++ b/Lib/test/test_email/test_email.py
@@ -11,6 +11,10 @@ import textwrap
from io import StringIO, BytesIO
from itertools import chain
from random import choice
+try:
+ from threading import Thread
+except ImportError:
+ from dummy_threading import Thread
import email
import email.policy
@@ -34,7 +38,7 @@ from email import iterators
from email import base64mime
from email import quoprimime
-from test.support import unlink
+from test.support import unlink, start_threads
from test.test_email import openfile, TestEmailBase
# These imports are documented to work, but we are testing them using a
@@ -3167,6 +3171,25 @@ Foo
addrs = utils.getaddresses(['User ((nested comment)) <foo@bar.com>'])
eq(addrs[0][1], 'foo@bar.com')
+ def test_make_msgid_collisions(self):
+ # Test make_msgid uniqueness, even with multiple threads
+ class MsgidsThread(Thread):
+ def run(self):
+ # generate msgids for 3 seconds
+ self.msgids = []
+ append = self.msgids.append
+ make_msgid = utils.make_msgid
+ clock = time.clock
+ tfin = clock() + 3.0
+ while clock() < tfin:
+ append(make_msgid(domain='testdomain-string'))
+
+ threads = [MsgidsThread() for i in range(5)]
+ with start_threads(threads):
+ pass
+ all_ids = sum([t.msgids for t in threads], [])
+ self.assertEqual(len(set(all_ids)), len(all_ids))
+
def test_utils_quote_unquote(self):
eq = self.assertEqual
msg = Message()