libs-gdl2/EOAdaptors/Postgres95/Postgres95Adaptor.m
David Ayers f3e520ee9a 2003-03-30 David Ayers <d.ayers@inode.at>
* *.h/m, *.h/m):
	Used #include instead of depricated #import.  Avoid including
	entire library headers.  Use export macro where applicable.

	* EOControl/GNUmakefile: Removed EOKeyValueCodingBase.m/h and
	EOUndoMananger.h.  Added EOArrayDataSource.m/h, EODefines.h
	and EODepricated.h.
	* EOControl/EODefines.h: Updated for GDL2 & gnustep-make.
	* EOControl/EODebug.h: Updated.
	* EOControl/EODepricated.h: Updated for current state of GDL2.
	([NSObject +flushClassKeyBindings]): Added declaration.
	([EOClassDescription +setDelegate:]): Added declaration.
	([EOClassDescription +delegate]): Added declaration.
	(EOUndoManager): Moved declaration of interface here.
	* EOControl/EOKeyValueCoding.m
	([EOClassDescription +flushClassKeyBindings]): Added empty
	implementation.
	([NSObject takeStoredValuesFromDictionary:]): Cache EONull
	instance and use it instead of isKindOfClass:.
	* EOControl/EOEditingContext.m
	([EOEditingContext +initialize]): Tidied.
	([EOEditingContest -deleteObject]): Use NSUndoManager instead of
	EOUndoManager.
	* EOControl/EOQualifier.m
	([NSArray -filteredArrayUsingQualifier:]): Added minor
	optimization tweak.
	* EOControl/EOClassDescription.m: Removed inactive commented
	code.  Added private declerations of EOAccess methods to avoid
	compiler warnings.
	([EOClassDescription +initialize]): Tidied.
	([EOClassDescription -classDescriptionForClass:]): Use
	GSObjCName() instead of objc runtime routines.
	* EOControl/EOFault.m: Exchanged direct usages of ObjC runtime
	routines with NS/GSObjCRuntime abstraction API.
	([EOFault +superclass]): Ditto.
	([EOFault +targetClassForFault:]): Ditto.
	([EOFault -respondsToSelector:]): Ditto.
	([EOFault +initialize]): Cache static class variable.
	([EOFault +isKindOfClass]): Use static class variable.
	([EOFault +handlerForFault:]): Ditto.
	([EOFault +targetClassForFault:]): Ditto.
	([EOFault -dealloc]): Ditto.
	* EOControl/EOGenericRecord.m:
	([EOGenericRecord +initialize]): Tidied.
	* EOControl/EOKeyComparisonQualifier.m: Tidied documentation.
	* EOControl/EOKeyValueQualifier.m: Ditto.
	* EOControl/EONSAddOns.h/m: Added declarations to surpress
	compiler warnings.
	([NSObject -eoCompareOnName:]): Adjusted casts to surpress
	compiler warnings.
	(GSUseStrictWO451Compatibility): Added function.
	(GDL2GlobalLock, GDL2GlobalRecursive): Ditto.
	* EOControl/EONull: Remove implementations for foundation
	libraries without key value coding and fully rely on NSNull.
	Added assertions in all intance methods as instances should never
	be created.
	([EONull +allocWithZone:]) Corrected method name so it will
	actually be used.
	* EOControl/EOSortOrdering.m
	([NSArray sortedArrayUsingKeyOrderArray:]): Tidied.
	([NSMutableArray sortUsingKeyOrderArray:]): Ditto.
	([EONull compareAscending:]): Sync with referencs implementation.
	([EONull compareDescending:]): Ditto.
	([EONull compareCaseInsensitiveAscending:]): Ditto.
	([EONull compareCaseInsensitiveDescending:]): Ditto.

	* EOAccess/EOAdaptor.h: Added comment about API compatibility.
	* EOAccess/EOAdaptor.m ([EOAdaptor -contexts]): Return array of
	adaptor contexts rather tham GC-wrapper objects containing adaptor
	contexts.
	([EOAdaptor -databaseEncoding]): Use GSEncodingName() instead of
	GetEncodingName().
	* EOAccess/EODatabaseContext.m
	([EODatabaseContext -_turnFault:gid:editingContext:isComplete:]):
	Use GSObjCClass() instead of trying to access isa by dereferencing
	from id with incorrect member.
	* EOAccess/EOModel.m ([EOModel -entityNames]): Sort returned array
	to insure comparable output.
	* EOAccess/EOSQLExpression.m
	([EOSQLExpression sqlStringForArrayOfQualifiers:operation:]):
	Added cast to surpress compiler warning.
	* EOAccess/EOUtilities.m
	([EOObjectStoreCoordinator setModelGroup:]): Ditto.
	* EOAccess/EORelationship.h ([EORelationship -docComment]):
	Added declaration.
	* EOAccess/GNUmakefile: Added EODefines.h and EODepricated.h.

	* Tools/*.m: Use RCS_ID macro.
	* Tools/EOAttribute+GSDoc.h:
	([EOAttribute gsdocContentWithTagName:idPtr:]): Corrected Typo.
	* Tools/EOModel+GSDoc.h/m:
	([EOModel gsdocContentSplittedByEntities:idPtr:]): Ditto.
	* Tools/EORelationship+GSDoc.m:
	([EORelationship gsdocContentWithTagName:idPtr:]): Change variable
	type to supress compiler warnings.
	* Tools/eoutil.m (dump): Initialize variables to supress compiler
	warnings.
	* Tools/gsdoc-model.m: Include GSCategories.h to supress compiler
	warnings.
	(main): Added cast to supress compiler warning.  Fixed typo in
	method invocation.

2003-03-25  Stephane Corthesy  <stephane@sente.ch>

	* EOControl/EODefines.h: Added new file for export/win32 support.
	* EOControl/EODepricated.h: Added new file for depricated features.
	* EOControl/EOControl.h: Added EOArrayDataSource.h and EODefines.h.
	* EOControl/EOArrayDataSource.h/m: Added new files.  Some methods
	(<NSCoding> and qualifier bindings) are empty stubs.
	* EOControl/EODebug.h: Use export macro instead of explicit extern for
	function and symbol declarations.
	* EOControl/EOGlobalID.h: Ditto.
	* EOControl/EONull.h: Ditto.
	* EOControl/EOObjectStore.h: Ditto.
	* EOControl/EOOrQualifier.m: Replaced autorelease by AUTORELEASE
	and fixed typo.
	* EOControl/EOQualifier.m
	([NSArray -filteredArrayUsingQualifier:]): Implemented.
	* EOControl/EONSAddOns.m: Use volatile in some exception handlers
	(man longjmp for more info).
	* EOControl/EOSortOrdering.h/m
	([EOSortOrdering -copyWithZone:]): Implemented <NSCopying>.
	([EOSortOrdering -encodeWithKeyValueArchiver:]): Implemented.

	* EOAccess/EODefines.h: Added new file for export/win32 support.
	* EOAccess/EODepricated.h: Added new file for depricated features.
	* EOAccess/EOAccess.h: Added EODefines.h.
	* EOAccess/EOSQLExpression.h/m: Fixed typo for
	EOPrimaryKeyConstraintKey.
	* EOAccess/EOExpressionArray.h/m: Use volatile for variables usein
	in exception handlers. (man longjmp for more info)
	* EOAccess/EODatabase.h: Use export macro instead of explicit
	extern for function and symbol declarations.
	* EOAccess/EOEntity.h: Ditto.
	* EOAccess/EOModel.h: Ditto.
	* EOAccess/EOSchemaGeneration.h: Ditto.
	* EOAccess/EOSQLExpression.h: Ditto.
	* EOAccess/EOUtilities.h: Ditto.
	* Tools/eoutil.m (dump): Implemented use of -postinstall option.
	Corrected bug when getting adaptor's expression class.  Renamed
	symbol EOPrimaryKeyContraintsKey into EOPrimaryKeyConstraintsKey.


git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gdl2/trunk@16298 72102866-910b-0410-8b05-ffd578937521
2003-03-31 00:24:15 +00:00

497 lines
12 KiB
Objective-C

/**
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>
**/
#include "config.h"
RCS_ID("$Id$")
#ifndef NeXT_Foundation_LIBRARY
#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
#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>
NSString *Postgres95Exception = @"Postgres95Exception";
static int pgConnTotalAllocated = 0;
static int pgConnCurrentAllocated = 0;
@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];
[_pgConnPool release];
[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;
}*/
static NSString *externalTypeNames[] = {
#warning (stephane@sente.ch) Needs to be updated!!!
@"boolean", @"bool",
@"char", @"char2", @"char4", @"char8", @"char16", @"filename",
@"date", @"reltime", @"time", @"tinterval", @"abstime", @"timestamp",
@"real", @"double precision", @"float4", @"float8",
@"int4", @"int2",
@"oid", @"oid8", @"oidint2", @"oidint4", @"oidchar16",
@"varchar", @"bpchar",
@"numeric", @"decimal", @"cid", @"tid", @"xid",
nil
};
static NSString *internalTypeNames[] = {
@"NSNumber", @"NSNumber",
@"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber",
@"NSCalendarDate", @"NSCalendarDate", @"NSCalendarDate", @"NSCalendarDate", @"NSCalendarDate", @"NSCalendarDate",
@"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber",
@"NSNumber", @"NSNumber",
@"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber", @"NSNumber",
@"NSString", @"NSString",
@"NSDecimalNumber", @"NSDecimalNumber", @"NSDecimalNumber", @"NSDecimalNumber", @"NSDecimalNumber",
nil
};
+ (NSDictionary *)externalToInternalTypeMap
{
static NSDictionary *externalToInternalTypeMap = nil;
if (!externalToInternalTypeMap)
{
int i;
for (i = 0; externalTypeNames[i]; i++);
externalToInternalTypeMap = [[NSDictionary alloc] initWithObjects: internalTypeNames forKeys: externalTypeNames count: i];
}
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;
[adaptorChannel closeChannel];
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;
}
- (void)createDatabaseWithAdministrativeConnectionDictionary: (NSDictionary *)connectionDictionary
{
}
- (void)dropDatabaseWithAdministrativeConnectionDictionary: (NSDictionary *)connectionDictionary
{
}
/* 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)
{
//TODO: report error in an exception ?
[self privateReportError: pgConn];
PQfinish(pgConn);
pgConn = NULL;
}
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
};
*/