/* vi: set sw=4 ts=4: * * Copyright (C) 2001 - 2007 Christian Hohnstaedt. * * All rights reserved. */ #include "pki_pkcs12.h" #include "pass_info.h" #include "exception.h" #include "func.h" #include #include pki_pkcs12::pki_pkcs12(const QString d, pki_x509 *acert, pki_key *akey, pem_password_cb *cb): pki_base(d) { class_name="pki_pkcs12"; key = new pki_key(akey); cert = new pki_x509(acert); certstack = sk_X509_new_null(); passcb = cb; openssl_error(); } pki_pkcs12::pki_pkcs12(const QString fname, pem_password_cb *cb) :pki_base(fname) { FILE *fp; char pass[30]; EVP_PKEY *mykey = NULL; X509 *mycert = NULL; key=NULL; cert=NULL; passcb = cb; class_name="pki_pkcs12"; certstack = sk_X509_new_null(); pass_info p(XCA_TITLE, tr("Please enter the password to decrypt the PKCS#12 file.") + "\n'" + fname + "'"); fp = fopen(CCHAR(fname), "rb"); if (fp) { PKCS12 *pkcs12 = d2i_PKCS12_fp(fp, NULL); fclose(fp); openssl_error(); if (PKCS12_verify_mac(pkcs12, "", 0) || PKCS12_verify_mac(pkcs12, NULL, 0)) pass[0] = '\0'; else if (passcb(pass, 30, 0, &p) < 0) { /* cancel pressed */ PKCS12_free(pkcs12); throw errorEx("",""); } PKCS12_parse(pkcs12, pass, &mykey, &mycert, &certstack); if ( ERR_peek_error() != 0) { ign_openssl_error(); PKCS12_free(pkcs12); throw errorEx(getClassName(),"The supplied password was wrong"); } openssl_error(); if (mycert) { if (mycert->aux && mycert->aux->alias){ alias = asn1ToQString(mycert->aux->alias); } cert = new pki_x509(mycert); if (alias.isEmpty()) { cert->autoIntName(); } else { cert->setIntName(alias); } alias = cert->getIntName(); } if (mykey) { key = new pki_key(mykey); key->setIntName(alias + "_key"); key->bogusEncryptKey(); } PKCS12_free(pkcs12); } else fopen_error(fname); } pki_pkcs12::~pki_pkcs12() { if (sk_X509_num(certstack)>0) { // free the certs itself, because we own a copy of them sk_X509_pop_free(certstack, X509_free); } if (key) { delete(key); } if (cert) { delete(cert); } openssl_error(); } void pki_pkcs12::addCaCert(pki_x509 *ca) { if (!ca) return; sk_X509_push(certstack, X509_dup(ca->getCert())); openssl_error(); } void pki_pkcs12::writePKCS12(const QString fname) { char pass[30]; char desc[100]; strncpy(desc, getIntName().toLocal8Bit(), 100); pass_info p(XCA_TITLE, tr("Please enter the password to encrypt the PKCS#12 file")); if (cert == NULL || key == NULL) { my_error(tr("No key or no Cert and no pkcs12")); } FILE *fp = fopen(CCHAR(fname),"wb"); if (fp != NULL) { passcb(pass, 30, 0, &p); PKCS12 *pkcs12 = PKCS12_create(pass, desc, key->decryptKey(), cert->getCert(), certstack, 0, 0, 0, 0, 0); i2d_PKCS12_fp(fp, pkcs12); openssl_error(); fclose (fp); PKCS12_free(pkcs12); } else fopen_error(fname); } int pki_pkcs12::numCa() { int n= sk_X509_num(certstack); openssl_error(); return n; } pki_key *pki_pkcs12::getKey() { if (!key) return NULL; return new pki_key(key); } pki_x509 *pki_pkcs12::getCert() { if (!cert) return NULL; return new pki_x509(cert); } pki_x509 *pki_pkcs12::getCa(int x) { pki_x509 *cert = NULL; X509 *crt = X509_dup(sk_X509_value(certstack, x)); if (crt) { cert = new pki_x509(crt); if (alias.isEmpty()) { cert->autoIntName(); } else { cert->setIntName(QString(alias + "_ca_%1").arg(x)); } } openssl_error(); return cert; }