Blame SOURCES/00255-Fix-ssl-module-parsing-of-GEN_RID-subject-alternative-name-fields-in-X.509-certs.patch

925e6b
925e6b
# HG changeset patch
925e6b
# User Christian Heimes <christian@python.org>
925e6b
# Date 1473197135 -7200
925e6b
# Node ID 74805fd9e7343649372d0b9c76b4490b2975a674
925e6b
# Parent  6f4f19217d9be12be7a9c86cf1e118b140564b4f
925e6b
Issue #27691: Fix ssl module's parsing of GEN_RID subject alternative name fields in X.509 certs.
925e6b
925e6b
diff --git a/Lib/test/allsans.pem b/Lib/test/allsans.pem
925e6b
new file mode 100644
925e6b
--- /dev/null
925e6b
+++ b/Lib/test/allsans.pem
925e6b
@@ -0,0 +1,37 @@
925e6b
+-----BEGIN PRIVATE KEY-----
925e6b
+MIICdgIBADANBgkqhkiG9w0BAQEFAASCAmAwggJcAgEAAoGBAOoy7/QOtTjQ0niE
925e6b
+6uDcTwtkC0R2Tvy1AjVnXohCntZfdzbTGDoYTgXSOLsP8A697jUiJ8VCePGH50xG
925e6b
+Z4DKnAF3a9O3a9nr2pLXb0iY3XOMv+YEBii7CfI+3oxFYgCl0sMgHzDD2ZTVYAsm
925e6b
+DWgLUVsE2gHEccRwrM2tPf2EgR+FAgMBAAECgYEA3qyfyYVSeTrTYxO93x6ZaVMu
925e6b
+A2IZp9zSxMQL9bKiI2GRj+cV2ebSCGbg2btFnD6qBor7FWsmYz+8g6FNN/9sY4az
925e6b
+61rMqMtQvLBe+7L8w70FeTze4qQ4Y1oQri0qD6tBWhDVlpnbI5Py9bkZKD67yVUk
925e6b
+elcEA/5x4PrYXkuqsAECQQD80NjT0mDvaY0JOOaQFSEpMv6QiUA8GGX8Xli7IoKb
925e6b
+tAolPG8rQBa+qSpcWfDMTrWw/aWHuMEEQoP/bVDH9W4FAkEA7SYQbBAKnojZ5A3G
925e6b
+kOHdV7aeivRQxQk/JN8Fb8oKB9Csvpv/BsuGxPKXHdhFa6CBTTsNRtHQw/szPo4l
925e6b
+xMIjgQJAPoMxqibR+0EBM6+TKzteSL6oPXsCnBl4Vk/J5vPgkbmR7KUl4+7j8N8J
925e6b
+b2554TrxKEN/w7CGYZRE6UrRd7ATNQJAWD7Yz41sli+wfPdPU2xo1BHljyl4wMk/
925e6b
+EPZYbI/PCbdyAH/F935WyQTIjNeEhZc1Zkq6FwdOWw8ns3hrv3rKgQJAHXv1BqUa
925e6b
+czGPIFxX2TNoqtcl6/En4vrxVB1wzsfzkkDAg98kBl7qsF+S3qujSzKikjeaVbI2
925e6b
+/CyWR2P3yLtOmA==
925e6b
+-----END PRIVATE KEY-----
925e6b
+-----BEGIN CERTIFICATE-----
925e6b
+MIIDcjCCAtugAwIBAgIJAN5dc9TOWjB7MA0GCSqGSIb3DQEBCwUAMF0xCzAJBgNV
925e6b
+BAYTAlhZMRcwFQYDVQQHDA5DYXN0bGUgQW50aHJheDEjMCEGA1UECgwaUHl0aG9u
925e6b
+IFNvZnR3YXJlIEZvdW5kYXRpb24xEDAOBgNVBAMMB2FsbHNhbnMwHhcNMTYwODA1
925e6b
+MTAyMTExWhcNMjYwODAzMTAyMTExWjBdMQswCQYDVQQGEwJYWTEXMBUGA1UEBwwO
925e6b
+Q2FzdGxlIEFudGhyYXgxIzAhBgNVBAoMGlB5dGhvbiBTb2Z0d2FyZSBGb3VuZGF0
925e6b
+aW9uMRAwDgYDVQQDDAdhbGxzYW5zMIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKB
925e6b
+gQDqMu/0DrU40NJ4hOrg3E8LZAtEdk78tQI1Z16IQp7WX3c20xg6GE4F0ji7D/AO
925e6b
+ve41IifFQnjxh+dMRmeAypwBd2vTt2vZ69qS129ImN1zjL/mBAYouwnyPt6MRWIA
925e6b
+pdLDIB8ww9mU1WALJg1oC1FbBNoBxHHEcKzNrT39hIEfhQIDAQABo4IBODCCATQw
925e6b
+ggEwBgNVHREEggEnMIIBI4IHYWxsc2Fuc6AeBgMqAwSgFwwVc29tZSBvdGhlciBp
925e6b
+ZGVudGlmaWVyoDUGBisGAQUCAqArMCmgEBsOS0VSQkVST1MuUkVBTE2hFTAToAMC
925e6b
+AQGhDDAKGwh1c2VybmFtZYEQdXNlckBleGFtcGxlLm9yZ4IPd3d3LmV4YW1wbGUu
925e6b
+b3JnpGcwZTELMAkGA1UEBhMCWFkxFzAVBgNVBAcMDkNhc3RsZSBBbnRocmF4MSMw
925e6b
+IQYDVQQKDBpQeXRob24gU29mdHdhcmUgRm91bmRhdGlvbjEYMBYGA1UEAwwPZGly
925e6b
+bmFtZSBleGFtcGxlhhdodHRwczovL3d3dy5weXRob24ub3JnL4cEfwAAAYcQAAAA
925e6b
+AAAAAAAAAAAAAAAAAYgEKgMEBTANBgkqhkiG9w0BAQsFAAOBgQAy16h+F+nOmeiT
925e6b
+VWR0fc8F/j6FcadbLseAUaogcC15OGxCl4UYpLV88HBkABOoGCpP155qwWTwOrdG
925e6b
+iYPGJSusf1OnJEbvzFejZf6u078bPd9/ZL4VWLjv+FPGkjd+N+/OaqMvgj8Lu99f
925e6b
+3Y/C4S7YbHxxwff6C6l2Xli+q6gnuQ==
925e6b
+-----END CERTIFICATE-----
925e6b
diff --git a/Lib/test/make_ssl_certs.py b/Lib/test/make_ssl_certs.py
925e6b
--- a/Lib/test/make_ssl_certs.py
925e6b
+++ b/Lib/test/make_ssl_certs.py
925e6b
@@ -20,7 +20,28 @@ req_template = """
925e6b
     CN                     = {hostname}
925e6b
 
925e6b
     [req_x509_extensions]
925e6b
-    subjectAltName         = DNS:{hostname}
925e6b
+    subjectAltName         = @san
925e6b
+
925e6b
+    [san]
925e6b
+    DNS.1 = {hostname}
925e6b
+    {extra_san}
925e6b
+
925e6b
+    [dir_sect]
925e6b
+    C                      = XY
925e6b
+    L                      = Castle Anthrax
925e6b
+    O                      = Python Software Foundation
925e6b
+    CN                     = dirname example
925e6b
+
925e6b
+    [princ_name]
925e6b
+    realm = EXP:0, GeneralString:KERBEROS.REALM
925e6b
+    principal_name = EXP:1, SEQUENCE:principal_seq
925e6b
+
925e6b
+    [principal_seq]
925e6b
+    name_type = EXP:0, INTEGER:1
925e6b
+    name_string = EXP:1, SEQUENCE:principals
925e6b
+
925e6b
+    [principals]
925e6b
+    princ1 = GeneralString:username
925e6b
 
925e6b
     [ ca ]
925e6b
     default_ca      = CA_default
925e6b
@@ -67,7 +88,7 @@ req_template = """
925e6b
 
