2002-11-15 22:57:05 +00:00
|
|
|
/**
|
|
|
|
Postgres95Adaptor.m <title>Postgres95Adaptor</title>
|
|
|
|
|
|
|
|
Copyright (C) 2000 Free Software Foundation, Inc.
|
|
|
|
|
|
|
|
Author: Mirko Viviani <mirko.viviani@rccr.cremona.it>
|
|
|
|
Date: February 2000
|
|
|
|
|
|
|
|
based on the Postgres95 adaptor written by
|
|
|
|
Mircea Oancea <mircea@jupiter.elcom.pub.ro>
|
|
|
|
|
|
|
|
Author: Manuel Guesdon <mguesdon@orange-concept.com>
|
|
|
|
Date: October 2000
|
|
|
|
|
|
|
|
$Revision$
|
|
|
|
$Date$
|
|
|
|
|
|
|
|
<abstract></abstract>
|
|
|
|
|
|
|
|
This file is part of the GNUstep Database Library.
|
|
|
|
|
|
|
|
<license>
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Library General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This library 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
|
|
|
|
Library General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Library General Public
|
|
|
|
License along with this library; see the file COPYING.LIB.
|
|
|
|
If not, write to the Free Software Foundation,
|
|
|
|
59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
</license>
|
|
|
|
**/
|
|
|
|
|
2002-12-31 16:25:21 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
RCS_ID("$Id$")
|
2002-11-15 22:57:05 +00:00
|
|
|
|
2004-03-19 16:41:21 +00:00
|
|
|
#ifdef GNUSTEP
|
2003-03-31 00:24:15 +00:00
|
|
|
#include <Foundation/NSArray.h>
|
|
|
|
#include <Foundation/NSDictionary.h>
|
|
|
|
#include <Foundation/NSSet.h>
|
|
|
|
#include <Foundation/NSValue.h>
|
|
|
|
#include <Foundation/NSString.h>
|
|
|
|
#include <Foundation/NSUtilities.h>
|
|
|
|
#include <Foundation/NSDate.h>
|
|
|
|
#include <Foundation/NSAutoreleasePool.h>
|
|
|
|
#include <Foundation/NSException.h>
|
|
|
|
#include <Foundation/NSDebug.h>
|
|
|
|
#else
|
|
|
|
#include <Foundation/Foundation.h>
|
|
|
|
#endif
|
|
|
|
|
2003-07-11 19:04:05 +00:00
|
|
|
#ifndef GNUSTEP
|
2003-08-25 20:01:59 +00:00
|
|
|
#include <GNUstepBase/GNUstep.h>
|
2004-01-31 13:57:38 +00:00
|
|
|
#include <GNUstepBase/GSCategories.h>
|
2003-07-11 19:04:05 +00:00
|
|
|
#endif
|
|
|
|
|
2003-06-24 16:27:01 +00:00
|
|
|
#include <EOControl/EONSAddOns.h>
|
2003-03-31 00:24:15 +00:00
|
|
|
#include <EOControl/EODebug.h>
|
|
|
|
|
|
|
|
#include <EOAccess/EOAccess.h>
|
|
|
|
#include <EOAccess/EOAttribute.h>
|
|
|
|
#include <EOAccess/EOExpressionArray.h>
|
|
|
|
#include <EOAccess/EOEntity.h>
|
|
|
|
#include <EOAccess/EOModel.h>
|
|
|
|
|
|
|
|
#include <Postgres95EOAdaptor/Postgres95Adaptor.h>
|
|
|
|
#include <Postgres95EOAdaptor/Postgres95Context.h>
|
|
|
|
#include <Postgres95EOAdaptor/Postgres95Channel.h>
|
|
|
|
#include <Postgres95EOAdaptor/Postgres95SQLExpression.h>
|
|
|
|
#include <Postgres95EOAdaptor/Postgres95Values.h>
|
2002-11-15 22:57:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
NSString *Postgres95Exception = @"Postgres95Exception";
|
|
|
|
static int pgConnTotalAllocated = 0;
|
|
|
|
static int pgConnCurrentAllocated = 0;
|
|
|
|
|
2003-06-24 16:27:01 +00:00
|
|
|
int
|
|
|
|
postgresClientVersion()
|
|
|
|
{
|
|
|
|
static int version = 0;
|
|
|
|
if (version == 0)
|
|
|
|
{
|
|
|
|
NSString *versionString = [NSString stringWithCString: PG_VERSION];
|
|
|
|
version = [versionString parsedFirstVersionSubstring];
|
|
|
|
}
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
2002-11-15 22:57:05 +00:00
|
|
|
|
|
|
|
@implementation Postgres95Adaptor
|
|
|
|
|
|
|
|
- init
|
|
|
|
{
|
|
|
|
return [self initWithName: @"Postgres95"];
|
|
|
|
}
|
|
|
|
|
|
|
|
- initWithName: (NSString *)name
|
|
|
|
{
|
|
|
|
if ((self = [super initWithName: name]))
|
|
|
|
{
|
|
|
|
_pgConnPool = [NSMutableArray new];
|
|
|
|
}
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)dealloc
|
|
|
|
{
|
|
|
|
NSEnumerator *enumerator;
|
|
|
|
PGconn *pgConn;
|
|
|
|
|
|
|
|
enumerator = [_pgConnPool objectEnumerator];
|
|
|
|
|
|
|
|
while ((pgConn = [[enumerator nextObject] pointerValue]))
|
|
|
|
[self releasePGconn: pgConn force: YES];
|
|
|
|
|
2004-04-17 12:41:32 +00:00
|
|
|
DESTROY(_pgConnPool);
|
2002-11-15 22:57:05 +00:00
|
|
|
|
|
|
|
[super dealloc];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)privateReportError: (PGconn*)pgConn
|
|
|
|
{
|
|
|
|
char *message = "NULL pgConn in privateReportError:";
|
|
|
|
|
|
|
|
EOFLOGObjectFnStart();
|
|
|
|
|
|
|
|
if (pgConn)
|
|
|
|
message = PQerrorMessage(pgConn);
|
|
|
|
|
|
|
|
NSLog(@"%s", message);
|
|
|
|
|
|
|
|
EOFLOGObjectFnStop();
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)setCachePGconn: (BOOL)flag
|
|
|
|
{
|
|
|
|
_flags.cachePGconn = flag;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)cachePGconn
|
|
|
|
{
|
|
|
|
return _flags.cachePGconn;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)setPGconnPoolLimit: (int)value
|
|
|
|
{
|
|
|
|
_pgConnPoolLimit = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (int)pgConnPoolLimit
|
|
|
|
{
|
|
|
|
return _pgConnPoolLimit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*+ (NSDictionary *)defaultConnectionDictionary
|
|
|
|
{
|
|
|
|
static NSDictionary *dict = nil;
|
|
|
|
|
|
|
|
if (!dict)
|
|
|
|
dict = [[NSDictionary dictionaryWithObjectsAndKeys:NSHomeDirectory(), FlatFilePathKey, [self defaultRowSeparator], FlatFileRowSeparatorKey, [self defaultColumnSeparator], FlatFileColumnSeparatorKey, @"Y", FlatFileUseHeadersKey, nil] retain];
|
|
|
|
|
|
|
|
return dict;
|
|
|
|
}*/
|
|
|
|
|
2004-02-25 12:06:54 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
The first 4 must correspond to EOAdaptorValueTypes:
|
|
|
|
EOAdaptorNumberType, EOAdaptorCharactersType,
|
|
|
|
EOAdaptorBytesType, EOAdaptorDateType
|
|
|
|
*/
|
2002-11-15 22:57:05 +00:00
|
|
|
static NSString *externalTypeNames[] = {
|
2003-02-03 14:53:03 +00:00
|
|
|
#warning (stephane@sente.ch) Needs to be updated!!!
|
2004-09-09 10:40:50 +00:00
|
|
|
@"numeric", @"varchar", @"bytea", @"date",
|
2003-02-03 14:53:03 +00:00
|
|
|
@"boolean", @"bool",
|
2002-11-15 22:57:05 +00:00
|
|
|
@"char", @"char2", @"char4", @"char8", @"char16", @"filename",
|
2004-02-25 12:06:54 +00:00
|
|
|
@"reltime", @"time", @"tinterval", @"abstime", @"timestamp",
|
2003-02-03 14:53:03 +00:00
|
|
|
@"real", @"double precision", @"float4", @"float8",
|
2005-02-03 17:20:51 +00:00
|
|
|
@"bigint", @"int8", @"integer", @"int4", @"smallint", @"int2",
|
2002-11-15 22:57:05 +00:00
|
|
|
@"oid", @"oid8", @"oidint2", @"oidint4", @"oidchar16",
|
2004-09-09 10:40:50 +00:00
|
|
|
@"decimal", @"cid",
|
|
|
|
@"tid", @"xid",
|
|
|
|
@"bpchar",
|
2002-11-15 22:57:05 +00:00
|
|
|
nil
|
|
|
|
};
|
|
|
|
|
|
|
|
static NSString *internalTypeNames[] = {
|
2004-02-25 12:06:54 +00:00
|
|
|
@"NSNumber", @"NSString", @"NSData", @"NSCalendarDate",
|
2002-11-15 22:57:05 +00:00
|
|
|
@"NSNumber", @"NSNumber",
|
2004-02-25 12:06:54 +00:00
|
|
|
@"NSString", @"NSString", @"NSString", @"NSString", @"NSString", @"NSString",
|
|
|
|
@"NSCalendarDate", @"NSCalendarDate", @"NSCalendarDate", @"NSCalendarDate", @"NSCalendarDate",
|
2003-02-03 14:53:03 +00:00
|
|
|
@"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber",
|
2005-02-03 17:20:51 +00:00
|
|
|
@"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber",
|
2002-11-15 22:57:05 +00:00
|
|
|
@"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber",
|
2004-09-09 10:40:50 +00:00
|
|
|
@"NSDecimalNumber", @"NSDecimalNumber",
|
|
|
|
@"NSDecimalNumber", @"NSDecimalNumber",
|
|
|
|
@"NSData",
|
2002-11-15 22:57:05 +00:00
|
|
|
nil
|
|
|
|
};
|
|
|
|
|
|
|
|
+ (NSDictionary *)externalToInternalTypeMap
|
|
|
|
{
|
|
|
|
static NSDictionary *externalToInternalTypeMap = nil;
|
|
|
|
|
|
|
|
if (!externalToInternalTypeMap)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; externalTypeNames[i]; i++);
|
|
|
|
|
2003-02-03 14:53:03 +00:00
|
|
|
externalToInternalTypeMap = [[NSDictionary alloc] initWithObjects: internalTypeNames forKeys: externalTypeNames count: i];
|
2002-11-15 22:57:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return externalToInternalTypeMap;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSString *)internalTypeForExternalType: (NSString *)extType
|
|
|
|
model: (EOModel *)model
|
|
|
|
{
|
|
|
|
return [[self externalToInternalTypeMap] objectForKey: extType];
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSArray *)externalTypesWithModel:(EOModel *)model
|
|
|
|
{
|
|
|
|
return [[self externalToInternalTypeMap] allKeys];
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (void)assignExternalInfoForAttribute: (EOAttribute *)attribute
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
EOAdaptorValueType value = [attribute adaptorValueType];
|
|
|
|
|
|
|
|
[attribute setExternalType: externalTypeNames[value]];
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (void)assignExternalInfoForEntity: (EOEntity *)entity
|
|
|
|
{
|
|
|
|
NSEnumerator *attributeEnumerator = [[entity attributes] objectEnumerator];
|
|
|
|
EOAttribute *attribute;
|
|
|
|
|
|
|
|
while ((attribute = [attributeEnumerator nextObject]))
|
|
|
|
[self assignExternalInfoForAttribute: attribute];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Inherited methods */
|
|
|
|
|
|
|
|
- (EOAdaptorContext *)createAdaptorContext
|
|
|
|
{
|
|
|
|
//OK
|
|
|
|
return [Postgres95Context adaptorContextWithAdaptor: self];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (Class)defaultExpressionClass
|
|
|
|
{
|
|
|
|
Class expressionClass;
|
|
|
|
|
|
|
|
EOFLOGObjectFnStart();
|
|
|
|
|
|
|
|
expressionClass = [Postgres95SQLExpression class];
|
|
|
|
|
|
|
|
EOFLOGObjectFnStop();
|
|
|
|
|
|
|
|
return expressionClass;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isValidQualifierType: (NSString *)typeName
|
|
|
|
model: (EOModel *)model
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; externalTypeNames[i]; i++)
|
|
|
|
{
|
|
|
|
//TODO REMOVE
|
|
|
|
NSDebugMLog(@"externalTypeNames[i]=%@", externalTypeNames[i]);
|
|
|
|
|
|
|
|
if ([externalTypeNames[i] isEqualToString: typeName])
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
//TODO REMOVE
|
|
|
|
|
|
|
|
NSDebugMLog(@"typeName=%@", typeName);
|
|
|
|
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)assertConnectionDictionaryIsValid
|
|
|
|
{
|
|
|
|
NSException *exception = nil;
|
|
|
|
EOAdaptorContext *adaptorContext;
|
|
|
|
EOAdaptorChannel *adaptorChannel;
|
|
|
|
|
|
|
|
if (![self hasOpenChannels])
|
|
|
|
{
|
|
|
|
adaptorContext = [self createAdaptorContext];
|
|
|
|
adaptorChannel = [adaptorContext createAdaptorChannel];
|
|
|
|
|
|
|
|
NS_DURING
|
|
|
|
[adaptorChannel openChannel];
|
|
|
|
NS_HANDLER
|
|
|
|
exception = localException;
|
|
|
|
NS_ENDHANDLER;
|
|
|
|
|
2004-03-06 11:12:49 +00:00
|
|
|
if ([adaptorChannel isOpen])
|
|
|
|
[adaptorChannel closeChannel];
|
2002-11-15 22:57:05 +00:00
|
|
|
|
|
|
|
if (exception)
|
|
|
|
[exception raise];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
-(NSString *)formatValue:(id)value
|
|
|
|
forAttribute:(EOAttribute*)attribute
|
|
|
|
{
|
|
|
|
return [value stringValueForPostgres95Type:[attribute externalType]
|
|
|
|
attribute:attribute];
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
- (NSString *)fetchedValueForString: (NSString *)value
|
|
|
|
attribute: (EOAttribute *)attribute
|
|
|
|
{
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//TODO: don't need to be overriden ??
|
|
|
|
- (NSNumber *)fetchedValueForNumberValue: (NSNumber *)value
|
|
|
|
attribute: (EOAttribute *)attribute
|
|
|
|
{
|
|
|
|
return value; // TODO scale and precision
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSCalendarDate *)fetchedValueForDateValue: (NSCalendarDate *)value
|
|
|
|
attribute: (EOAttribute *)attribute
|
|
|
|
{
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSData *)fetchedValueForDataValue: (NSData *)value
|
|
|
|
attribute: (EOAttribute *)attribute
|
|
|
|
{
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Private methods for Postgres Adaptor */
|
|
|
|
|
|
|
|
- (PGconn *)createPGconn
|
|
|
|
{
|
|
|
|
char *pg_host = NULL;
|
|
|
|
char *pg_database = NULL;
|
|
|
|
char *pg_port = NULL;
|
|
|
|
char *pg_options = NULL;
|
|
|
|
char *pg_tty = NULL;
|
|
|
|
char *pg_user = NULL;
|
|
|
|
char *pg_pwd = NULL;
|
|
|
|
PGconn *pgConn = NULL;
|
|
|
|
PGresult *pgResult = NULL;
|
|
|
|
NSString *str = nil;
|
|
|
|
|
|
|
|
EOFLOGObjectFnStart();
|
|
|
|
|
|
|
|
//OK
|
|
|
|
str = [_connectionDictionary objectForKey: @"databaseServer"];
|
|
|
|
if (!str)
|
|
|
|
str = [_connectionDictionary objectForKey: @"hostName"];
|
|
|
|
|
|
|
|
pg_host = (char*)[str cString];
|
|
|
|
|
|
|
|
pg_database = (char*)[[_connectionDictionary objectForKey: @"databaseName"]
|
|
|
|
cString];
|
|
|
|
pg_port = (char*)[[_connectionDictionary objectForKey: @"port"] cString];
|
|
|
|
if (!pg_port)
|
|
|
|
pg_port = (char*)[[_connectionDictionary objectForKey: @"hostPort"]
|
|
|
|
cString];
|
|
|
|
|
|
|
|
pg_options = (char*)[[_connectionDictionary objectForKey: @"options"]
|
|
|
|
cString];
|
|
|
|
pg_tty = (char*)[[_connectionDictionary objectForKey: @"debugTTY"] cString];
|
|
|
|
pg_user = (char*)[[_connectionDictionary objectForKey: @"userName"]
|
|
|
|
cString];
|
|
|
|
pg_pwd = (char*)[[_connectionDictionary objectForKey: @"password"]
|
|
|
|
cString];
|
|
|
|
|
|
|
|
NSDebugMLog(@"%s %s %s %s %s", pg_host, pg_port, pg_database, pg_user, pg_pwd);
|
|
|
|
|
|
|
|
// Try to connect to the Postgres95 server
|
|
|
|
if (pg_user)
|
|
|
|
pgConn = PQsetdbLogin(pg_host, pg_port, pg_options, pg_tty,
|
|
|
|
pg_database,pg_user,pg_pwd);
|
|
|
|
else
|
|
|
|
pgConn = PQsetdb(pg_host, pg_port, pg_options, pg_tty, pg_database);
|
|
|
|
|
|
|
|
NSDebugMLog(@"%s %s %s %s %s", pg_host, pg_port, pg_database, pg_user,
|
|
|
|
pg_pwd);
|
|
|
|
|
|
|
|
// Check connection
|
|
|
|
if (PQstatus(pgConn) == CONNECTION_BAD)
|
|
|
|
{
|
2004-03-08 11:13:58 +00:00
|
|
|
NSString *reason;
|
|
|
|
|
|
|
|
reason = [NSString stringWithCString:PQerrorMessage(pgConn)];
|
2002-11-15 22:57:05 +00:00
|
|
|
[self privateReportError: pgConn];
|
|
|
|
PQfinish(pgConn);
|
2004-03-08 11:13:58 +00:00
|
|
|
[[NSException exceptionWithName:@"InvalidConnection"
|
|
|
|
reason: reason
|
|
|
|
userInfo:nil] raise];
|
2002-11-15 22:57:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (pgConn)
|
|
|
|
{
|
|
|
|
pgResult = PQexec(pgConn, "SET DATESTYLE TO 'SQL'");
|
|
|
|
PQclear(pgResult);
|
|
|
|
pgResult = NULL;
|
|
|
|
|
|
|
|
if (pgConn)
|
|
|
|
{
|
|
|
|
pgConnTotalAllocated++;
|
|
|
|
pgConnCurrentAllocated++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
EOFLOGObjectFnStop();
|
|
|
|
|
|
|
|
return pgConn;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (PGconn *)newPGconn
|
|
|
|
{
|
|
|
|
PGconn *pgConn = NULL;
|
|
|
|
|
|
|
|
if(_flags.cachePGconn && [_pgConnPool count])
|
|
|
|
{
|
|
|
|
NSDebugMLog(@"newPGconn cached %p (pgConn=%p) total=%d current=%d",
|
|
|
|
self,
|
|
|
|
pgConn,
|
|
|
|
pgConnTotalAllocated,
|
|
|
|
pgConnCurrentAllocated);
|
|
|
|
|
|
|
|
pgConn = [[_pgConnPool lastObject] pointerValue];
|
|
|
|
[_pgConnPool removeLastObject];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pgConn = [self createPGconn];
|
|
|
|
|
|
|
|
NSDebugMLog(@"newPGconn not cached %p (pgConn=%p) total=%d current=%d",
|
|
|
|
self,
|
|
|
|
pgConn,
|
|
|
|
pgConnTotalAllocated,
|
|
|
|
pgConnCurrentAllocated);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pgConn;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)releasePGconn: (PGconn *)pgConn
|
|
|
|
force: (BOOL)flag
|
|
|
|
{
|
|
|
|
if (!flag
|
|
|
|
&& _flags.cachePGconn
|
|
|
|
&& (PQstatus(pgConn) == CONNECTION_OK)
|
|
|
|
&& [_pgConnPool count] < _pgConnPoolLimit)
|
|
|
|
{
|
|
|
|
NSDebugMLog(@"releasePGconn -> in pool %p (pgConn=%p) total=%d current=%d",
|
|
|
|
self,
|
|
|
|
pgConn,
|
|
|
|
pgConnTotalAllocated,
|
|
|
|
pgConnCurrentAllocated);
|
|
|
|
|
|
|
|
[_pgConnPool addObject: [NSValue value: pgConn
|
|
|
|
withObjCType: @encode(PGconn*)]];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NSDebugMLog(@"releasePGconn really %p (pgConn=%p) total=%d current=%d",
|
|
|
|
self,
|
|
|
|
pgConn,
|
|
|
|
pgConnTotalAllocated,
|
|
|
|
pgConnCurrentAllocated);
|
|
|
|
|
|
|
|
pgConnCurrentAllocated--;
|
|
|
|
PQfinish(pgConn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// format is something like @"%@_SEQ" or @"EOSEQ_%@", "%@" is replaced by external table name
|
|
|
|
- (void)setPrimaryKeySequenceNameFormat: (NSString*)format
|
|
|
|
{
|
|
|
|
ASSIGN(_primaryKeySequenceNameFormat, format);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString*)primaryKeySequenceNameFormat
|
|
|
|
{
|
|
|
|
if (!_primaryKeySequenceNameFormat)
|
|
|
|
_primaryKeySequenceNameFormat = [_connectionDictionary objectForKey: @"primaryKeySequenceNameFormat"];
|
|
|
|
|
|
|
|
if (!_primaryKeySequenceNameFormat)
|
|
|
|
_primaryKeySequenceNameFormat = @"%@_SEQ";
|
|
|
|
|
|
|
|
return _primaryKeySequenceNameFormat;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end /* Postgres95Adaptor */
|
|
|
|
|
|
|
|
/*
|
|
|
|
//TODO
|
|
|
|
databaseEncoding
|
|
|
|
{
|
|
|
|
self connectionDictionary
|
|
|
|
call dict obj for key databaseEncoding
|
|
|
|
return 2 par defaut
|
|
|
|
};
|
|
|
|
*/
|