/*
* Copyright (c) 2008-2024 Jonathan Schleifer <js@nil.im>
*
* All rights reserved.
*
* This program is free software: you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License version 3.0 only,
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
* version 3.0 for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* version 3.0 along with this program. If not, see
* <https://www.gnu.org/licenses/>.
*/
#include "config.h"
#import "OFGnuTLSX509Certificate.h"
#import "OFArray.h"
#import "OFData.h"
#import "OFInvalidFormatException.h"
#import "OFOutOfRangeException.h"
@implementation OFGnuTLSX509Certificate
@synthesize of_gnuTLSCertificate = _certificate;
+ (void)load
{
if (OFX509CertificateImplementation == Nil)
OFX509CertificateImplementation = self;
}
+ (OFArray OF_GENERIC(OFX509Certificate *) *)
certificateChainFromIRI: (OFIRI *)IRI
{
OFMutableArray *chain = [OFMutableArray array];
void *pool = objc_autoreleasePoolPush();
OFData *data = [OFData dataWithContentsOfIRI: IRI];
gnutls_datum_t datum;
gnutls_x509_crt_t *certs;
unsigned int i, size;
if (data.count * data.itemSize > UINT_MAX)
@throw [OFOutOfRangeException exception];
datum.data = (unsigned char *)data.items;
datum.size = (unsigned int)(data.count * data.itemSize);
if (gnutls_x509_crt_list_import2(&certs, &size, &datum,
GNUTLS_X509_FMT_PEM, 0) != GNUTLS_E_SUCCESS)
@throw [OFInvalidFormatException exception];
for (i = 0; i < size; i++) {
OFGnuTLSX509Certificate *certificate;
@try {
certificate = [[self alloc]
of_initWithGnuTLSCertificate: certs[i]];
} @catch (id e) {
gnutls_x509_crt_deinit(certs[i]);
gnutls_free(certs);
@throw e;
}
@try {
[chain addObject: certificate];
} @catch (id e) {
gnutls_free(certs);
@throw e;
} @finally {
[certificate release];
}
}
gnutls_free(certs);
objc_autoreleasePoolPop(pool);
return chain;
}
- (instancetype)of_initWithGnuTLSCertificate: (gnutls_x509_crt_t)certificate
{
self = [super init];
_certificate = certificate;
return self;
}
- (void)dealloc
{
gnutls_x509_crt_deinit(_certificate);
[super dealloc];
}
@end