925e6b
 here = os.path.abspath(os.path.dirname(__file__))
925e6b
 
925e6b
-def make_cert_key(hostname, sign=False):
925e6b
+def make_cert_key(hostname, sign=False, extra_san=''):
925e6b
     print("creating cert for " + hostname)
925e6b
     tempnames = []
925e6b
     for i in range(3):
925e6b
@@ -75,8 +96,9 @@ def make_cert_key(hostname, sign=False):
925e6b
             tempnames.append(f.name)
925e6b
     req_file, cert_file, key_file = tempnames
925e6b
     try:
925e6b
+        req = req_template.format(hostname=hostname, extra_san=extra_san)
925e6b
         with open(req_file, 'w') as f:
925e6b
-            f.write(req_template.format(hostname=hostname))
925e6b
+            f.write(req)
925e6b
         args = ['req', '-new', '-days', '3650', '-nodes',
925e6b
                 '-newkey', 'rsa:1024', '-keyout', key_file,
925e6b
                 '-config', req_file]
925e6b
@@ -120,7 +142,7 @@ def make_ca():
925e6b
         f.write('unique_subject = no')
925e6b
 
925e6b
     with tempfile.NamedTemporaryFile("w") as t:
925e6b
-        t.write(req_template.format(hostname='our-ca-server'))
925e6b
+        t.write(req_template.format(hostname='our-ca-server', extra_san=''))
925e6b
         t.flush()
