blob: 179217842c64a7c28965ab8512d646143b301376 [file] [log] [blame]
Jean-Paul Calderone897bc252008-02-18 20:50:23 -05001/*
2 * netscape_spki.c
3 *
Jean-Paul Calderone8671c852011-03-02 19:26:20 -05004 * Copyright (C) Tollef Fog Heen
5 * See LICENSE for details.
Jean-Paul Calderone897bc252008-02-18 20:50:23 -05006 *
7 * Netscape SPKI handling, thin wrapper
8 */
9#include <Python.h>
10#define crypto_MODULE
11#include "crypto.h"
12
Jean-Paul Calderone897bc252008-02-18 20:50:23 -050013/*
14 * Constructor for Nestcape_SPKI, never called by Python code directly
15 *
16 * Arguments: name - A "real" NetscapeSPKI object
17 * dealloc - Boolean value to specify whether the destructor should
18 * free the "real" NetscapeSPKI object
19 * Returns: The newly created NetscapeSPKI object
20 */
21crypto_NetscapeSPKIObj *
22crypto_NetscapeSPKI_New(NETSCAPE_SPKI *name, int dealloc)
23{
24 crypto_NetscapeSPKIObj *self;
25
26 self = PyObject_New(crypto_NetscapeSPKIObj, &crypto_NetscapeSPKI_Type);
27
28 if (self == NULL)
29 return NULL;
30
31 self->netscape_spki = name;
32 self->dealloc = dealloc;
33
34 return self;
35}
36
Jean-Paul Calderonedc138fa2009-06-27 14:32:07 -040037
38static char crypto_NetscapeSPKI_doc[] = "\n\
39NetscapeSPKI([enc]) -> NetscapeSPKI instance\n\
40\n\
41@param enc: Base64 encoded NetscapeSPKI object.\n\
42@type enc: C{str}\n\
43@return: The NetscapeSPKI object\n\
44";
45
46static PyObject *
47crypto_NetscapeSPKI_new(PyTypeObject *subtype, PyObject *args, PyObject *kwargs) {
48 char *enc = NULL;
49 int enc_len = -1;
50 NETSCAPE_SPKI *spki;
51
52 if (!PyArg_ParseTuple(args, "|s#:NetscapeSPKI", &enc, &enc_len))
53 return NULL;
54
55 if (enc_len >= 0)
56 spki = NETSCAPE_SPKI_b64_decode(enc, enc_len);
57 else
58 spki = NETSCAPE_SPKI_new();
59 if (spki == NULL)
60 {
Rick Deand369c932009-07-08 11:48:33 -050061 exception_from_error_queue(crypto_Error);
Jean-Paul Calderonedc138fa2009-06-27 14:32:07 -040062 return NULL;
63 }
64 return (PyObject *)crypto_NetscapeSPKI_New(spki, 1);
65}
66
67
Jean-Paul Calderone897bc252008-02-18 20:50:23 -050068/*
69 * Deallocate the memory used by the NetscapeSPKI object
70 *
71 * Arguments: self - The NetscapeSPKI object
72 * Returns: None
73 */
74static void
75crypto_NetscapeSPKI_dealloc(crypto_NetscapeSPKIObj *self)
76{
77 /* Sometimes we don't have to dealloc this */
78 if (self->dealloc)
79 NETSCAPE_SPKI_free(self->netscape_spki);
80
81 PyObject_Del(self);
82}
83
84static char crypto_NetscapeSPKI_sign_doc[] = "\n\
85Sign the certificate request using the supplied key and digest\n\
86\n\
Jean-Paul Calderone54bcc832009-05-27 14:06:48 -040087@param pkey: The key to sign with\n\
88@param digest: The message digest to use\n\
89@return: None\n\
Jean-Paul Calderone897bc252008-02-18 20:50:23 -050090";
91
92static PyObject *
93crypto_NetscapeSPKI_sign(crypto_NetscapeSPKIObj *self, PyObject *args)
94{
95 crypto_PKeyObj *pkey;
96 char *digest_name;
97 const EVP_MD *digest;
98
99 if (!PyArg_ParseTuple(args, "O!s:sign", &crypto_PKey_Type, &pkey,
100 &digest_name))
101 return NULL;
102
Jean-Paul Calderoneb9725592010-08-03 18:17:22 -0400103 if (pkey->only_public) {
104 PyErr_SetString(PyExc_ValueError, "Key has only public part");
105 return NULL;
106 }
107
108 if (!pkey->initialized) {
109 PyErr_SetString(PyExc_ValueError, "Key is uninitialized");
110 return NULL;
111 }
112
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500113 if ((digest = EVP_get_digestbyname(digest_name)) == NULL)
114 {
115 PyErr_SetString(PyExc_ValueError, "No such digest method");
116 return NULL;
117 }
118
119 if (!NETSCAPE_SPKI_sign(self->netscape_spki, pkey->pkey, digest))
120 {
Rick Deand369c932009-07-08 11:48:33 -0500121 exception_from_error_queue(crypto_Error);
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500122 return NULL;
123 }
124
125 Py_INCREF(Py_None);
126 return Py_None;
127}
128
129static char crypto_NetscapeSPKI_verify_doc[] = "\n\
130Verifies a certificate request using the supplied public key\n\
Jean-Paul Calderone54bcc832009-05-27 14:06:48 -0400131\n\
132@param key: a public key\n\
Jean-Paul Calderoneb9725592010-08-03 18:17:22 -0400133@return: True if the signature is correct.\n\
134@raise OpenSSL.crypto.Error: If the signature is invalid or there is a\n\
135 problem verifying the signature.\n\
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500136";
137
138PyObject *
139crypto_NetscapeSPKI_verify(crypto_NetscapeSPKIObj *self, PyObject *args)
140{
141 crypto_PKeyObj *pkey;
142 int answer;
143
Jean-Paul Calderoneb9725592010-08-03 18:17:22 -0400144 if (!PyArg_ParseTuple(args, "O!:verify", &crypto_PKey_Type, &pkey)) {
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500145 return NULL;
Jean-Paul Calderoneb9725592010-08-03 18:17:22 -0400146 }
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500147
Jean-Paul Calderoneb9725592010-08-03 18:17:22 -0400148 if ((answer = NETSCAPE_SPKI_verify(self->netscape_spki, pkey->pkey)) <= 0) {
Rick Deand369c932009-07-08 11:48:33 -0500149 exception_from_error_queue(crypto_Error);
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500150 return NULL;
151 }
152
Jean-Paul Calderone2f6c66f2010-08-11 19:53:43 -0400153 return PyLong_FromLong((long)answer);
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500154}
155
156static char crypto_NetscapeSPKI_b64_encode_doc[] = "\n\
157Generate a base64 encoded string from an SPKI\n\
Jean-Paul Calderone54bcc832009-05-27 14:06:48 -0400158\n\
159@return: The base64 encoded string\n\
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500160";
161
162PyObject *
163crypto_NetscapeSPKI_b64_encode(crypto_NetscapeSPKIObj *self, PyObject *args)
164{
165 char *str;
166
167 if (!PyArg_ParseTuple(args, ":b64_encode"))
168 return NULL;
169
170 str = NETSCAPE_SPKI_b64_encode(self->netscape_spki);
Jean-Paul Calderone2f6c66f2010-08-11 19:53:43 -0400171 return PyBytes_FromString(str);
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500172}
173
174
175static char crypto_NetscapeSPKI_get_pubkey_doc[] = "\n\
176Get the public key of the certificate\n\
177\n\
Jean-Paul Calderone54bcc832009-05-27 14:06:48 -0400178@return: The public key\n\
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500179";
180
181static PyObject *
182crypto_NetscapeSPKI_get_pubkey(crypto_NetscapeSPKIObj *self, PyObject *args)
183{
184 crypto_PKeyObj *crypto_PKey_New(EVP_PKEY *, int);
185 EVP_PKEY *pkey;
Jean-Paul Calderoneb9725592010-08-03 18:17:22 -0400186 crypto_PKeyObj *py_pkey;
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500187
188 if (!PyArg_ParseTuple(args, ":get_pubkey"))
189 return NULL;
190
191 if ((pkey = NETSCAPE_SPKI_get_pubkey(self->netscape_spki)) == NULL)
192 {
Rick Deand369c932009-07-08 11:48:33 -0500193 exception_from_error_queue(crypto_Error);
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500194 return NULL;
195 }
196
Jean-Paul Calderoneb9725592010-08-03 18:17:22 -0400197 py_pkey = crypto_PKey_New(pkey, 1);
198 if (py_pkey != NULL) {
199 py_pkey->only_public = 1;
200 }
201 return (PyObject *)py_pkey;
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500202}
203
204static char crypto_NetscapeSPKI_set_pubkey_doc[] = "\n\
205Set the public key of the certificate\n\
206\n\
Jean-Paul Calderone54bcc832009-05-27 14:06:48 -0400207@param pkey: The public key\n\
208@return: None\n\
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500209";
210
211static PyObject *
212crypto_NetscapeSPKI_set_pubkey(crypto_NetscapeSPKIObj *self, PyObject *args)
213{
214 crypto_PKeyObj *pkey;
215
216 if (!PyArg_ParseTuple(args, "O!:set_pubkey", &crypto_PKey_Type, &pkey))
217 return NULL;
218
219 if (!NETSCAPE_SPKI_set_pubkey(self->netscape_spki, pkey->pkey))
220 {
Rick Deand369c932009-07-08 11:48:33 -0500221 exception_from_error_queue(crypto_Error);
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500222 return NULL;
223 }
224
225 Py_INCREF(Py_None);
226 return Py_None;
227}
228
229/*
230 * ADD_METHOD(name) expands to a correct PyMethodDef declaration
231 * { 'name', (PyCFunction)crypto_NetscapeSPKI_name, METH_VARARGS }
232 * for convenience
233 */
234#define ADD_METHOD(name) \
235 { #name, (PyCFunction)crypto_NetscapeSPKI_##name, METH_VARARGS, crypto_NetscapeSPKI_##name##_doc }
236static PyMethodDef crypto_NetscapeSPKI_methods[] =
237{
238 ADD_METHOD(get_pubkey),
239 ADD_METHOD(set_pubkey),
240 ADD_METHOD(b64_encode),
241 ADD_METHOD(sign),
242 ADD_METHOD(verify),
243 { NULL, NULL }
244};
245#undef ADD_METHOD
246
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500247PyTypeObject crypto_NetscapeSPKI_Type = {
Jean-Paul Calderone3fe7f672010-08-11 23:55:10 -0400248 PyOpenSSL_HEAD_INIT(&PyType_Type, 0)
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500249 "NetscapeSPKI",
250 sizeof(crypto_NetscapeSPKIObj),
251 0,
252 (destructor)crypto_NetscapeSPKI_dealloc,
253 NULL, /* print */
Jean-Paul Calderone2f6c66f2010-08-11 19:53:43 -0400254 NULL, /* getattr */
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500255 NULL, /* setattr */
256 NULL, /* compare */
257 NULL, /* repr */
258 NULL, /* as_number */
259 NULL, /* as_sequence */
260 NULL, /* as_mapping */
Jean-Paul Calderonedc138fa2009-06-27 14:32:07 -0400261 NULL, /* hash */
262 NULL, /* call */
263 NULL, /* str */
264 NULL, /* getattro */
265 NULL, /* setattro */
266 NULL, /* as_buffer */
267 Py_TPFLAGS_DEFAULT,
268 crypto_NetscapeSPKI_doc, /* doc */
269 NULL, /* traverse */
270 NULL, /* clear */
271 NULL, /* tp_richcompare */
272 0, /* tp_weaklistoffset */
273 NULL, /* tp_iter */
274 NULL, /* tp_iternext */
275 crypto_NetscapeSPKI_methods, /* tp_methods */
276 NULL, /* tp_members */
277 NULL, /* tp_getset */
278 NULL, /* tp_base */
279 NULL, /* tp_dict */
280 NULL, /* tp_descr_get */
281 NULL, /* tp_descr_set */
282 0, /* tp_dictoffset */
283 NULL, /* tp_init */
284 NULL, /* tp_alloc */
285 crypto_NetscapeSPKI_new, /* tp_new */
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500286};
287
288
289/*
290 * Initialize the X509Name part of the crypto module
291 *
Jean-Paul Calderonedc138fa2009-06-27 14:32:07 -0400292 * Arguments: module - The crypto module
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500293 * Returns: None
294 */
295int
Jean-Paul Calderonedc138fa2009-06-27 14:32:07 -0400296init_crypto_netscape_spki(PyObject *module) {
297 if (PyType_Ready(&crypto_NetscapeSPKI_Type) < 0) {
298 return 0;
299 }
300
301 if (PyModule_AddObject(module, "NetscapeSPKI", (PyObject *)&crypto_NetscapeSPKI_Type) != 0) {
302 return 0;
303 }
304
305 if (PyModule_AddObject(module, "NetscapeSPKIType", (PyObject *)&crypto_NetscapeSPKI_Type) != 0) {
306 return 0;
307 }
308
Jean-Paul Calderone897bc252008-02-18 20:50:23 -0500309 return 1;
310}