libs-gdl2/EOControl/EOObjectStoreCoordinator.m
David Ayers 53d93cb550 * EOAccess/EOAdaptor.m: Include GNUstep-specific headers for
non-GNUstep systems.
        * EOAccess/EOAdaptorChannel.m: Ditto.
        * EOAccess/EOAdaptorContext.m: Ditto.
        * EOAccess/EOAttribute.m: Ditto.
        * EOAccess/EODatabaseChannel.m: Ditto.
        * EOAccess/EODatabaseDataSource.m: Ditto.
        * EOAccess/EOEntity.m: Ditto.
        * EOAccess/EOModel.m: Ditto.
        * EOAccess/EOModelGroup.m: Ditto.
        * EOAccess/EORelationship.m: Ditto.
        * EOAccess/EOSQLQualifier.m: Ditto.
        * EOAccess/EOUtilities.m: Ditto.
        * EOControl/EOAndQualifier.m: Ditto.
        * EOControl/EOClassDescription.m: Ditto.
        * EOControl/EODataSource.m: Ditto.
        * EOControl/EOEditingContext.m: Ditto.
        * EOControl/EOFault.m: Ditto.
        * EOControl/EOFaultHandler.m: Ditto.
        * EOControl/EOFetchSpecification.m: Ditto.
        * EOControl/EOKeyValueArchiver.m: Ditto.
        * EOControl/EOKeyValueQualifier.m: Ditto.
        * EOControl/EOMutableKnownKeyDictionary.m: Ditto.
        * EOControl/EONSAddOns.m: Ditto.
        * EOControl/EOObjectStore.m: Ditto.
        * EOControl/EOObjectStoreCoordinator.m: Ditto.
        * EOControl/EOObserver.m: Ditto.
        * EOControl/EOOrQualifier.m: Ditto.
        * EOControl/EOQualifier.m: Ditto.
        * EOControl/EOUndoManager.m: Ditto.
        * EOModeler/EOModelExtensions.m: Ditto.

        * EOAccess/EOModel.h/m (+/-[EOModel version]) Remove.
        ([EOModel init]): Handle version internally.
        ([EOModel encodeTableOfContentsIntoPropertyList:]): Ditto.
        ([EOModel encodeIntoPropertyList:]): Ditto.
        * EOAccess/EODatabaseContext.m ([EODatabaseContext dealloc]):
        Use NSCountHashTable instead of NSCountMapTable on hash table.
        * EOAccess/EOAttribute.m ([EOAttribute
        newValueForBytes:length:]):
        Added casts to avoid warnings.


git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gdl2/trunk@18500 72102866-910b-0410-8b05-ffd578937521
2004-01-29 13:23:22 +00:00

607 lines
14 KiB
Objective-C