925e6b
         with tempfile.NamedTemporaryFile() as f:
925e6b
             args = ['req', '-new', '-days', '3650', '-extensions', 'v3_ca', '-nodes',
925e6b
@@ -171,6 +193,25 @@ if __name__ == '__main__':
925e6b
         f.write(key)
925e6b
         f.write(cert)
925e6b
 
925e6b
+    extra_san = [
925e6b
+        'otherName.1 = 1.2.3.4;UTF8:some other identifier',
925e6b
+        'otherName.2 = 1.3.6.1.5.2.2;SEQUENCE:princ_name',
925e6b
+        'email.1 = user@example.org',
925e6b
+        'DNS.2 = www.example.org',
925e6b
+        # GEN_X400
925e6b
+        'dirName.1 = dir_sect',
925e6b
+        # GEN_EDIPARTY
925e6b
+        'URI.1 = https://www.python.org/',
925e6b
+        'IP.1 = 127.0.0.1',
925e6b
+        'IP.2 = ::1',
925e6b
+        'RID.1 = 1.2.3.4.5',
925e6b
+    ]
925e6b
+
925e6b
+    cert, key = make_cert_key('allsans', extra_san='\n'.join(extra_san))
925e6b
+    with open('allsans.pem', 'w') as f:
925e6b
+        f.write(key)
925e6b
+        f.write(cert)
925e6b
+
925e6b
     unmake_ca()
925e6b
     print("\n\nPlease change the values in test_ssl.py, test_parse_cert function related to notAfter,notBefore and serialNumber")
925e6b
     check_call(['openssl','x509','-in','keycert.pem','-dates','-serial','-noout'])
925e6b
diff --git a/Lib/test/test_ssl.py b/Lib/test/test_ssl.py
925e6b
diff --git a/Lib/test/test_ssl.py b/Lib/test/test_ssl.py
925e6b
index fa59641..9d5816b 100644
925e6b
--- a/Lib/test/test_ssl.py
925e6b
+++ b/Lib/test/test_ssl.py
925e6b
@@ -57,6 +57,8 @@ CRLFILE = data_file("revocation.crl")
925e6b
 SIGNED_CERTFILE = data_file("keycert3.pem")
925e6b
 SIGNED_CERTFILE2 = data_file("keycert4.pem")
925e6b
 SIGNING_CA = data_file("pycacert.pem")
925e6b
+# cert with all kinds of subject alt names
925e6b
+ALLSANFILE = data_file("allsans.pem")
925e6b
 
925e6b
 SVN_PYTHON_ORG_ROOT_CERT = data_file("https_svn_python_org_root.pem")
925e6b
 
925e6b
@@ -236,6 +238,28 @@ class BasicSocketTests(unittest.TestCase):
925e6b
                          ('IP Address', '2001:DB8:0:0:0:0:0:1\n'))
