FossilOrigin-Name: 0af9225d647a95b3135fc711065584dcff39b523377528c734edc0ecb2b379f4
241 lines
6 KiB
Objective-C
241 lines
6 KiB
Objective-C
/*
|
|
* Copyright (c) 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019
|
|
* Jonathan Schleifer <js@nil.im>
|
|
*
|
|
* https://fossil.nil.im/objpgsql
|
|
*
|
|
* 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 "PGConnection.h"
|
|
#import "PGConnection+Private.h"
|
|
#import "PGResult.h"
|
|
#import "PGResult+Private.h"
|
|
|
|
#import "PGConnectionFailedException.h"
|
|
#import "PGCommandFailedException.h"
|
|
|
|
@implementation PGConnection
|
|
@synthesize pg_connection = _connection, parameters = _parameters;
|
|
|
|
- (void)dealloc
|
|
{
|
|
[_parameters release];
|
|
|
|
[self close];
|
|
|
|
[super dealloc];
|
|
}
|
|
|
|
- (void)connect
|
|
{
|
|
void *pool = objc_autoreleasePoolPush();
|
|
OFEnumerator OF_GENERIC(OFString *) *keyEnumerator =
|
|
[_parameters keyEnumerator];
|
|
OFEnumerator OF_GENERIC(OFString *) *objectEnumerator =
|
|
[_parameters objectEnumerator];
|
|
OFMutableString *connectionInfo = nil;
|
|
OFString *key, *object;
|
|
|
|
while ((key = [keyEnumerator nextObject]) != nil &&
|
|
(object = [objectEnumerator nextObject]) != nil) {
|
|
if (connectionInfo != nil)
|
|
[connectionInfo appendFormat: @" %@=%@", key, object];
|
|
else
|
|
connectionInfo = [OFMutableString stringWithFormat:
|
|
@"%@=%@", key, object];
|
|
}
|
|
|
|
if ((_connection = PQconnectdb(connectionInfo.UTF8String)) == NULL)
|
|
@throw [OFOutOfMemoryException exception];
|
|
|
|
if (PQstatus(_connection) == CONNECTION_BAD)
|
|
@throw [PGConnectionFailedException
|
|
exceptionWithConnection: self];
|
|
|
|
objc_autoreleasePoolPop(pool);
|
|
}
|
|
|
|
- (void)reset
|
|
{
|
|
PQreset(_connection);
|
|
}
|
|
|
|
- (void)close
|
|
{
|
|
if (_connection != NULL)
|
|
PQfinish(_connection);
|
|
|
|
_connection = NULL;
|
|
}
|
|
|
|
- (PGResult *)executeCommand: (OFConstantString *)command
|
|
{
|
|
PGresult *result = PQexec(_connection, command.UTF8String);
|
|
|
|
if (PQresultStatus(result) == PGRES_FATAL_ERROR) {
|
|
PQclear(result);
|
|
@throw [PGCommandFailedException
|
|
exceptionWithConnection: self
|
|
command: command];
|
|
}
|
|
|
|
switch (PQresultStatus(result)) {
|
|
case PGRES_TUPLES_OK:
|
|
return [PGResult pg_resultWithResult: result];
|
|
case PGRES_COMMAND_OK:
|
|
PQclear(result);
|
|
return nil;
|
|
default:
|
|
PQclear(result);
|
|
@throw [PGCommandFailedException
|
|
exceptionWithConnection: self
|
|
command: command];
|
|
}
|
|
}
|
|
|
|
- (PGResult *)executeCommand: (OFConstantString *)command
|
|
parameters: (id)parameter, ...
|
|
{
|
|
void *pool = objc_autoreleasePoolPush();
|
|
PGresult *result;
|
|
const char **values;
|
|
va_list args, args2;
|
|
int argsCount;
|
|
|
|
va_start(args, parameter);
|
|
va_copy(args2, args);
|
|
|
|
for (argsCount = 1; va_arg(args2, id) != nil; argsCount++);
|
|
|
|
values = OFAllocMemory(argsCount, sizeof(*values));
|
|
@try {
|
|
size_t i = 0;
|
|
|
|
do {
|
|
if ([parameter isKindOfClass: [OFString class]])
|
|
values[i++] = [parameter UTF8String];
|
|
else if ([parameter isKindOfClass: [OFNumber class]]) {
|
|
OFNumber *number = parameter;
|
|
|
|
if (strcmp(number.objCType,
|
|
@encode(bool)) == 0) {
|
|
if (number.boolValue)
|
|
values[i++] = "t";
|
|
else
|
|
values[i++] = "f";
|
|
} else
|
|
values[i++] =
|
|
number.description.UTF8String;
|
|
} else if ([parameter isKindOfClass: [OFNull class]])
|
|
values[i++] = NULL;
|
|
else
|
|
values[i++] =
|
|
[parameter description].UTF8String;
|
|
} while ((parameter = va_arg(args, id)) != nil);
|
|
|
|
result = PQexecParams(_connection, command.UTF8String,
|
|
argsCount, NULL, values, NULL, NULL, 0);
|
|
} @finally {
|
|
OFFreeMemory(values);
|
|
}
|
|
|
|
objc_autoreleasePoolPop(pool);
|
|
|
|
switch (PQresultStatus(result)) {
|
|
case PGRES_TUPLES_OK:
|
|
return [PGResult pg_resultWithResult: result];
|
|
case PGRES_COMMAND_OK:
|
|
PQclear(result);
|
|
return nil;
|
|
default:
|
|
PQclear(result);
|
|
@throw [PGCommandFailedException
|
|
exceptionWithConnection: self
|
|
command: command];
|
|
}
|
|
}
|
|
|
|
- (void)insertRow: (PGRow)row intoTable: (OFString *)table
|
|
{
|
|
void *pool = objc_autoreleasePoolPush();
|
|
OFMutableString *command;
|
|
OFEnumerator *enumerator;
|
|
const char **values;
|
|
PGresult *result;
|
|
OFString *key, *value;
|
|
size_t i, count;
|
|
|
|
command = [OFMutableString stringWithString: @"INSERT INTO "];
|
|
[command appendString: table];
|
|
[command appendString: @" ("];
|
|
|
|
count = row.count;
|
|
|
|
i = 0;
|
|
enumerator = [row keyEnumerator];
|
|
while ((key = [enumerator nextObject]) != nil) {
|
|
if (i > 0)
|
|
[command appendString: @", "];
|
|
|
|
[command appendString: key];
|
|
|
|
i++;
|
|
}
|
|
|
|
[command appendString: @") VALUES ("];
|
|
|
|
values = OFAllocMemory(count, sizeof(*values));
|
|
@try {
|
|
i = 0;
|
|
enumerator = [row objectEnumerator];
|
|
while ((value = [enumerator nextObject]) != nil) {
|
|
if (i > 0)
|
|
[command appendString: @", "];
|
|
|
|
values[i] = value.UTF8String;
|
|
|
|
[command appendFormat: @"$%zd", ++i];
|
|
}
|
|
|
|
[command appendString: @")"];
|
|
|
|
result = PQexecParams(_connection, command.UTF8String,
|
|
(int)count, NULL, values, NULL, NULL, 0);
|
|
} @finally {
|
|
OFFreeMemory(values);
|
|
}
|
|
|
|
objc_autoreleasePoolPop(pool);
|
|
|
|
if (PQresultStatus(result) != PGRES_COMMAND_OK) {
|
|
PQclear(result);
|
|
@throw [PGCommandFailedException
|
|
exceptionWithConnection: self
|
|
command: command];
|
|
}
|
|
|
|
PQclear(result);
|
|
}
|
|
|
|
- (void)insertRows: (OFArray OF_GENERIC(PGRow) *)rows
|
|
intoTable: (OFString *)table
|
|
{
|
|
for (OFDictionary *row in rows)
|
|
[self insertRow: row intoTable: table];
|
|
}
|
|
@end
|