Add SSLConnectionFailedException

This way, when an SSLSocket is passed where an OFTCPSocket is expected,
the description will still contain the error and no special code for
SSLSocket is necessary.
This commit is contained in:
Jonathan Schleifer 2016-07-09 22:55:40 +02:00
parent 7c85e21d8f
commit 272a6f4c7f
No known key found for this signature in database
GPG key ID: 338C3541DB54E169
6 changed files with 241 additions and 14 deletions

View file

@ -5,7 +5,7 @@
<key>CFBundleExecutable</key> <key>CFBundleExecutable</key>
<string>${EXECUTABLE_NAME}</string> <string>${EXECUTABLE_NAME}</string>
<key>CFBundleIdentifier</key> <key>CFBundleIdentifier</key>
<string>zone.heap.${PRODUCT_NAME:rfc1034identifier}</string> <string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
<key>CFBundleInfoDictionaryVersion</key> <key>CFBundleInfoDictionaryVersion</key>
<string>6.0</string> <string>6.0</string>
<key>CFBundleName</key> <key>CFBundleName</key>

View file

@ -15,6 +15,8 @@
4B9671B6193E55C800F9F80D /* ObjFW.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4B9671B5193E55C800F9F80D /* ObjFW.framework */; }; 4B9671B6193E55C800F9F80D /* ObjFW.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4B9671B5193E55C800F9F80D /* ObjFW.framework */; };
4BD0AAEC1341289500445289 /* SSLSocket.h in Headers */ = {isa = PBXBuildFile; fileRef = 4BD0AAEA1341289500445289 /* SSLSocket.h */; settings = {ATTRIBUTES = (Public, ); }; }; 4BD0AAEC1341289500445289 /* SSLSocket.h in Headers */ = {isa = PBXBuildFile; fileRef = 4BD0AAEA1341289500445289 /* SSLSocket.h */; settings = {ATTRIBUTES = (Public, ); }; };
4BD0AAED1341289500445289 /* SSLSocket.m in Sources */ = {isa = PBXBuildFile; fileRef = 4BD0AAEB1341289500445289 /* SSLSocket.m */; }; 4BD0AAED1341289500445289 /* SSLSocket.m in Sources */ = {isa = PBXBuildFile; fileRef = 4BD0AAEB1341289500445289 /* SSLSocket.m */; };
4BDE04741D319BFC0051EDB8 /* SSLConnectionFailedException.h in Headers */ = {isa = PBXBuildFile; fileRef = 4BDE04721D319BFC0051EDB8 /* SSLConnectionFailedException.h */; settings = {ATTRIBUTES = (Public, ); }; };
4BDE04751D319BFC0051EDB8 /* SSLConnectionFailedException.m in Sources */ = {isa = PBXBuildFile; fileRef = 4BDE04731D319BFC0051EDB8 /* SSLConnectionFailedException.m */; };
/* End PBXBuildFile section */ /* End PBXBuildFile section */
/* Begin PBXFileReference section */ /* Begin PBXFileReference section */
@ -28,6 +30,8 @@
4BD0AAE91341286B00445289 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = SOURCE_ROOT; }; 4BD0AAE91341286B00445289 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = SOURCE_ROOT; };
4BD0AAEA1341289500445289 /* SSLSocket.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = SSLSocket.h; path = src/SSLSocket.h; sourceTree = SOURCE_ROOT; }; 4BD0AAEA1341289500445289 /* SSLSocket.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = SSLSocket.h; path = src/SSLSocket.h; sourceTree = SOURCE_ROOT; };
4BD0AAEB1341289500445289 /* SSLSocket.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = SSLSocket.m; path = src/SSLSocket.m; sourceTree = SOURCE_ROOT; }; 4BD0AAEB1341289500445289 /* SSLSocket.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = SSLSocket.m; path = src/SSLSocket.m; sourceTree = SOURCE_ROOT; };
4BDE04721D319BFC0051EDB8 /* SSLConnectionFailedException.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = SSLConnectionFailedException.h; path = src/SSLConnectionFailedException.h; sourceTree = SOURCE_ROOT; };
4BDE04731D319BFC0051EDB8 /* SSLConnectionFailedException.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = SSLConnectionFailedException.m; path = src/SSLConnectionFailedException.m; sourceTree = SOURCE_ROOT; };
/* End PBXFileReference section */ /* End PBXFileReference section */
/* Begin PBXFrameworksBuildPhase section */ /* Begin PBXFrameworksBuildPhase section */
@ -72,6 +76,8 @@
children = ( children = (
4B1918F41341272300D82152 /* Supporting Files */, 4B1918F41341272300D82152 /* Supporting Files */,
4B4F087713A01EEF00B60C3F /* ObjOpenSSL.h */, 4B4F087713A01EEF00B60C3F /* ObjOpenSSL.h */,
4BDE04721D319BFC0051EDB8 /* SSLConnectionFailedException.h */,
4BDE04731D319BFC0051EDB8 /* SSLConnectionFailedException.m */,
4B19F58714D17250005D52DC /* SSLInvalidCertificateException.h */, 4B19F58714D17250005D52DC /* SSLInvalidCertificateException.h */,
4B19F58814D17250005D52DC /* SSLInvalidCertificateException.m */, 4B19F58814D17250005D52DC /* SSLInvalidCertificateException.m */,
4BD0AAEA1341289500445289 /* SSLSocket.h */, 4BD0AAEA1341289500445289 /* SSLSocket.h */,
@ -98,6 +104,7 @@
buildActionMask = 2147483647; buildActionMask = 2147483647;
files = ( files = (
4B4F087813A01EEF00B60C3F /* ObjOpenSSL.h in Headers */, 4B4F087813A01EEF00B60C3F /* ObjOpenSSL.h in Headers */,
4BDE04741D319BFC0051EDB8 /* SSLConnectionFailedException.h in Headers */,
4B19F58B14D17250005D52DC /* SSLInvalidCertificateException.h in Headers */, 4B19F58B14D17250005D52DC /* SSLInvalidCertificateException.h in Headers */,
4BD0AAEC1341289500445289 /* SSLSocket.h in Headers */, 4BD0AAEC1341289500445289 /* SSLSocket.h in Headers */,
4B19F58D14D17250005D52DC /* X509Certificate.h in Headers */, 4B19F58D14D17250005D52DC /* X509Certificate.h in Headers */,
@ -131,7 +138,7 @@
4B1918E01341272300D82152 /* Project object */ = { 4B1918E01341272300D82152 /* Project object */ = {
isa = PBXProject; isa = PBXProject;
attributes = { attributes = {
LastUpgradeCheck = 0510; LastUpgradeCheck = 0730;
}; };
buildConfigurationList = 4B1918E31341272300D82152 /* Build configuration list for PBXProject "ObjOpenSSL" */; buildConfigurationList = 4B1918E31341272300D82152 /* Build configuration list for PBXProject "ObjOpenSSL" */;
compatibilityVersion = "Xcode 3.2"; compatibilityVersion = "Xcode 3.2";
@ -165,6 +172,7 @@
isa = PBXSourcesBuildPhase; isa = PBXSourcesBuildPhase;
buildActionMask = 2147483647; buildActionMask = 2147483647;
files = ( files = (
4BDE04751D319BFC0051EDB8 /* SSLConnectionFailedException.m in Sources */,
4B19F58C14D17250005D52DC /* SSLInvalidCertificateException.m in Sources */, 4B19F58C14D17250005D52DC /* SSLInvalidCertificateException.m in Sources */,
4BD0AAED1341289500445289 /* SSLSocket.m in Sources */, 4BD0AAED1341289500445289 /* SSLSocket.m in Sources */,
4B19F58E14D17250005D52DC /* X509Certificate.m in Sources */, 4B19F58E14D17250005D52DC /* X509Certificate.m in Sources */,
@ -177,6 +185,7 @@
4B1918FA1341272300D82152 /* Debug */ = { 4B1918FA1341272300D82152 /* Debug */ = {
isa = XCBuildConfiguration; isa = XCBuildConfiguration;
buildSettings = { buildSettings = {
ENABLE_TESTABILITY = YES;
GCC_C_LANGUAGE_STANDARD = gnu99; GCC_C_LANGUAGE_STANDARD = gnu99;
GCC_OPTIMIZATION_LEVEL = 0; GCC_OPTIMIZATION_LEVEL = 0;
GCC_PREPROCESSOR_DEFINITIONS = DEBUG; GCC_PREPROCESSOR_DEFINITIONS = DEBUG;
@ -224,6 +233,7 @@
"-lcrypto", "-lcrypto",
"-lz", "-lz",
); );
PRODUCT_BUNDLE_IDENTIFIER = "zone.heap.${PRODUCT_NAME:rfc1034identifier}";
PRODUCT_NAME = "$(TARGET_NAME)"; PRODUCT_NAME = "$(TARGET_NAME)";
WARNING_CFLAGS = ( WARNING_CFLAGS = (
"-Wall", "-Wall",
@ -263,6 +273,7 @@
"-lcrypto", "-lcrypto",
"-lz", "-lz",
); );
PRODUCT_BUNDLE_IDENTIFIER = "zone.heap.${PRODUCT_NAME:rfc1034identifier}";
PRODUCT_NAME = "$(TARGET_NAME)"; PRODUCT_NAME = "$(TARGET_NAME)";
WARNING_CFLAGS = ( WARNING_CFLAGS = (
"-Wall", "-Wall",

View file

@ -5,7 +5,8 @@ STATIC_LIB = ${OBJOPENSSL_STATIC_LIB}
LIB_MAJOR = 0 LIB_MAJOR = 0
LIB_MINOR = 0 LIB_MINOR = 0
SRCS = SSLInvalidCertificateException.m \ SRCS = SSLConnectionFailedException.m \
SSLInvalidCertificateException.m \
SSLSocket.m \ SSLSocket.m \
X509Certificate.m X509Certificate.m

View file

@ -0,0 +1,54 @@
/*
* Copyright (c) 2016, Jonathan Schleifer <js@heap.zone>
*
* https://heap.zone/git/?p=objopenssl.git
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice is present in all copies.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#import <ObjFW/OFConnectionFailedException.h>
@class SSLSocket;
@interface SSLConnectionFailedException: OFConnectionFailedException
{
unsigned long _SSLError;
long _verifyResult;
}
@property (readonly) unsigned long SSLError;
@property (readonly) long verifyResult;
+ (instancetype)exceptionWithHost: (OFString*)host
port: (uint16_t)port
socket: (SSLSocket*)socket
SSLError: (unsigned long)SSLError;
+ (instancetype)exceptionWithHost: (OFString*)host
port: (uint16_t)port
socket: (SSLSocket*)socket
SSLError: (unsigned long)SSLError
verifyResult: (long)verifyResult;
- initWithHost: (OFString*)host
port: (uint16_t)port
socket: (SSLSocket*)socket
SSLError: (unsigned long)SSLError;
- initWithHost: (OFString*)host
port: (uint16_t)port
socket: (SSLSocket*)socket
SSLError: (unsigned long)SSLError
verifyResult: (long)verifyResult;
@end

View file

@ -0,0 +1,124 @@
/*
* Copyright (c) 2016, Jonathan Schleifer <js@heap.zone>
*
* https://heap.zone/git/?p=objopenssl.git
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice is present in all copies.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#include <inttypes.h>
#import <ObjFW/OFString.h>
#import "SSLConnectionFailedException.h"
#if defined(__clang__)
# pragma clang diagnostic push
# pragma clang diagnostic ignored "-Wdocumentation"
#endif
#include <openssl/err.h>
#include <openssl/ssl.h>
#include <openssl/x509v3.h>
#if defined(__clang__)
# pragma clang diagnostic pop
#endif
@implementation SSLConnectionFailedException
@synthesize SSLError = _SSLError, verifyResult = _verifyResult;
+ (instancetype)exceptionWithHost: (OFString*)host
port: (uint16_t)port
socket: (SSLSocket*)socket
SSLError: (unsigned long)SSLError
{
return [[[self alloc] initWithHost: host
port: port
socket: socket
SSLError: SSLError] autorelease];
}
+ (instancetype)exceptionWithHost: (OFString*)host
port: (uint16_t)port
socket: (SSLSocket*)socket
SSLError: (unsigned long)SSLError
verifyResult: (long)verifyResult
{
return [[[self alloc] initWithHost: host
port: port
socket: socket
SSLError: SSLError
verifyResult: verifyResult] autorelease];
}
- initWithHost: (OFString*)host
port: (uint16_t)port
socket: (SSLSocket*)socket
SSLError: (unsigned long)SSLError
{
self = [super initWithHost: host
port: port
socket: socket];
_SSLError = SSLError;
return self;
}
- initWithHost: (OFString*)host
port: (uint16_t)port
socket: (SSLSocket*)socket
SSLError: (unsigned long)SSLError
verifyResult: (long)verifyResult
{
self = [super initWithHost: host
port: port
socket: socket];
_SSLError = SSLError;
_verifyResult = verifyResult;
return self;
}
- (OFString*)description
{
if (_SSLError != SSL_ERROR_NONE) {
char error[512];
ERR_error_string_n(_SSLError, error, 512);
if (_verifyResult != X509_V_OK)
return [OFString stringWithFormat:
@"A connection to %@ on port %" @PRIu16 @" could "
@"not be established in socket of type %@: "
@"Verification failed: %s [%s]",
_host, _port, [_socket class],
X509_verify_cert_error_string(_verifyResult),
error];
else
return [OFString stringWithFormat:
@"A connection to %@ on port %" @PRIu16 @" could "
@"not be established in socket of type %@: %s",
_host, _port, [_socket class], error];
}
return [super description];
}
@end

View file

@ -34,6 +34,7 @@
#include <openssl/crypto.h> #include <openssl/crypto.h>
#include <openssl/err.h> #include <openssl/err.h>
#include <openssl/ssl.h>
#include <openssl/x509v3.h> #include <openssl/x509v3.h>
#if defined(__clang__) #if defined(__clang__)
@ -46,7 +47,6 @@
#import <ObjFW/OFSystemInfo.h> #import <ObjFW/OFSystemInfo.h>
#import <ObjFW/OFAcceptFailedException.h> #import <ObjFW/OFAcceptFailedException.h>
#import <ObjFW/OFConnectionFailedException.h>
#import <ObjFW/OFInitializationFailedException.h> #import <ObjFW/OFInitializationFailedException.h>
#import <ObjFW/OFInvalidArgumentException.h> #import <ObjFW/OFInvalidArgumentException.h>
#import <ObjFW/OFNotOpenException.h> #import <ObjFW/OFNotOpenException.h>
@ -58,9 +58,11 @@
#import <ObjFW/threading.h> #import <ObjFW/threading.h>
#import "SSLSocket.h" #import "SSLSocket.h"
#import "SSLInvalidCertificateException.h"
#import "X509Certificate.h" #import "X509Certificate.h"
#import "SSLConnectionFailedException.h"
#import "SSLInvalidCertificateException.h"
#ifndef INVALID_SOCKET #ifndef INVALID_SOCKET
# define INVALID_SOCKET -1 # define INVALID_SOCKET -1
#endif #endif
@ -175,12 +177,16 @@ locking_callback(int mode, int n, const char *file, int line)
{ {
of_string_encoding_t encoding; of_string_encoding_t encoding;
if ((_SSL = SSL_new(ctx)) == NULL || !SSL_set_fd(_SSL, _socket)) { if ((_SSL = SSL_new(ctx)) == NULL || SSL_set_fd(_SSL, _socket) != 1) {
unsigned long error = ERR_get_error();
[super close]; [super close];
@throw [OFConnectionFailedException
@throw [SSLConnectionFailedException
exceptionWithHost: host exceptionWithHost: host
port: port port: port
socket: self]; socket: self
SSLError: error];
} }
if (_certificateVerificationEnabled) { if (_certificateVerificationEnabled) {
@ -190,11 +196,17 @@ locking_callback(int mode, int n, const char *file, int line)
X509_CHECK_FLAG_NO_PARTIAL_WILDCARDS); X509_CHECK_FLAG_NO_PARTIAL_WILDCARDS);
if (X509_VERIFY_PARAM_set1_host(param, if (X509_VERIFY_PARAM_set1_host(param,
[host UTF8String], [host UTF8StringLength]) == 0) [host UTF8String], [host UTF8StringLength]) != 1) {
@throw [OFConnectionFailedException unsigned long error = ERR_get_error();
[self close];
@throw [SSLConnectionFailedException
exceptionWithHost: host exceptionWithHost: host
port: port port: port
socket: self]; socket: self
SSLError: error];
}
SSL_set_verify(_SSL, SSL_VERIFY_PEER, NULL); SSL_set_verify(_SSL, SSL_VERIFY_PEER, NULL);
} }
@ -208,12 +220,37 @@ locking_callback(int mode, int n, const char *file, int line)
SSL_FILETYPE_PEM)) || (_certificateFile != nil && SSL_FILETYPE_PEM)) || (_certificateFile != nil &&
!SSL_use_certificate_file(_SSL, [_certificateFile !SSL_use_certificate_file(_SSL, [_certificateFile
cStringWithEncoding: encoding], cStringWithEncoding: encoding],
SSL_FILETYPE_PEM)) || SSL_connect(_SSL) != 1) { SSL_FILETYPE_PEM))) {
unsigned long error = ERR_get_error();
[super close]; [super close];
@throw [OFConnectionFailedException
@throw [SSLConnectionFailedException
exceptionWithHost: host exceptionWithHost: host
port: port port: port
socket: self]; socket: self
SSLError: error];
}
if (SSL_connect(_SSL) != 1) {
unsigned long error = ERR_get_error();
long res;
[super close];
if ((res = SSL_get_verify_result(_SSL)) != X509_V_OK)
@throw [SSLConnectionFailedException
exceptionWithHost: host
port: port
socket: self
SSLError: error
verifyResult: res];
else
@throw [SSLConnectionFailedException
exceptionWithHost: host
port: port
socket: self
SSLError: error];
} }
} }