925e6b
                         )
925e6b
 
925e6b
+    def test_parse_all_sans(self):
925e6b
+        p = ssl._ssl._test_decode_cert(ALLSANFILE)
925e6b
+        self.assertEqual(p['subjectAltName'],
925e6b
+            (
925e6b
+                ('DNS', 'allsans'),
925e6b
+                ('othername', '<unsupported>'),
925e6b
+                ('othername', '<unsupported>'),
925e6b
+                ('email', 'user@example.org'),
925e6b
+                ('DNS', 'www.example.org'),
925e6b
+                ('DirName',
925e6b
+                    ((('countryName', 'XY'),),
925e6b
+                    (('localityName', 'Castle Anthrax'),),
925e6b
+                    (('organizationName', 'Python Software Foundation'),),
925e6b
+                    (('commonName', 'dirname example'),))),
925e6b
+                ('URI', 'https://www.python.org/'),
925e6b
+                ('IP Address', '127.0.0.1'),
925e6b
+                ('IP Address', '0:0:0:0:0:0:0:1\n'),
925e6b
+                ('Registered ID', '1.2.3.4.5')
925e6b
+            )
925e6b
+        )
925e6b
+
925e6b
+
925e6b
     def test_DER_to_PEM(self):
925e6b
         with open(SVN_PYTHON_ORG_ROOT_CERT, 'r') as f:
925e6b
             pem = f.read()
925e6b
diff --git a/Modules/_ssl.c b/Modules/_ssl.c
925e6b
--- a/Modules/_ssl.c
925e6b
+++ b/Modules/_ssl.c
925e6b
@@ -953,6 +953,35 @@ static PyObject *
925e6b
                 PyTuple_SET_ITEM(t, 1, v);
925e6b
                 break;
925e6b
 
925e6b
+            case GEN_RID:
925e6b
+                t = PyTuple_New(2);
925e6b
+                if (t == NULL)
925e6b
+                    goto fail;
925e6b
+
925e6b
+                v = PyUnicode_FromString("Registered ID");
925e6b
+                if (v == NULL) {
925e6b
+                    Py_DECREF(t);
925e6b
+                    goto fail;
925e6b
+                }
925e6b
+                PyTuple_SET_ITEM(t, 0, v);
925e6b
+
925e6b
+                len = i2t_ASN1_OBJECT(buf, sizeof(buf)-1, name->d.rid);
925e6b
+                if (len < 0) {
925e6b
+                    Py_DECREF(t);
925e6b
+                    _setSSLError(NULL, 0, __FILE__, __LINE__);
925e6b
+                    goto fail;
925e6b
+                } else if (len >= (int)sizeof(buf)) {
925e6b
+                    v = PyUnicode_FromString("<INVALID>");
925e6b
+                } else {
925e6b
+                    v = PyUnicode_FromStringAndSize(buf, len);
925e6b
+                }
925e6b
+                if (v == NULL) {
925e6b
+                    Py_DECREF(t);
925e6b
+                    goto fail;
925e6b
+                }
925e6b
+                PyTuple_SET_ITEM(t, 1, v);
925e6b
+                break;
925e6b
+
925e6b
             default:
925e6b
                 /* for everything else, we use the OpenSSL print form */
925e6b
                 switch (gntype) {
925e6b
@@ -978,8 +1007,12 @@ static PyObject *
925e6b
                     goto fail;
925e6b
                 }
925e6b
                 vptr = strchr(buf, ':');
925e6b
-                if (vptr == NULL)
925e6b
+                if (vptr == NULL) {
925e6b
+                    PyErr_Format(PyExc_ValueError,
925e6b
+                                 "Invalid value %.200s",
925e6b
+                                 buf);
925e6b
                     goto fail;
925e6b
+                }
925e6b
                 t = PyTuple_New(2);
925e6b
                 if (t == NULL)
925e6b
                     goto fail;
925e6b