blob: 7d46f91f6d2cdbfdc5c7019d80b6a2357b861765 [file] [log] [blame]
Jean-Paul Calderone8b63d452008-03-21 18:31:12 -04001# Copyright (C) Jean-Paul Calderone 2008, All rights reserved
2
Jean-Paul Calderoned8782ad2008-03-04 23:39:59 -05003"""
4Unit tests for L{OpenSSL.crypto}.
5"""
6
7from unittest import TestCase
8
9from OpenSSL.crypto import TYPE_RSA, TYPE_DSA, Error, PKey, PKeyType
Jean-Paul Calderone78381d22008-03-06 23:35:22 -050010from OpenSSL.crypto import X509, X509Type, X509Name, X509NameType
Jean-Paul Calderone2aa2b332008-03-06 21:43:14 -050011from OpenSSL.crypto import X509Req, X509ReqType
Jean-Paul Calderone828c9cb2008-04-26 18:06:54 -040012from OpenSSL.crypto import FILETYPE_PEM, load_certificate, load_privatekey
13from OpenSSL.crypto import dump_privatekey
14
15
16cleartextPrivateKeyPEM = (
17 "-----BEGIN RSA PRIVATE KEY-----\n"
18 "MIICXAIBAAKBgQDaemNe1syksAbFFpF3aoOrZ18vB/IQNZrAjFqXPv9iieJm7+Tc\n"
19 "g+lA/v0qmoEKrpT2xfwxXmvZwBNM4ZhyRC3DPIFEyJV7/3IA1p5iuMY/GJI1VIgn\n"
20 "aikQCnrsyxtaRpsMBeZRniaVzcUJ+XnEdFGEjlo+k0xlwfVclDEMwgpXAQIDAQAB\n"
21 "AoGBALi0a7pMQqqgnriVAdpBVJveQtxSDVWi2/gZMKVZfzNheuSnv4amhtaKPKJ+\n"
22 "CMZtHkcazsE2IFvxRN/kgato9H3gJqq8nq2CkdpdLNVKBoxiCtkLfutdY4SQLtoY\n"
23 "USN7exk131pchsAJXYlR6mCW+ZP+E523cNwpPgsyKxVbmXSBAkEA9470fy2W0jFM\n"
24 "taZFslpntKSzbvn6JmdtjtvWrM1bBaeeqFiGBuQFYg46VaCUaeRWYw02jmYAsDYh\n"
25 "ZQavmXThaQJBAOHtlAQ0IJJEiMZr6vtVPH32fmbthSv1AUSYPzKqdlQrUnOXPQXu\n"
26 "z70cFoLG1TvPF5rBxbOkbQ/s8/ka5ZjPfdkCQCeC7YsO36+UpsWnUCBzRXITh4AC\n"
27 "7eYLQ/U1KUJTVF/GrQ/5cQrQgftwgecAxi9Qfmk4xqhbp2h4e0QAmS5I9WECQH02\n"
28 "0QwrX8nxFeTytr8pFGezj4a4KVCdb2B3CL+p3f70K7RIo9d/7b6frJI6ZL/LHQf2\n"
29 "UP4pKRDkgKsVDx7MELECQGm072/Z7vmb03h/uE95IYJOgY4nfmYs0QKA9Is18wUz\n"
30 "DpjfE33p0Ha6GO1VZRIQoqE24F8o5oimy3BEjryFuw4=\n"
31 "-----END RSA PRIVATE KEY-----\n")
32
33
34encryptedPrivateKeyPEM = (
35 "-----BEGIN RSA PRIVATE KEY-----\n"
36 "Proc-Type: 4,ENCRYPTED\n"
37 "DEK-Info: BF-CBC,8306665233D056B1\n"
38 "\n"
39 "BwxghOcX1F+M108qRGBfpUBrfaeKOszDEV18OjEE55p0yGsiDxvdol3c4bwI5ITy\n"
40 "ltP8w9O33CDUCjr+Ymj8xLpPP60TTfr/aHq+2fEuG4TfkeHb5fVYm0mgVnaOhJs3\n"
41 "a2n5IL/KNCdP3zMZa0IaMJ0M+VK90SLpq5nzXOWkufLyZL1+n8srkk06gepmHS7L\n"
42 "rH3rALNboG8yTH1qjE8PwcMrJAQfRMd4/4RTQv+4pUuKj7I2en+YwSQ/gomy7qN1\n"
43 "3s/gMgV/2GUbEcTVch4thZ9l3WsX18V76rBQkiZ7yrJkxwNMv+Qc2GfHtBnsXAyA\n"
44 "0nIE4Mm/OQqX8h7EJ4c2s1DMGVS0YZGU+75HN0A3iD01h8C5utqSScWzBA45j/Vy\n"
45 "3aypQVqQeW7kBMQlpc6pHvJ1EsjiAJRCto7tZNLxRdjMKBV4w75JNLaAFSraqA+R\n"
46 "/WPcdcXAQuhmCeh31fzmVOHJGRF7/5pAR/b7AnFTD4YbYVcglNis/jpdiI9k2AYP\n"
47 "wZNwXOIh6Ibq5hMvyV4/pySyLbgDOrfrOGpi8N6lBbzewByYQKiXwUEZf+Y5499/\n"
48 "CckajBhgYynPpe6mgsSeklWGc845iIwAtzavBNZIkn1hKP1P+TFjbl2O75u/9JLJ\n"
49 "6i4IFYCyQmwiHX8nTR717SpCN2gyZ2HrX7z2mKP/KokkAX2yidwoKh9FMUV5lOGO\n"
50 "JPc4MfPo4lPB7SP30AtOh7y7zlS3x8Uo0+0wCg5Z5Fn/73x3W+p5nyI0G9n7RGzL\n"
51 "ZeCWLdG/Cm6ZyIpYZGbZ5m+U3Fr6/El9V6LSxrB1TB+8G1NTdLlbeA==\n"
52 "-----END RSA PRIVATE KEY-----\n")
53encryptedPrivateKeyPEMPassphrase = "foobar"
54
55
Jean-Paul Calderoned8782ad2008-03-04 23:39:59 -050056
Jean-Paul Calderoneac930e12008-03-06 18:50:51 -050057class _Python23TestCaseHelper:
Jean-Paul Calderone7da26a72008-03-06 00:35:20 -050058 # Python 2.3 compatibility.
59 def assertTrue(self, *a, **kw):
60 return self.failUnless(*a, **kw)
61
62
Jean-Paul Calderone7535dab2008-03-06 18:53:11 -050063 def assertFalse(self, *a, **kw):
64 return self.failIf(*a, **kw)
65
66
67
Jean-Paul Calderoneac930e12008-03-06 18:50:51 -050068class PKeyTests(TestCase, _Python23TestCaseHelper):
69 """
70 Unit tests for L{OpenSSL.crypto.PKey}.
71 """
Jean-Paul Calderoned8782ad2008-03-04 23:39:59 -050072 def test_construction(self):
73 """
74 L{PKey} takes no arguments and returns a new L{PKeyType} instance.
75 """
76 self.assertRaises(TypeError, PKey, None)
77 key = PKey()
78 self.assertTrue(
79 isinstance(key, PKeyType),
80 "%r is of type %r, should be %r" % (key, type(key), PKeyType))
81
82
83 def test_pregeneration(self):
84 """
85 L{PKeyType.bits} and L{PKeyType.type} return C{0} before the key is
86 generated.
87 """
88 key = PKey()
89 self.assertEqual(key.type(), 0)
90 self.assertEqual(key.bits(), 0)
91
92
93 def test_failedGeneration(self):
94 """
Jean-Paul Calderoneab82db72008-03-06 00:09:31 -050095 L{PKeyType.generate_key} takes two arguments, the first giving the key
96 type as one of L{TYPE_RSA} or L{TYPE_DSA} and the second giving the
97 number of bits to generate. If an invalid type is specified or
98 generation fails, L{Error} is raised. If an invalid number of bits is
99 specified, L{ValueError} or L{Error} is raised.
Jean-Paul Calderoned8782ad2008-03-04 23:39:59 -0500100 """
101 key = PKey()
102 self.assertRaises(TypeError, key.generate_key)
103 self.assertRaises(TypeError, key.generate_key, 1, 2, 3)
104 self.assertRaises(TypeError, key.generate_key, "foo", "bar")
105 self.assertRaises(Error, key.generate_key, -1, 0)
Jean-Paul Calderoneab82db72008-03-06 00:09:31 -0500106
Jean-Paul Calderoneab82db72008-03-06 00:09:31 -0500107 self.assertRaises(ValueError, key.generate_key, TYPE_RSA, -1)
108 self.assertRaises(ValueError, key.generate_key, TYPE_RSA, 0)
Jean-Paul Calderoned71fe982008-03-06 00:31:50 -0500109
110 # XXX RSA generation for small values of bits is fairly buggy in a wide
111 # range of OpenSSL versions. I need to figure out what the safe lower
112 # bound for a reasonable number of OpenSSL versions is and explicitly
113 # check for that in the wrapper. The failure behavior is typically an
114 # infinite loop inside OpenSSL.
115
116 # self.assertRaises(Error, key.generate_key, TYPE_RSA, 2)
Jean-Paul Calderoned8782ad2008-03-04 23:39:59 -0500117
118 # XXX DSA generation seems happy with any number of bits. The DSS
119 # says bits must be between 512 and 1024 inclusive. OpenSSL's DSA
120 # generator doesn't seem to care about the upper limit at all. For
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500121 # the lower limit, it uses 512 if anything smaller is specified.
Jean-Paul Calderoned8782ad2008-03-04 23:39:59 -0500122 # So, it doesn't seem possible to make generate_key fail for
123 # TYPE_DSA with a bits argument which is at least an int.
124
125 # self.assertRaises(Error, key.generate_key, TYPE_DSA, -7)
126
127
128 def test_rsaGeneration(self):
129 """
130 L{PKeyType.generate_key} generates an RSA key when passed
131 L{TYPE_RSA} as a type and a reasonable number of bits.
132 """
133 bits = 128
134 key = PKey()
135 key.generate_key(TYPE_RSA, bits)
136 self.assertEqual(key.type(), TYPE_RSA)
137 self.assertEqual(key.bits(), bits)
138
139
140 def test_dsaGeneration(self):
141 """
142 L{PKeyType.generate_key} generates a DSA key when passed
143 L{TYPE_DSA} as a type and a reasonable number of bits.
144 """
145 # 512 is a magic number. The DSS (Digital Signature Standard)
146 # allows a minimum of 512 bits for DSA. DSA_generate_parameters
147 # will silently promote any value below 512 to 512.
148 bits = 512
149 key = PKey()
150 key.generate_key(TYPE_DSA, bits)
151 self.assertEqual(key.type(), TYPE_DSA)
152 self.assertEqual(key.bits(), bits)
153
154
155 def test_regeneration(self):
156 """
157 L{PKeyType.generate_key} can be called multiple times on the same
158 key to generate new keys.
159 """
160 key = PKey()
161 for type, bits in [(TYPE_RSA, 512), (TYPE_DSA, 576)]:
162 key.generate_key(type, bits)
163 self.assertEqual(key.type(), type)
164 self.assertEqual(key.bits(), bits)
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500165
166
167
Jean-Paul Calderoneac930e12008-03-06 18:50:51 -0500168class X509NameTests(TestCase, _Python23TestCaseHelper):
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500169 """
170 Unit tests for L{OpenSSL.crypto.X509Name}.
171 """
Jean-Paul Calderonee098dc72008-03-06 18:36:19 -0500172 def _x509name(self, **attrs):
173 # XXX There's no other way to get a new X509Name yet.
174 name = X509().get_subject()
175 attrs = attrs.items()
176 # Make the order stable - order matters!
177 attrs.sort(lambda (k1, v1), (k2, v2): cmp(v1, v2))
178 for k, v in attrs:
179 setattr(name, k, v)
180 return name
181
182
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500183 def test_attributes(self):
184 """
185 L{X509NameType} instances have attributes for each standard (?)
186 X509Name field.
187 """
Jean-Paul Calderonee098dc72008-03-06 18:36:19 -0500188 name = self._x509name()
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500189 name.commonName = "foo"
190 self.assertEqual(name.commonName, "foo")
191 self.assertEqual(name.CN, "foo")
192 name.CN = "baz"
193 self.assertEqual(name.commonName, "baz")
194 self.assertEqual(name.CN, "baz")
195 name.commonName = "bar"
196 self.assertEqual(name.commonName, "bar")
197 self.assertEqual(name.CN, "bar")
198 name.CN = "quux"
199 self.assertEqual(name.commonName, "quux")
200 self.assertEqual(name.CN, "quux")
201
202
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500203 def test_copy(self):
204 """
205 L{X509Name} creates a new L{X509NameType} instance with all the same
206 attributes as an existing L{X509NameType} instance when called with
207 one.
208 """
Jean-Paul Calderonee098dc72008-03-06 18:36:19 -0500209 name = self._x509name(commonName="foo", emailAddress="bar@example.com")
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500210
211 copy = X509Name(name)
212 self.assertEqual(copy.commonName, "foo")
213 self.assertEqual(copy.emailAddress, "bar@example.com")
Jean-Paul Calderonee098dc72008-03-06 18:36:19 -0500214
215 # Mutate the copy and ensure the original is unmodified.
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500216 copy.commonName = "baz"
217 self.assertEqual(name.commonName, "foo")
Jean-Paul Calderonee098dc72008-03-06 18:36:19 -0500218
219 # Mutate the original and ensure the copy is unmodified.
Jean-Paul Calderoneeff3cd92008-03-05 22:35:26 -0500220 name.emailAddress = "quux@example.com"
221 self.assertEqual(copy.emailAddress, "bar@example.com")
222
Jean-Paul Calderonee098dc72008-03-06 18:36:19 -0500223
224 def test_repr(self):
225 """
226 L{repr} passed an L{X509NameType} instance should return a string
227 containing a description of the type and the NIDs which have been set
228 on it.
229 """
230 name = self._x509name(commonName="foo", emailAddress="bar")
231 self.assertEqual(
232 repr(name),
233 "<X509Name object '/emailAddress=bar/CN=foo'>")
234
235
236 def test_comparison(self):
237 """
238 L{X509NameType} instances should compare based on their NIDs.
239 """
240 def _equality(a, b, assertTrue, assertFalse):
241 assertTrue(a == b, "(%r == %r) --> False" % (a, b))
242 assertFalse(a != b)
243 assertTrue(b == a)
244 assertFalse(b != a)
245
246 def assertEqual(a, b):
247 _equality(a, b, self.assertTrue, self.assertFalse)
248
249 # Instances compare equal to themselves.
250 name = self._x509name()
251 assertEqual(name, name)
252
253 # Empty instances should compare equal to each other.
254 assertEqual(self._x509name(), self._x509name())
255
256 # Instances with equal NIDs should compare equal to each other.
257 assertEqual(self._x509name(commonName="foo"),
258 self._x509name(commonName="foo"))
259
260 # Instance with equal NIDs set using different aliases should compare
261 # equal to each other.
262 assertEqual(self._x509name(commonName="foo"),
263 self._x509name(CN="foo"))
264
265 # Instances with more than one NID with the same values should compare
266 # equal to each other.
267 assertEqual(self._x509name(CN="foo", organizationalUnitName="bar"),
268 self._x509name(commonName="foo", OU="bar"))
269
270 def assertNotEqual(a, b):
271 _equality(a, b, self.assertFalse, self.assertTrue)
272
273 # Instances with different values for the same NID should not compare
274 # equal to each other.
275 assertNotEqual(self._x509name(CN="foo"),
276 self._x509name(CN="bar"))
277
278 # Instances with different NIDs should not compare equal to each other.
279 assertNotEqual(self._x509name(CN="foo"),
280 self._x509name(OU="foo"))
281
282 def _inequality(a, b, assertTrue, assertFalse):
283 assertTrue(a < b)
284 assertTrue(a <= b)
285 assertTrue(b > a)
286 assertTrue(b >= a)
287 assertFalse(a > b)
288 assertFalse(a >= b)
289 assertFalse(b < a)
290 assertFalse(b <= a)
291
292 def assertLessThan(a, b):
293 _inequality(a, b, self.assertTrue, self.assertFalse)
294
295 # An X509Name with a NID with a value which sorts less than the value
296 # of the same NID on another X509Name compares less than the other
297 # X509Name.
298 assertLessThan(self._x509name(CN="abc"),
299 self._x509name(CN="def"))
300
301 def assertGreaterThan(a, b):
302 _inequality(a, b, self.assertFalse, self.assertTrue)
303
304 # An X509Name with a NID with a value which sorts greater than the
305 # value of the same NID on another X509Name compares greater than the
306 # other X509Name.
307 assertGreaterThan(self._x509name(CN="def"),
308 self._x509name(CN="abc"))
Jean-Paul Calderone2aa2b332008-03-06 21:43:14 -0500309
310
Jean-Paul Calderone110cd092008-03-24 17:27:42 -0400311 def test_hash(self):
312 """
313 L{X509Name.hash} returns an integer hash based on the value of the
314 name.
315 """
316 a = self._x509name(CN="foo")
317 b = self._x509name(CN="foo")
318 self.assertEqual(a.hash(), b.hash())
319 a.CN = "bar"
320 self.assertNotEqual(a.hash(), b.hash())
321
322
Jean-Paul Calderonee957a002008-03-25 15:16:51 -0400323 def test_der(self):
324 """
325 L{X509Name.der} returns the DER encoded form of the name.
326 """
327 a = self._x509name(CN="foo", C="US")
328 self.assertEqual(
329 a.der(),
330 '0\x1b1\x0b0\t\x06\x03U\x04\x06\x13\x02US'
331 '1\x0c0\n\x06\x03U\x04\x03\x13\x03foo')
332
333
Jean-Paul Calderonec54cc182008-03-26 21:11:07 -0400334 def test_get_components(self):
335 """
336 L{X509Name.get_components} returns a C{list} of two-tuples of C{str}
337 giving the NIDs and associated values which make up the name.
338 """
339 a = self._x509name()
340 self.assertEqual(a.get_components(), [])
341 a.CN = "foo"
342 self.assertEqual(a.get_components(), [("CN", "foo")])
343 a.organizationalUnitName = "bar"
344 self.assertEqual(
345 a.get_components(),
346 [("CN", "foo"), ("OU", "bar")])
347
Jean-Paul Calderone110cd092008-03-24 17:27:42 -0400348
Jean-Paul Calderoneac0d95f2008-03-10 00:00:42 -0400349class _PKeyInteractionTestsMixin:
350 """
351 Tests which involve another thing and a PKey.
352 """
353 def signable(self):
354 """
355 Return something with a C{set_pubkey}, C{set_pubkey}, and C{sign} method.
356 """
357 raise NotImplementedError()
358
359
360 def test_signWithUngenerated(self):
361 """
362 L{X509Req.sign} raises L{ValueError} when pass a L{PKey} with no parts.
363 """
364 request = self.signable()
365 key = PKey()
366 self.assertRaises(ValueError, request.sign, key, 'MD5')
367
368
369 def test_signWithPublicKey(self):
370 """
371 L{X509Req.sign} raises L{ValueError} when pass a L{PKey} with no
372 private part as the signing key.
373 """
374 request = self.signable()
375 key = PKey()
376 key.generate_key(TYPE_RSA, 512)
377 request.set_pubkey(key)
378 pub = request.get_pubkey()
379 self.assertRaises(ValueError, request.sign, pub, 'MD5')
380
381
382
383class X509ReqTests(TestCase, _PKeyInteractionTestsMixin, _Python23TestCaseHelper):
Jean-Paul Calderone2aa2b332008-03-06 21:43:14 -0500384 """
385 Tests for L{OpenSSL.crypto.X509Req}.
386 """
Jean-Paul Calderoneac0d95f2008-03-10 00:00:42 -0400387 def signable(self):
388 """
389 Create and return a new L{X509Req}.
390 """
391 return X509Req()
392
393
Jean-Paul Calderone2aa2b332008-03-06 21:43:14 -0500394 def test_construction(self):
395 """
396 L{X509Req} takes no arguments and returns an L{X509ReqType} instance.
397 """
398 request = X509Req()
399 self.assertTrue(
400 isinstance(request, X509ReqType),
401 "%r is of type %r, should be %r" % (request, type(request), X509ReqType))
402
403
404 def test_get_subject(self):
405 """
406 L{X509ReqType.get_subject} returns an L{X509Name} for the subject of
407 the request and which is valid even after the request object is
408 otherwise dead.
409 """
410 request = X509Req()
411 subject = request.get_subject()
412 self.assertTrue(
413 isinstance(subject, X509NameType),
414 "%r is of type %r, should be %r" % (subject, type(subject), X509NameType))
415 subject.commonName = "foo"
416 self.assertEqual(request.get_subject().commonName, "foo")
417 del request
418 subject.commonName = "bar"
419 self.assertEqual(subject.commonName, "bar")
Jean-Paul Calderone78381d22008-03-06 23:35:22 -0500420
421
422
Jean-Paul Calderoneac0d95f2008-03-10 00:00:42 -0400423class X509Tests(TestCase, _PKeyInteractionTestsMixin, _Python23TestCaseHelper):
Jean-Paul Calderone78381d22008-03-06 23:35:22 -0500424 """
425 Tests for L{OpenSSL.crypto.X509}.
426 """
Jean-Paul Calderone8114b452008-03-25 15:27:59 -0400427 pemData = """
428-----BEGIN CERTIFICATE-----
429MIICfTCCAeYCAQEwDQYJKoZIhvcNAQEEBQAwgYYxCzAJBgNVBAYTAlVTMRkwFwYD
430VQQDExBweW9wZW5zc2wuc2YubmV0MREwDwYDVQQHEwhOZXcgWW9yazESMBAGA1UE
431ChMJUHlPcGVuU1NMMREwDwYDVQQIEwhOZXcgWW9yazEQMA4GCSqGSIb3DQEJARYB
432IDEQMA4GA1UECxMHVGVzdGluZzAeFw0wODAzMjUxOTA0MTNaFw0wOTAzMjUxOTA0
433MTNaMIGGMQswCQYDVQQGEwJVUzEZMBcGA1UEAxMQcHlvcGVuc3NsLnNmLm5ldDER
434MA8GA1UEBxMITmV3IFlvcmsxEjAQBgNVBAoTCVB5T3BlblNTTDERMA8GA1UECBMI
435TmV3IFlvcmsxEDAOBgkqhkiG9w0BCQEWASAxEDAOBgNVBAsTB1Rlc3RpbmcwgZ8w
436DQYJKoZIhvcNAQEBBQADgY0AMIGJAoGBANp6Y17WzKSwBsUWkXdqg6tnXy8H8hA1
437msCMWpc+/2KJ4mbv5NyD6UD+/SqagQqulPbF/DFea9nAE0zhmHJELcM8gUTIlXv/
438cgDWnmK4xj8YkjVUiCdqKRAKeuzLG1pGmwwF5lGeJpXNxQn5ecR0UYSOWj6TTGXB
4399VyUMQzCClcBAgMBAAEwDQYJKoZIhvcNAQEEBQADgYEAmm0Vzvv1O91WLl2LnF2P
440q55LJdOnJbCCXIgxLdoVmvYAz1ZJq1eGKgKWI5QLgxiSzJLEU7KK//aVfiZzoCd5
441RipBiEEMEV4eAY317bHPwPP+4Bj9t0l8AsDLseC5vLRHgxrLEu3bn08DYx6imB5Q
442UBj849/xpszEM7BhwKE0GiQ=
443-----END CERTIFICATE-----
Jean-Paul Calderone828c9cb2008-04-26 18:06:54 -0400444""" + cleartextPrivateKeyPEM
Jean-Paul Calderone8114b452008-03-25 15:27:59 -0400445
Jean-Paul Calderoneac0d95f2008-03-10 00:00:42 -0400446 def signable(self):
447 """
448 Create and return a new L{X509}.
449 """
450 return X509()
451
452
Jean-Paul Calderone78381d22008-03-06 23:35:22 -0500453 def test_construction(self):
454 """
455 L{X509} takes no arguments and returns an instance of L{X509Type}.
456 """
457 certificate = X509()
458 self.assertTrue(
459 isinstance(certificate, X509Type),
460 "%r is of type %r, should be %r" % (certificate,
461 type(certificate),
462 X509Type))
463
464
465 def test_serial_number(self):
466 """
467 The serial number of an L{X509Type} can be retrieved and modified with
468 L{X509Type.get_serial_number} and L{X509Type.set_serial_number}.
469 """
470 certificate = X509()
471 self.assertRaises(TypeError, certificate.set_serial_number)
472 self.assertRaises(TypeError, certificate.set_serial_number, 1, 2)
473 self.assertRaises(TypeError, certificate.set_serial_number, "1")
474 self.assertRaises(TypeError, certificate.set_serial_number, 5.5)
475 self.assertEqual(certificate.get_serial_number(), 0)
476 certificate.set_serial_number(1)
477 self.assertEqual(certificate.get_serial_number(), 1)
478 certificate.set_serial_number(2 ** 32 + 1)
479 self.assertEqual(certificate.get_serial_number(), 2 ** 32 + 1)
480 certificate.set_serial_number(2 ** 64 + 1)
481 self.assertEqual(certificate.get_serial_number(), 2 ** 64 + 1)
Jean-Paul Calderone525ef802008-03-09 20:39:42 -0400482 certificate.set_serial_number(2 ** 128 + 1)
483 self.assertEqual(certificate.get_serial_number(), 2 ** 128 + 1)
484
485
486 def _setBoundTest(self, which):
487 """
488 L{X509Type.set_notBefore} takes a string in the format of an ASN1
489 GENERALIZEDTIME and sets the beginning of the certificate's validity
490 period to it.
491 """
492 certificate = X509()
493 set = getattr(certificate, 'set_not' + which)
494 get = getattr(certificate, 'get_not' + which)
495
Jean-Paul Calderonee0615b52008-03-09 21:44:46 -0400496 # Starts with no value.
497 self.assertEqual(get(), None)
498
Jean-Paul Calderone525ef802008-03-09 20:39:42 -0400499 # GMT (Or is it UTC?) -exarkun
500 when = "20040203040506Z"
501 set(when)
502 self.assertEqual(get(), when)
503
504 # A plus two hours and thirty minutes offset
505 when = "20040203040506+0530"
506 set(when)
507 self.assertEqual(get(), when)
508
509 # A minus one hour fifteen minutes offset
510 when = "20040203040506-0115"
511 set(when)
512 self.assertEqual(get(), when)
513
514 # An invalid string results in a ValueError
515 self.assertRaises(ValueError, set, "foo bar")
516
517
518 def test_set_notBefore(self):
519 """
520 L{X509Type.set_notBefore} takes a string in the format of an ASN1
521 GENERALIZEDTIME and sets the beginning of the certificate's validity
522 period to it.
523 """
524 self._setBoundTest("Before")
525
526
527 def test_set_notAfter(self):
528 """
529 L{X509Type.set_notAfter} takes a string in the format of an ASN1
530 GENERALIZEDTIME and sets the end of the certificate's validity period
531 to it.
532 """
533 self._setBoundTest("After")
Jean-Paul Calderone76576d52008-03-24 16:04:46 -0400534
535
Jean-Paul Calderone38a646d2008-03-25 15:16:18 -0400536 def test_get_notBefore(self):
537 """
538 L{X509Type.get_notBefore} returns a string in the format of an ASN1
539 GENERALIZEDTIME even for certificates which store it as UTCTIME
540 internally.
541 """
Jean-Paul Calderone8114b452008-03-25 15:27:59 -0400542 cert = load_certificate(FILETYPE_PEM, self.pemData)
Jean-Paul Calderone38a646d2008-03-25 15:16:18 -0400543 self.assertEqual(cert.get_notBefore(), "20080325190413Z")
544
545
546 def test_get_notAfter(self):
547 """
548 L{X509Type.get_notAfter} returns a string in the format of an ASN1
549 GENERALIZEDTIME even for certificates which store it as UTCTIME
550 internally.
551 """
Jean-Paul Calderone8114b452008-03-25 15:27:59 -0400552 cert = load_certificate(FILETYPE_PEM, self.pemData)
Jean-Paul Calderone38a646d2008-03-25 15:16:18 -0400553 self.assertEqual(cert.get_notAfter(), "20090325190413Z")
554
555
Jean-Paul Calderone76576d52008-03-24 16:04:46 -0400556 def test_digest(self):
557 """
558 L{X509.digest} returns a string giving ":"-separated hex-encoded words
559 of the digest of the certificate.
560 """
561 cert = X509()
562 self.assertEqual(
563 cert.digest("md5"),
564 "A8:EB:07:F8:53:25:0A:F2:56:05:C5:A5:C4:C4:C7:15")
Jean-Paul Calderone828c9cb2008-04-26 18:06:54 -0400565
566
567
568class FunctionTests(TestCase):
569 """
570 Tests for free-functions in the L{OpenSSL.crypto} module.
571 """
572 def test_load_privatekey_wrongPassphrase(self):
573 """
574 L{load_privatekey} raises L{OpenSSL.crypto.Error} when it is passed an
575 encrypted PEM and an incorrect passphrase.
576 """
577 self.assertRaises(
578 Error,
579 load_privatekey, FILETYPE_PEM, encryptedPrivateKeyPEM, "quack")
580
581
582 def test_load_privatekey_passphrase(self):
583 """
584 L{load_privatekey} can create a L{PKey} object from an encrypted PEM
585 string if given the passphrase.
586 """
587 key = load_privatekey(
588 FILETYPE_PEM, encryptedPrivateKeyPEM,
589 encryptedPrivateKeyPEMPassphrase)
590 self.assertTrue(isinstance(key, PKeyType))
591
592
593 def test_load_privatekey_wrongPassphraseCallback(self):
594 """
595 L{load_privatekey} raises L{OpenSSL.crypto.Error} when it is passed an
596 encrypted PEM and a passphrase callback which returns an incorrect
597 passphrase.
598 """
599 called = []
600 def cb(*a):
601 called.append(None)
602 return "quack"
603 self.assertRaises(
604 Error,
605 load_privatekey, FILETYPE_PEM, encryptedPrivateKeyPEM, cb)
606 self.assertTrue(called)
607
608 def test_load_privatekey_passphraseCallback(self):
609 """
610 L{load_privatekey} can create a L{PKey} object from an encrypted PEM
611 string if given a passphrase callback which returns the correct
612 password.
613 """
614 called = []
615 def cb(writing):
616 called.append(writing)
617 return encryptedPrivateKeyPEMPassphrase
618 key = load_privatekey(FILETYPE_PEM, encryptedPrivateKeyPEM, cb)
619 self.assertTrue(isinstance(key, PKeyType))
620 self.assertEqual(called, [False])
621
622
623 def test_dump_privatekey_passphrase(self):
624 """
625 L{dump_privatekey} writes an encrypted PEM when given a passphrase.
626 """
627 passphrase = "foo"
628 key = load_privatekey(FILETYPE_PEM, cleartextPrivateKeyPEM)
629 pem = dump_privatekey(FILETYPE_PEM, key, "blowfish", passphrase)
630 self.assertTrue(isinstance(pem, str))
631 loadedKey = load_privatekey(FILETYPE_PEM, pem, passphrase)
632 self.assertTrue(isinstance(loadedKey, PKeyType))
633 self.assertEqual(loadedKey.type(), key.type())
634 self.assertEqual(loadedKey.bits(), key.bits())
635
636
637 def test_dump_privatekey_passphraseCallback(self):
638 """
639 L{dump_privatekey} writes an encrypted PEM when given a callback which
640 returns the correct passphrase.
641 """
642 passphrase = "foo"
643 called = []
644 def cb(writing):
645 called.append(writing)
646 return passphrase
647 key = load_privatekey(FILETYPE_PEM, cleartextPrivateKeyPEM)
648 pem = dump_privatekey(FILETYPE_PEM, key, "blowfish", cb)
649 self.assertTrue(isinstance(pem, str))
650 self.assertEqual(called, [True])
651 loadedKey = load_privatekey(FILETYPE_PEM, pem, passphrase)
652 self.assertTrue(isinstance(loadedKey, PKeyType))
653 self.assertEqual(loadedKey.type(), key.type())
654 self.assertEqual(loadedKey.bits(), key.bits())