/**
EOObjectStoreCoordinator.m <title>EOObjectStoreCoordinator</title>
Copyright (C) 2000-2002 Free Software Foundation, Inc.
Date: June 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/NSString.h>
#include <Foundation/NSArray.h>
#include <Foundation/NSEnumerator.h>
#include <Foundation/NSDictionary.h>
#include <Foundation/NSNotification.h>
#include <Foundation/NSException.h>
#include <Foundation/NSDebug.h>
#else
#include <Foundation/Foundation.h>
#endif
#ifndef GNUSTEP
#include <GNUstepBase/GNUstep.h>
#include <GNUstepBase/GSCategories.h>
#endif
#include <EOControl/EOObjectStoreCoordinator.h>
#include <EOControl/EOEditingContext.h>
#include <EOControl/EODebug.h>
@implementation EOObjectStoreCoordinator
static EOObjectStoreCoordinator *defaultCoordinator = nil;
NSString *EOCooperatingObjectStoreWasAdded = @"EOCooperatingObjectStoreWasAdded";
NSString *EOCooperatingObjectStoreWasRemoved = @"EOCooperatingObjectStoreWasRemoved";
NSString *EOCooperatingObjectStoreNeeded = @"EOCooperatingObjectStoreNeeded";
+ (void) initialize
{
if (self == [EOObjectStoreCoordinator class])
{
Class cls = NSClassFromString(@"EODatabaseContext");
if (cls != Nil)
[cls class]; // Insure correct initialization.
}
}
- init
{
self = [super init];
_stores = [NSMutableArray new];
return self;
}
- (void)dealloc
{
NSDebugMLog(@"dealloc coordinator", "");
DESTROY(_stores);
DESTROY(_userInfo);
[super dealloc];
NSDebugMLog(@"dealloc coordinator end", "");
}
- (void)addCooperatingObjectStore: (EOCooperatingObjectStore *)store
{
if ([_stores containsObject:store] == NO)
{
[_stores addObject:store];
[[NSNotificationCenter defaultCenter]
postNotificationName: EOCooperatingObjectStoreWasAdded
object: store];
[[NSNotificationCenter defaultCenter]
addObserver: self
selector: @selector(_objectsChangedInSubStore:)
name: EOObjectsChangedInStoreNotification
object: store];
[[NSNotificationCenter defaultCenter]
addObserver: self
selector: @selector(_invalidatedAllObjectsInSubStore:)
name: EOInvalidatedAllObjectsInStoreNotification
object: store];
}
}
- (void)removeCooperatingObjectStore: (EOCooperatingObjectStore *)store
{
if ([_stores containsObject:store] == YES)
{
[_stores removeObject: store];
[[NSNotificationCenter defaultCenter]
postNotificationName: EOCooperatingObjectStoreWasRemoved
object: store];
//ODO remove aboservers
}
}
- (NSArray *)cooperatingObjectStores
{
return _stores;
}
- (void)forwardUpdateForObject: object
changes: (NSDictionary *)changes
{
[[self objectStoreForObject: object]
recordUpdateForObject: object changes: changes];
}
- (NSDictionary *)valuesForKeys: (NSArray *)keys
object: object
{
return [[self objectStoreForObject: object]
valuesForKeys:keys object: object];
}
- (void) requestStoreForGlobalID: (EOGlobalID *)globalID
fetchSpecification: (id)param1
object: (id)object
{
//TODO if object!=nil or fetchsec !=nil
[[NSNotificationCenter defaultCenter]
postNotificationName: EOCooperatingObjectStoreNeeded
object: self
userInfo: [NSDictionary dictionaryWithObject: globalID
forKey: @"globalID"]];
}
- (EOCooperatingObjectStore*)objectStoreForGlobalID: (EOGlobalID *)globalID
{
EOCooperatingObjectStore *store = nil;
NSEnumerator *storeEnum = nil;
int num = 2;
while (num)
{
storeEnum = [_stores objectEnumerator];
while ((store = [storeEnum nextObject]))
if ([store ownsGlobalID: globalID] == YES)
return store;
NSDebugMLLog(@"gsdb", @"num=%d", num);
if(--num)
[self requestStoreForGlobalID: globalID
fetchSpecification: nil
object: nil];
}
return nil;
}
- (EOCooperatingObjectStore *)objectStoreForObject: object
{
EOCooperatingObjectStore *store;
NSEnumerator *storeEnum;
int num = 2;
while (num)
{
storeEnum = [_stores objectEnumerator];
while ((store = [storeEnum nextObject]))
if ([store ownsObject: object] == YES)
return store;
NSDebugMLLog(@"gsdb", @"num=%d", num);
if(--num)
[[NSNotificationCenter defaultCenter]
postNotificationName: EOCooperatingObjectStoreNeeded
object: self
userInfo: [NSDictionary dictionaryWithObject: object
forKey: @"object"]];
}
return nil;
}
- (EOCooperatingObjectStore *)objectStoreForFetchSpecification: (EOFetchSpecification *)fetchSpecification
{
EOCooperatingObjectStore *store = nil;
NSEnumerator *storeEnum = nil;
int num = 2;
while (num)
{
storeEnum = [_stores objectEnumerator];
while ((store = [storeEnum nextObject]))
if ([store handlesFetchSpecification: fetchSpecification] == YES)
return store;
NSDebugMLLog(@"gsdb", @"num=%d", num);
if(--num)
[[NSNotificationCenter defaultCenter]
postNotificationName: EOCooperatingObjectStoreNeeded
object: self
userInfo: [NSDictionary dictionaryWithObject: fetchSpecification
forKey: @"fetchSpecification"]];
}
return nil;
}
- (NSDictionary *)userInfo
{
return _userInfo;
}
- (void)setUserInfo: (NSDictionary *)info
{
ASSIGN(_userInfo, info);
}
// TODO THREADS
+ (void)setDefaultCoordinator: (EOObjectStoreCoordinator *)coordinator
{
if (defaultCoordinator)
DESTROY(defaultCoordinator);
ASSIGN(defaultCoordinator, coordinator);
}
+ (id)defaultCoordinator
{
if (defaultCoordinator == nil)
defaultCoordinator = [EOObjectStoreCoordinator new];
return defaultCoordinator;
}
// EOObjectStore methods
- (id)faultForGlobalID: (EOGlobalID *)globalID
editingContext: (EOEditingContext *)context
{
id fault = nil;
EOCooperatingObjectStore *objectStore = [self objectStoreForGlobalID:
globalID];
if (objectStore)
{
fault = [objectStore faultForGlobalID: globalID
editingContext: context];
}
return fault;
}
- (NSArray *)arrayFaultWithSourceGlobalID: (EOGlobalID *)globalID
relationshipName: (NSString *)name
editingContext: (EOEditingContext *)context
{
return [[self objectStoreForGlobalID: globalID]
arrayFaultWithSourceGlobalID: globalID
relationshipName: name
editingContext: context];
}
- (NSArray *)objectsForSourceGlobalID: (EOGlobalID *)globalID
relationshipName: (NSString *)name
editingContext: (EOEditingContext *)context
{
return [[self objectStoreForGlobalID: globalID]
objectsForSourceGlobalID: globalID
relationshipName: name
editingContext: context];
}
- (void)refaultObject: object
withGlobalID: (EOGlobalID *)globalID
editingContext: (EOEditingContext *)context
{
[[self objectStoreForGlobalID: globalID]
refaultObject: object
withGlobalID: globalID
editingContext: context];
}
- (void)initializeObject: (id)object
withGlobalID: (EOGlobalID *)globalID
editingContext: (EOEditingContext *)context
{
EOCooperatingObjectStore *objectStore = [self objectStoreForGlobalID:
globalID];
[objectStore initializeObject: object
withGlobalID: globalID
editingContext: context];
}
- (void)saveChangesInEditingContext: (EOEditingContext *)context
{
NSArray *insertedObjects;
EOCooperatingObjectStore *objectStore = nil;
NSException *exception = nil;
int i, count;
EOFLOGObjectFnStart();
insertedObjects = [context insertedObjects];
count = [insertedObjects count];
//TODO for inserted: verify
for (i = 0; i < count; i++)
{
id object = [insertedObjects objectAtIndex: i];
objectStore = [self objectStoreForObject: object];
//SO WHAT //TODO
}
count = [_stores count];
for (i = 0; i < count; i++)
{
objectStore = [_stores objectAtIndex: i];
if ([objectStore respondsToSelector: @selector(lock)] == YES)
[(id)objectStore lock];
}
NS_DURING
{
count = [_stores count];
for (i = 0; i < count; i++)
{
objectStore = [_stores objectAtIndex: i];
[objectStore prepareForSaveWithCoordinator: self
editingContext: context];
}
count = [_stores count];
for (i = 0; i < count; i++)
{
// Contructs a list of EODatabaseOperations
// for all changes in the EditingContext
objectStore = [_stores objectAtIndex: i];
[objectStore recordChangesInEditingContext];
}
NS_DURING
{
count = [_stores count];
for (i = 0; i < count; i++)
{
objectStore = [_stores objectAtIndex: i];
[objectStore performChanges];
}
count = [_stores count];
for (i = 0; i < count; i++)
{
objectStore = [_stores objectAtIndex: i];
[objectStore commitChanges];
}
}
NS_HANDLER
{
NSDebugMLog(@"Exception: %@", localException);
exception = localException;
count = [_stores count];
for (i = 0; i < count; i++)
{
NS_DURING
{
[objectStore rollbackChanges];
}
NS_HANDLER
{
NSEmitTODO();
NSDebugMLog(@"Exception in exception: %@", localException);
NSLog(@"Exception in exception: %@", localException);
}
NS_ENDHANDLER;
}
}
NS_ENDHANDLER;
}
NS_HANDLER
{
exception = localException;
}
NS_ENDHANDLER;
count = [_stores count];
for (i = 0; i < count; i++)
{
objectStore = [_stores objectAtIndex: i];
if ([objectStore respondsToSelector: @selector(unlock)] == YES)
[(id)objectStore unlock];
}
if (exception)
[exception raise];
EOFLOGObjectFnStop();
}
- (NSArray *)objectsWithFetchSpecification: (EOFetchSpecification *)fetch
editingContext: (EOEditingContext *)context
{
EOCooperatingObjectStore *objectStore =
[self objectStoreForFetchSpecification: fetch];
return [objectStore objectsWithFetchSpecification: fetch
editingContext: context];
}
- (void) _invalidatedAllObjectsInSubStore: (NSNotification*)notification
{
[self notImplemented: _cmd]; //TODO
}
- (void) _objectsChangedInSubStore: (NSNotification*)notification
{
EOFLOGObjectFnStart(); //TODO
if ([notification object] != self)
{
[[NSNotificationCenter defaultCenter]
postNotificationName: EOObjectsChangedInStoreNotification
object: self
userInfo: [notification userInfo]]; //call _objectsChangedInStore: # EditingContext
}
EOFLOGObjectFnStop();
}
- (void)invalidateAllObjects
{
EOCooperatingObjectStore *store;
NSEnumerator *storeEnum;
EOFLOGObjectFnStart();
storeEnum = [_stores objectEnumerator];
while ((store = [storeEnum nextObject]))
[store invalidateAllObjects];
EOFLOGObjectFnStop();
}
- (void)invalidateObjectsWithGlobalIDs: (NSArray *)globalIDs
{
NSMutableArray *buf;
NSMutableArray *gids, *gidsToRemove;
NSEnumerator *gidEnum;
EOCooperatingObjectStore *store;
EOGlobalID *gid;
EOFLOGObjectFnStart();
buf = [NSMutableArray arrayWithArray: globalIDs];
gids = [NSMutableArray arrayWithCapacity: [globalIDs count]];
gidsToRemove = [NSMutableArray arrayWithCapacity: 16];
while (YES)
{
if (![buf count])
break;
store = nil;
gidEnum = [buf objectEnumerator];
while ((gid = [gidEnum nextObject]))
{
if (store == nil)
{
if ((store = [self objectStoreForGlobalID: gid]) == nil)
[buf addObject: gid];
}
else if ([store ownsGlobalID: gid] == YES)
[gids addObject: gid];
}
[store invalidateObjectsWithGlobalIDs: gids];
[buf removeObjectsInArray: gids];
[buf removeObjectsInArray: gidsToRemove];
[gids removeAllObjects];
[gidsToRemove removeAllObjects];
}
EOFLOGObjectFnStop();
}
@end
@implementation EOCooperatingObjectStore
- (BOOL)ownsGlobalID: (EOGlobalID *)globalID
{
[self subclassResponsibility: _cmd];
return NO;
}
- (BOOL)ownsObject: (id)object
{
[self subclassResponsibility: _cmd];
return NO;
}
- (BOOL)ownsEntityNamed: (NSString *)entityName
{
[self subclassResponsibility: _cmd];
return NO;
}
- (BOOL)handlesFetchSpecification: (EOFetchSpecification *)fetchSpecification
{
[self subclassResponsibility: _cmd];
return NO;
}
- (void)prepareForSaveWithCoordinator: (EOObjectStoreCoordinator *)coordinator
editingContext: (EOEditingContext *)context
{
[self subclassResponsibility: _cmd];
}
- (void)recordChangesInEditingContext
{
[self subclassResponsibility: _cmd];
}
- (void)recordUpdateForObject: object
changes: (NSDictionary *)changes
{
[self subclassResponsibility: _cmd];
}
- (void)performChanges
{
[self subclassResponsibility: _cmd];
}
- (void)commitChanges
{
[self subclassResponsibility: _cmd];
}
- (void)rollbackChanges
{
[self subclassResponsibility: _cmd];
}
- (NSDictionary *)valuesForKeys: (NSArray *)keys
object: object
{
[self subclassResponsibility: _cmd];
return nil;
}
@end