/** GSWSession.m - GSWeb: Class GSWSession Copyright (C) 1999-2004 Free Software Foundation, Inc. Written by: Manuel Guesdon Date: Jan 1999 $Revision$ $Date$ $Id$ This file is part of the GNUstep Web Library. 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; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. **/ #include "config.h" #include "GSWeb.h" #include "GSWPrivate.h" #include #include #include #include #if __linux__ #include #include #endif //==================================================================== @interface GSWSession (Private) -(void)_setContextArrayStack:(NSArray*)contextArrayStack; -(void)_setContextRecords:(NSDictionary*)contextRecords; -(void)_setComponentState:(NSDictionary*)componentState; -(void)_setStatistics:(NSArray*)statistics; -(void)_setFormattedStatistics:(NSString*)formattedStatistics; -(void)_setContextCounter:(int)contextCounter; -(void)_setRequestCounter:(int)requestCounter; @end //==================================================================== @implementation GSWSession extern id gcObjectsToBeVisited; //-------------------------------------------------------------------- // init -(id)init { if ((self = [super init])) { NSNumber *mytimeOutNum = [[GSWApp class] sessionTimeOut]; NSTimeInterval mySessionTimeOut = [mytimeOutNum doubleValue]; _autoreleasePool = nil; [self setTimeOut:mySessionTimeOut]; [self _initWithSessionID:[[self class] createSessionID]]; } return self; } //-------------------------------------------------------------------- -(id)copyWithZone: (NSZone*)zone { GSWSession* clone = [[isa allocWithZone: zone] init]; [clone setSessionID:_sessionID]; [clone setTimeOut:_timeOut]; [clone _setContextArrayStack:_contextArrayStack]; [clone _setContextRecords:_contextRecords]; //_editingContext: no [clone setLanguages:_languages]; [clone _setComponentState:_componentState]; [clone _setBirthDate:_birthDate]; //_wasTimedOut: no [clone _setStatistics:_statistics]; [clone _setFormattedStatistics:_formattedStatistics]; [clone _setContext:_currentContext]; //_permanentPageCache: //_permanentContextIDArray: no [clone _setContextCounter:_contextCounter]; [clone _setRequestCounter:_requestCounter]; [clone _setAllowedToViewStatistics:_isAllowedToViewStatistics]; [clone _setAllowedToViewEvents:_isAllowedToViewEvents]; //_isTerminating: no [clone setDistributionEnabled:_isDistributionEnabled]; [clone setStoresIDsInURLs:_storesIDsInURLs]; [clone setStoresIDsInCookies:_storesIDsInCookies]; //_hasSessionLockedEditingContext: no return clone; } //-------------------------------------------------------------------- +(NSString*)createSessionID { // The idea is to have uniq sessionID generated. // Parts are: // o a modified TimeStamp (modified because we don't want to give // information on server exact time which can be always a security // problem), so we can remember this sessionID for long time without conflict // o a md5 sum of various elements // The generated session ID is a sizeof(time_t)+16 bytes string NSString* sessionID=nil; NSMutableData* data=nil; NSMutableData* md5Data=nil; NSData* md5Sum=nil; void* pMd5Data=NULL; time_t ts=time(NULL); int sizeToFill=64; md5Data=[NSMutableData dataWithLength:64]; pMd5Data=[md5Data mutableBytes]; // initialize random generator // We xor time stamp with a pointer so 2 sessions created at the same // time won't have the same random generator initializer srand(((unsigned long int)ts) ^ ((unsigned long int)md5Data)); // We randomize on 60s ts=ts+(int)(60*rand()/(RAND_MAX+1.0)); data=[NSMutableData dataWithBytes:&ts length:sizeof(ts)]; // Now, use some system related chnaging info ( #if __linux__ { struct sysinfo info; if ((sysinfo(&info)) == 0) { NSUInteger rnd; // >0 test is to ignore not changing elements if (sizeToFill>=sizeof(NSUInteger) && info.uptime>0) { rnd=(unsigned)(((float)UINT_MAX)*rand()/(RAND_MAX+1.0)); *((NSUInteger*)pMd5Data)=(((NSUInteger)(info.uptime)) ^ rnd); sizeToFill-=sizeof(NSUInteger); pMd5Data+=sizeof(NSUInteger); } if (sizeToFill>=sizeof(NSUInteger) && info.loads[0]>0) { rnd=(unsigned)(((float)UINT_MAX)*rand()/(RAND_MAX+1.0)); *((NSUInteger*)pMd5Data)=(((NSUInteger)(info.loads[0] >> 4)) ^ rnd); sizeToFill-=sizeof(NSUInteger); pMd5Data+=sizeof(NSUInteger); } if (sizeToFill>=sizeof(NSUInteger) && info.loads[1]>0) { rnd=(unsigned)(((float)UINT_MAX)*rand()/(RAND_MAX+1.0)); *((NSUInteger*)pMd5Data)=(((NSUInteger)(info.loads[1] >> 4)) ^ rnd); sizeToFill-=sizeof(NSUInteger); pMd5Data+=sizeof(NSUInteger); } if (sizeToFill>=sizeof(NSUInteger) && info.loads[2]>0) { rnd=(unsigned)(((float)UINT_MAX)*rand()/(RAND_MAX+1.0)); *((NSUInteger*)pMd5Data)=(((NSUInteger)(info.loads[2] >> 4)) ^ rnd); sizeToFill-=sizeof(NSUInteger); pMd5Data+=sizeof(NSUInteger); } if (sizeToFill>=sizeof(NSUInteger) && info.freeram>0) { rnd=(unsigned)(((float)UINT_MAX)*rand()/(RAND_MAX+1.0)); *((NSUInteger*)pMd5Data)=(((NSUInteger)(info.freeram >> 4)) ^ rnd); // Drop 4 minor bits sizeToFill-=sizeof(NSUInteger); pMd5Data+=sizeof(NSUInteger); } if (sizeToFill>=sizeof(NSUInteger) && info.sharedram>0) { rnd=(unsigned)(((float)UINT_MAX)*rand()/(RAND_MAX+1.0)); *((NSUInteger*)pMd5Data)=(((NSUInteger)(info.sharedram >> 4)) ^ rnd); // Drop 4 minor bits sizeToFill-=sizeof(NSUInteger); pMd5Data+=sizeof(NSUInteger); } if (sizeToFill>=sizeof(NSUInteger) && info.freeswap>0) { rnd=(unsigned)(((float)UINT_MAX)*rand()/(RAND_MAX+1.0)); *((NSUInteger*)pMd5Data)=(((NSUInteger)(info.freeswap >> 4)) ^ rnd); // Drop 4 minor bits sizeToFill-=sizeof(NSUInteger); pMd5Data+=sizeof(NSUInteger); } if (sizeToFill>=sizeof(NSUInteger) && info.bufferram>0) { rnd=(unsigned)(((float)UINT_MAX)*rand()/(RAND_MAX+1.0)); *((NSUInteger*)pMd5Data)=(((NSUInteger)(info.bufferram >> 4)) ^ rnd); // Drop 4 minor bits sizeToFill-=sizeof(NSUInteger); pMd5Data+=sizeof(NSUInteger); } } } #endif while(sizeToFill>0) { *((unsigned char*)pMd5Data)=(unsigned char)(256.0*rand()/(RAND_MAX+1.0)); sizeToFill--; pMd5Data++; } //Now do md5 on bytes after sizeof(ts) md5Sum=[md5Data md5Digest]; [data appendData:md5Sum]; sessionID=[data hexadecimalRepresentation]; return sessionID; } //-------------------------------------------------------------------- -(void)encodeWithCoder:(NSCoder*)coder { [self notImplemented: _cmd]; //TODOFN /* [coder_ encodeObject:sessionID]; [coder_ encodeObject:languages]; [coder_ encodeValueOfObjCType: @encode(NSTimeInterval) at: &timeOut]; [coder_ encodeObject:variables]; [coder_ encodeObject:pageCache]; */ } //-------------------------------------------------------------------- -(id)initWithCoder: (NSCoder*)coder { [self notImplemented: _cmd]; //TODOFN /* [coder_ decodeValueOfObjCType: @encode(id) at:&sessionID]; [coder_ decodeValueOfObjCType: @encode(id) at:&languages]; [coder_ decodeValueOfObjCType: @encode(NSTimeInterval) at: &timeOut]; [coder_ decodeValueOfObjCType: @encode(id) at:&variables]; [coder_ decodeValueOfObjCType: @encode(id) at:&pageCache]; */ return self; } //-------------------------------------------------------------------- -(void)dealloc { DESTROY(_sessionID); DESTROY(_contextArrayStack); DESTROY(_contextRecords); DESTROY(_editingContext); DESTROY(_languages); DESTROY(_componentState); DESTROY(_birthDate); DESTROY(_statistics); DESTROY(_formattedStatistics); _currentContext=nil; DESTROY(_permanentPageCache); DESTROY(_permanentContextIDArray); DESTROY(_domainForIDCookies); // we are NOT destroying the _autoreleasePool here. // this is triggered externally. [super dealloc]; } //-------------------------------------------------------------------- -(NSString*)description { NSString* dscr=nil; dscr=[NSString stringWithFormat:@"<%s %p - sessionID=%@ autoreleasePool=%p timeOut=%f contextArrayStack=%@", object_getClassName(self), (void*)self, _sessionID, (void*)_autoreleasePool, _timeOut, _contextArrayStack]; dscr=[dscr stringByAppendingFormat:@" isTerminating=%s isDistributionEnabled=%s storesIDsInCookies=%s storesIDsInURLs=%s hasSessionLockedEditingContext=%s>", _isTerminating ? "YES" : "NO", _isDistributionEnabled ? "YES" : "NO", _storesIDsInCookies ? "YES" : "NO", _storesIDsInURLs ? "YES" : "NO", _hasSessionLockedEditingContext ? "YES" : "NO"]; return dscr; } //-------------------------------------------------------------------- // sessionID -(NSString*)sessionID { return _sessionID; } //-------------------------------------------------------------------- // sessionID -(void)setSessionID:(NSString*)sessionID { ASSIGN(_sessionID,sessionID); } //-------------------------------------------------------------------- -(NSString*)domainForIDCookies { //OK if (!_domainForIDCookies) { GSWContext* context=nil; GSWRequest* request=nil; NSString* applicationName=nil; NSString* adaptorPrefix=nil; [[GSWApplication application]lock]; context=[self context]; request=[context request]; applicationName=[request applicationName]; adaptorPrefix=[request adaptorPrefix]; [[GSWApplication application]unlock]; ASSIGN(_domainForIDCookies, ([NSString stringWithFormat:@"%@/%@.%@", adaptorPrefix, applicationName, GSWApplicationSuffix[GSWebNamingConv]])); } return _domainForIDCookies; } //-------------------------------------------------------------------- -(BOOL)storesIDsInURLs { return _storesIDsInURLs; } //-------------------------------------------------------------------- -(void)setStoresIDsInURLs:(BOOL)flag { if (flag!=_storesIDsInURLs) _storesIDsInURLs=flag; } //-------------------------------------------------------------------- -(NSDate*)expirationDateForIDCookies { NSDate* expirationDateForIDCookies=nil; expirationDateForIDCookies=[NSDate dateWithTimeIntervalSinceNow:_timeOut]; return expirationDateForIDCookies; } //-------------------------------------------------------------------- -(BOOL)storesIDsInCookies { return _storesIDsInCookies; } //-------------------------------------------------------------------- -(void)setStoresIDsInCookies:(BOOL)flag { if (flag!=_storesIDsInCookies) { _storesIDsInCookies=flag; [_currentContext _synchronizeForDistribution]; } } //-------------------------------------------------------------------- /** Returns NO if URLs contains application number so requests are directed to the specific application instance. Resturns YES if URLs doesn't contain application number so requests can be directed to any instance (load balancing) Default value is NO **/ -(BOOL)isDistributionEnabled { return _isDistributionEnabled; } //-------------------------------------------------------------------- /** Enables or disables application instance number in URLs. If flag is NO, URLs contains application number so requests are directed to the specific application instance. If flag is YES, URLs doesn't contain application number so requests can be directed to any instance (load balancing) **/ -(void)setDistributionEnabled:(BOOL)flag { if (flag!=_isDistributionEnabled) { _isDistributionEnabled=flag; [_currentContext _synchronizeForDistribution]; } } //-------------------------------------------------------------------- -(void)_setContextArrayStack:(NSArray*)contextArrayStack { DESTROY(_contextArrayStack); _contextArrayStack=[contextArrayStack mutableCopy]; } //-------------------------------------------------------------------- -(void)_setContextRecords:(NSDictionary*)contextRecords { DESTROY(_contextRecords); _contextRecords=[contextRecords mutableCopy]; } //-------------------------------------------------------------------- -(void)_setComponentState:(NSDictionary*)componentState { DESTROY(_componentState); _componentState=[componentState mutableCopy]; } //-------------------------------------------------------------------- -(void)_setStatistics:(NSArray*)statistics { DESTROY(_statistics); _statistics=[statistics mutableCopy]; } //-------------------------------------------------------------------- -(void)_setFormattedStatistics:(NSString*)formattedStatistics { DESTROY(_formattedStatistics); _formattedStatistics=[formattedStatistics mutableCopy]; } //-------------------------------------------------------------------- -(void)_setContextCounter:(int)contextCounter { _contextCounter = contextCounter; } //-------------------------------------------------------------------- -(void)_setRequestCounter:(int)requestCounter { _requestCounter = requestCounter; } //-------------------------------------------------------------------- -(id)_initWithSessionID:(NSString*)aSessionID { //OK GSWApplication* application=nil; GSWStatisticsStore* statisticsStore=nil; statisticsStore=[GSWApp statisticsStore]; [statisticsStore _applicationCreatedSession:self]; ASSIGNCOPY(_sessionID,aSessionID); application=[GSWApplication application]; //applic statisticsStore //applic _activeSessionsCount [self _setBirthDate:[NSDate date]]; ASSIGN(_statistics,[NSMutableArray array]); _storesIDsInURLs=YES; [application _finishInitializingSession:self]; return self; } //-------------------------------------------------------------------- // isTerminating //-------------------------------------------------------------------- -(BOOL)isTerminating { return _isTerminating; } //-------------------------------------------------------------------- // terminate -(void)terminate { if (!_isTerminating) // don't do it multiple times ! { GSWApplication* application=[GSWApplication application]; NSString* sessionID=[self sessionID]; _isTerminating=YES; [[NSNotificationCenter defaultCenter] postNotificationName:GSWNotification__SessionDidTimeOutNotification[GSWebNamingConv] object:sessionID]; [application _discountTerminatedSession]; [[application statisticsStore] _sessionTerminating:self]; if (_editingContext) { if (_hasSessionLockedEditingContext) { [_editingContext unlock]; _hasSessionLockedEditingContext = NO; } DESTROY(_editingContext); } /* [self setTimeOut:(NSTimeInterval) 1]; // forces to call removeSessionWithID in GSWServerSessionStore to dealloc it //TODO: VERIFY [self setTimeOut:(NSTimeInterval) 1]; // forces to call removeSessionWithID in GSWServerSessionStore to dealloc it */ } } //-------------------------------------------------------------------- -(void)_terminateByTimeout { _wasTimedOut = YES; [self terminate]; } // componentDefinition _notifyObserversForDyingComponent:Main component //.... //-------------------------------------------------------------------- // timeOut -(NSTimeInterval)timeOut { return _timeOut; } //-------------------------------------------------------------------- // setTimeOut: -(void)setTimeOut:(NSTimeInterval)timeOut { if (timeOut==0) _timeOut=[[NSDate distantFuture]timeIntervalSinceDate:_birthDate]; else _timeOut=timeOut; } //-------------------------------------------------------------------- -(void)debugWithFormat:(NSString*)aFormat,... { va_list ap; va_start(ap,aFormat); [GSWApp debugWithFormat:aFormat arguments:ap]; va_end(ap); } //-------------------------------------------------------------------- -(void)_debugWithString:(NSString*)string { [GSWApp debugWithString:string]; } //-------------------------------------------------------------------- -(void)savePage:(GSWComponent*)page { GSWTransactionRecord * transactionrec = nil; BOOL pageChanged = [_currentContext _pageChanged]; BOOL createArray = NO; NSString * ctxID = [_currentContext contextID]; NSMutableArray *ctxArray = nil; int i=0; int k=0; NSString * currentCtx = nil; if (_contextArrayStack == nil) { ASSIGN(_contextArrayStack, [NSMutableArray arrayWithCapacity:64]); ASSIGN(_contextRecords, [NSMutableDictionary dictionaryWithCapacity:64]); } else { ctxArray = [_contextArrayStack lastObject]; [self _rearrangeContextArrayStackForContextID:ctxID]; } if (pageChanged || (ctxArray == nil)) { createArray = YES; } else { if ([ctxArray count] > 0) { GSWTransactionRecord * transactionrecord = [_contextRecords objectForKey:[ctxArray lastObject]]; GSWComponent * otherComponent = nil; if (transactionrecord != nil) { otherComponent = [transactionrecord responsePage]; } if (otherComponent == page) { createArray = NO; } else { createArray = YES; } } } if (createArray) { ctxArray = [NSMutableArray arrayWithCapacity:64]; [_contextArrayStack addObject:ctxArray]; } if (([_currentContext response] != nil) && ([[_currentContext response] _isClientCachingDisabled])) { transactionrec = [GSWTransactionRecord transactionRecordWithResponsePage:page context:_currentContext]; } else { transactionrec = [GSWTransactionRecord transactionRecordWithResponsePage:page context:nil]; } [_contextRecords setObject: transactionrec forKey: ctxID]; [ctxArray addObject:ctxID]; for (i = [GSWApp pageCacheSize]; ([ctxArray count] > i); [ctxArray removeObjectAtIndex:0]) { currentCtx = [ctxArray objectAtIndex:0]; [_contextRecords removeObjectForKey:currentCtx]; } if ([ctxArray count] == 0) { [_contextArrayStack removeObjectAtIndex:([_contextArrayStack count] - 1)]; } for (; ([_contextArrayStack count] > i); [_contextArrayStack removeObjectAtIndex:0]) { NSMutableArray * stackArray = [_contextArrayStack objectAtIndex:0]; int stackArraySize = [stackArray count]; for (k = 0; k < stackArraySize; k++){ [_contextRecords removeObjectForKey:[stackArray objectAtIndex:0]]; } } } //-------------------------------------------------------------------- -(GSWComponent*)restorePageForContextID:(NSString*)aContextID { //OK GSWComponent* page=nil; NSArray* contextArray=nil; GSWTransactionRecord* transactionRecord=nil; NSAssert(aContextID,@"No contextID"); NSAssert([aContextID length]>0,@"contextID empty"); transactionRecord=[_contextRecords objectForKey:aContextID]; if (transactionRecord) { page=[transactionRecord responsePage]; } if (page) // will put it at the end of the stack { NSUInteger stackIndex=0; NSUInteger contextArrayIndex=0; contextArray=[self _contextArrayForContextID:aContextID stackIndex:&stackIndex contextArrayIndex:&contextArrayIndex]; if (contextArray) { if (stackIndex!=([_contextArrayStack count]-1)) { [_contextArrayStack addObject:contextArray]; //add before removing to avoid release [_contextArrayStack removeObjectAtIndex:stackIndex]; } } } if ([_permanentPageCache objectForKey:aContextID]) page=[self _permanentPageWithContextID:aContextID]; [page _awakeInContext:_currentContext]; return page; } //-------------------------------------------------------------------- //NDFN -(NSUInteger)permanentPageCacheSize { return [GSWApp permanentPageCacheSize]; } //-------------------------------------------------------------------- -(void)savePageInPermanentCache:(GSWComponent*)page { GSWContext* context=nil; NSMutableDictionary* permanentPageCache=nil; NSUInteger permanentPageCacheSize=0; NSString* contextID=nil; context=[self context]; permanentPageCache=[self _permanentPageCache]; permanentPageCacheSize=[self permanentPageCacheSize]; // first we'll remove excessive cached pages. while([_permanentContextIDArray count]>0 && [_permanentContextIDArray count]>=permanentPageCacheSize) { id deletePage=nil; NSString* deleteContextID=nil; deleteContextID=[_permanentContextIDArray objectAtIndex:0]; RETAIN(deleteContextID); // We'll remove it from array [_permanentContextIDArray removeObjectAtIndex:0]; deletePage=[_contextRecords objectForKey:deleteContextID]; [permanentPageCache removeObjectForKey:deleteContextID]; RELEASE(deleteContextID); } contextID=[context contextID]; NSAssert(contextID,@"No contextID"); if ([_permanentContextIDArray containsObject:contextID]) { [_permanentContextIDArray removeObject:contextID]; if (![permanentPageCache objectForKey:contextID]) { } } else if ([permanentPageCache objectForKey:contextID]) { } [permanentPageCache setObject:page forKey:contextID]; [_permanentContextIDArray addObject:contextID]; /* { int i=0; id anObject=nil; id anotherContextID=nil; for(i=0;i<[_permanentContextIDArray count];i++) { anotherContextID=[_permanentContextIDArray objectAtIndex:i]; anObject=[permanentPageCache objectForKey:anotherContextID]; [GSWApplication statusLogWithFormat:@"%d contextID=%@ page class=%@",i,anotherContextID,[anObject class]]; } } */ if ([_permanentContextIDArray count]!=[permanentPageCache count]) { } } //-------------------------------------------------------------------- -(void)clearCookieFromResponse:(GSWResponse*)aResponse { NSString* domainForIDCookies=nil; NSString* sessionID=nil; NSDate* anExpireDate=nil; GSWCookie* sessionIDCookie=nil; GSWCookie* instanceIDCookie=nil; domainForIDCookies=[self domainForIDCookies]; sessionID=[self sessionID]; anExpireDate=[NSDate date]; // Expire now sessionIDCookie=[GSWCookie cookieWithName:GSWKey_SessionID[GSWebNamingConv] value:sessionID path:domainForIDCookies domain:nil expires:anExpireDate isSecure:NO]; NSDebugMLLog(@"sessions",@"sessionIDCookie=%@",sessionIDCookie); [aResponse addCookie:sessionIDCookie]; instanceIDCookie=[GSWCookie cookieWithName:GSWKey_InstanceID[GSWebNamingConv] value:@"-1" path:domainForIDCookies domain:nil expires:anExpireDate isSecure:NO]; NSDebugMLLog(@"sessions",@"instanceIDCookie=%@",instanceIDCookie); [aResponse addCookie:instanceIDCookie]; } //-------------------------------------------------------------------- -(void)appendCookieToResponse:(GSWResponse*)aResponse { if ([self storesIDsInCookies]) { NSString* domainForIDCookies=[self domainForIDCookies]; NSString* sessionID=nil; int instance=-1; NSDate* anExpireDate=nil; if ([self isTerminating]) { sessionID=@""; anExpireDate=[NSDate date]; //expire now ! } else { sessionID=[self sessionID]; anExpireDate=[self expirationDateForIDCookies]; } // SessionID cookie [aResponse addCookie:[GSWCookie cookieWithName:GSWKey_SessionID[GSWebNamingConv] value:sessionID path:domainForIDCookies domain:nil expires:anExpireDate isSecure:NO]]; // Instance Cookie // No Instance if distribution enabled or this session is terminating if ([self isDistributionEnabled] || [self isTerminating]) { instance=-1; anExpireDate=[NSDate date]; //expire now ! } else { GSWRequest* request = [_currentContext request]; if (request) instance=[request applicationNumber]; // use the request instance number else instance=-1; } [aResponse addCookie:[GSWCookie cookieWithName:GSWKey_InstanceID[GSWebNamingConv] value:GSWIntToNSString(instance) path:domainForIDCookies domain:nil expires:anExpireDate isSecure:NO]]; } } -(void)_releaseAutoreleasePool { DESTROY(_autoreleasePool); } //-------------------------------------------------------------------- -(void)_createAutoreleasePool { if (!_autoreleasePool) { _autoreleasePool=[NSAutoreleasePool new]; } else { [NSException raise:NSInternalInconsistencyException format:@"%s - Can't create an autorelease pool when one already exists.", __PRETTY_FUNCTION__]; } } //-------------------------------------------------------------------- -(GSWComponent*)_permanentPageWithContextID:(NSString*)aContextID { GSWComponent* page=nil; page=[_permanentPageCache objectForKey:aContextID]; return page; } //-------------------------------------------------------------------- -(NSMutableDictionary*)_permanentPageCache { if (!_permanentPageCache) _permanentPageCache=[NSMutableDictionary new]; if (!_permanentContextIDArray) _permanentContextIDArray=[NSMutableArray new]; return _permanentPageCache; } //-------------------------------------------------------------------- -(NSString*)_contextIDMatchingContextID:(NSString*)aContextID requestSenderID:(NSString*)aSenderID { NSAssert(NO,@"Deprecated. use _contextIDMatchingIDsInContext:"); return nil; } //-------------------------------------------------------------------- -(NSString*)_contextIDMatchingIDsInContext:(GSWContext*)aContext { NSString* contextID=nil; NSString* requestContextID=[aContext _requestContextID]; if (_contextRecords && requestContextID) { NSArray* contextIDs = [_contextRecords allKeys]; int count = [contextIDs count]; int i=0; IMP oaiIMP=NULL; for(i=0;!contextID && i0) { [self savePageInPermanentCache:component]; } else { if ([self pageCacheSize]>0) [self savePage:component]; } } } } } //-------------------------------------------------------------------- -(int)_requestCounter { //OK return _requestCounter; } //-------------------------------------------------------------------- -(void)_contextDidIncrementContextID { _contextCounter++; } //-------------------------------------------------------------------- -(int)_contextCounter { //OK return _contextCounter; } //-------------------------------------------------------------------- -(void)_setContext:(GSWContext*)aContext { if (aContext!=_currentContext) _currentContext=aContext; } //-------------------------------------------------------------------- -(void)sleepInContext:(GSWContext*)aContext { [self sleep]; if (_hasSessionLockedEditingContext) { if (_editingContext) { [_editingContext unlock]; } _hasSessionLockedEditingContext = NO; } if (([self isTerminating]) && (_editingContext != nil)) { // TODO: discuss with Mr. Ayers if we need that. // [_editingContext dispose]; DESTROY(_editingContext); } [self _setContext:nil]; } //-------------------------------------------------------------------- -(void)awakeInContext:(GSWContext*)aContext { [self _setContext:aContext]; _contextCounter++; _requestCounter++; if (_editingContext && !_hasSessionLockedEditingContext && [GSWApplication _lockDefaultEditingContext]) { [_editingContext lock]; _hasSessionLockedEditingContext=YES; } [self awake]; } //-------------------------------------------------------------------- -(void)setLanguages:(NSArray*)someLanguages { ASSIGN(_languages,someLanguages); } //-------------------------------------------------------------------- /** GSWeb specific Insert language language at the begining of session languages array **/ -(void)insertLanguage:(NSString*)language { NSArray* languages=nil; if ([language length]>0) { languages=[self languages]; if ([languages count]>0) { if (![language isEqualToString:[languages objectAtIndex:0]]) { NSMutableArray* mutableLanguages=[[languages mutableCopy]autorelease]; [mutableLanguages removeObject:language];//Remove language if it exists in languages [mutableLanguages insertObject:language atIndex:0]; [self setLanguages:mutableLanguages]; } } else [self setLanguages:[NSArray arrayWithObject:language]]; } } //-------------------------------------------------------------------- /** GSWeb specific Add language language at the end of session languages array if language is not present **/ -(void)addLanguage:(NSString*)language { NSArray* languages=nil; if ([language length]>0) { languages=[self languages]; if ([languages count]>0) { if (![languages containsObject:language]) [self setLanguages:[languages arrayByAddingObject:language]]; } else [self setLanguages:[NSArray arrayWithObject:language]]; } } //-------------------------------------------------------------------- /** GSWeb specific Returns first element of languages or nil if languages is empty **/ -(NSString*)firstLanguage { NSArray* languages=nil; NSString* firstLanguage=nil; languages=[self languages]; if ([languages count]>0) firstLanguage=[languages objectAtIndex:0]; return firstLanguage; } //-------------------------------------------------------------------- -(NSArray*)languages { NSDebugMLLog(@"sessions",@"_languages=%@",_languages); if (!_languages) { GSWContext* aContext=[self context]; GSWRequest* request=[aContext request]; NSArray* languages=[request browserLanguages]; [self setLanguages:languages]; NSDebugMLLog(@"sessions",@"_languages=%@",_languages); } return _languages; } //-------------------------------------------------------------------- -(NSArray*)_languages { return _languages; } //-------------------------------------------------------------------- // objectForKey: -(id)objectForKey:(NSString*)key { id object=nil; object=[_componentState objectForKey:key]; NSDebugMLLog(@"sessions",@"key=%@ object=%@",key,object); return object; } //-------------------------------------------------------------------- // setObject:forKey: -(void)setObject:(id)object forKey:(NSString*)key { NSAssert(object,@"No object"); NSAssert(key,@"No key"); if (!_componentState) _componentState=[NSMutableDictionary new]; NSDebugMLLog(@"sessions",@"key=%@ object=%@",key,object); [_componentState setObject:object forKey:key]; } //-------------------------------------------------------------------- -(void)removeObjectForKey:(NSString*)key { NSDebugMLLog(@"sessions",@"key=%@",key); [_componentState removeObjectForKey:key]; } //-------------------------------------------------------------------- //NDFN -(NSMutableDictionary*)componentState { return _componentState; } //-------------------------------------------------------------------- -(EOEditingContext*)defaultEditingContext { if(!_editingContext) { ASSIGN(_editingContext,[[[NSClassFromString(@"EOEditingContext") alloc] init] autorelease]); [_editingContext setLevelsOfUndo:[GSWApplication defaultUndoStackLimit]]; if ([GSWApplication _lockDefaultEditingContext]) { [_editingContext lock]; _hasSessionLockedEditingContext=YES; } } return _editingContext; } //-------------------------------------------------------------------- -(void)setDefaultEditingContext:(EOEditingContext*)editingContext { if (_editingContext) { // We can't set the editing context if one has already been created [NSException raise:NSInvalidArgumentException format:@"%s Can't set a defautEditingContext when one already exists", object_getClassName(self)]; } else { ASSIGN(_editingContext,editingContext); if ([GSWApplication _lockDefaultEditingContext]) { [_editingContext lock]; _hasSessionLockedEditingContext=YES; } } } //-------------------------------------------------------------------- -(GSWContext*)context { return _currentContext; } //-------------------------------------------------------------------- // awake -(void)awake { DESTROY(_domainForIDCookies); } //-------------------------------------------------------------------- // sleep -(void)sleep { // We destroy domainForIDCookies because applictaion name may // change between pages DESTROY(_domainForIDCookies); } //-------------------------------------------------------------------- // takeValuesFromRequest:inContext: -(void)takeValuesFromRequest:(GSWRequest*)aRequest inContext:(GSWContext*)aContext { GSWElement* pageElement=nil; GSWComponent* pageComponent=nil; pageElement=[aContext _pageElement]; pageComponent=[aContext _pageComponent]; #ifndef NDEBUG [aContext addDocStructureStep:@"Take Values From Request"]; #endif [aContext _setCurrentComponent:pageComponent]; [pageElement takeValuesFromRequest:aRequest inContext:aContext]; [aContext _setCurrentComponent:nil]; } //-------------------------------------------------------------------- // invokeActionForRequest:inContext: -(id ) invokeActionForRequest:(GSWRequest*)aRequest inContext:(GSWContext*)aContext { id results = nil; GSWElement* pageElement=nil; GSWComponent* pageComponent=nil; NS_DURING pageElement = [aContext _pageElement]; pageComponent = [aContext _pageComponent]; [aContext _setCurrentComponent:pageComponent]; results = [pageElement invokeActionForRequest:aRequest inContext:aContext]; [aContext _setCurrentComponent:nil]; NS_HANDLER localException=ExceptionByAddingUserInfoObjectFrameInfo(localException, @"In %s", __PRETTY_FUNCTION__); [localException raise]; NS_ENDHANDLER; return results; } //-------------------------------------------------------------------- // appendToResponse:inContext: -(void)appendToResponse:(GSWResponse*)aResponse inContext:(GSWContext*)aContext { GSWStatisticsStore* statisticsStore=nil; NSString* logFile=nil; GSWElement* pageElement=nil; GSWComponent* pageComponent=nil; statisticsStore=[[GSWApplication application] statisticsStore]; pageElement=[aContext _pageElement]; pageComponent=[aContext _pageComponent]; #ifndef NDEBUG [aContext addDocStructureStep:@"Append To Response"]; #endif [aContext _setCurrentComponent:pageComponent]; NS_DURING { [pageElement appendToResponse:aResponse inContext:aContext]; } NS_HANDLER { localException=ExceptionByAddingUserInfoObjectFrameInfo(localException, @"In %@ appendToResponse:inContext", [self class]); [localException raise]; } NS_ENDHANDLER; [aContext _setCurrentComponent:nil]; [self appendCookieToResponse:aResponse]; { NSString* descr=nil; [statisticsStore recordStatisticsForResponse:aResponse inContext:aContext]; descr=[statisticsStore descriptionForResponse:aResponse inContext:aContext]; if (descr) { [_statistics addObject:descr]; logFile=[statisticsStore logFile]; if (logFile) { NSString* formattedDescr=[GSWStatisticsStore formatDescription:descr forResponse:aResponse inContext:aContext]; if (formattedDescr) { if (!_formattedStatistics) _formattedStatistics = [NSMutableString new]; [_formattedStatistics appendString:formattedDescr]; } } } } } //-------------------------------------------------------------------- -(NSArray*)statistics { return _statistics; } //-------------------------------------------------------------------- -(BOOL)_allowedToViewStatistics { return _isAllowedToViewStatistics; } //-------------------------------------------------------------------- -(void)_allowToViewStatistics { _isAllowedToViewStatistics=YES; } //-------------------------------------------------------------------- -(void)_setAllowedToViewStatistics:(BOOL)flag { _isAllowedToViewStatistics=flag; } //-------------------------------------------------------------------- -(BOOL)validateStatisticsLogin:(NSString*)login withPassword:(NSString*)password { GSWStatisticsStore* statsStore = [[GSWApplication application]statisticsStore]; if (statsStore) { NSString* statsStorePassword=[statsStore _password]; if ([statsStorePassword isEqual:password]) [self _allowToViewStatistics]; } return [self _allowedToViewStatistics]; } //-------------------------------------------------------------------- -(NSString*)_formattedStatistics { return [NSString stringWithString:_formattedStatistics]; } //-------------------------------------------------------------------- -(NSDate*)_birthDate { return _birthDate; } //-------------------------------------------------------------------- -(void)_setBirthDate:(NSDate*)birthDate { ASSIGN(_birthDate,birthDate); } //-------------------------------------------------------------------- -(BOOL)_allowedToViewEvents { return _isAllowedToViewEvents; } //-------------------------------------------------------------------- -(void)_allowToViewEvents { _isAllowedToViewEvents=YES; } //-------------------------------------------------------------------- -(void)_setAllowedToViewEvents:(BOOL)flag { _isAllowedToViewEvents=flag; } -(void) _clearCookieFromResponse:(GSWResponse*) aResponse { NSString *cookiePath = [self domainForIDCookies]; NSCalendarDate *today = [NSCalendarDate date]; GSWCookie * instanceCookie; GSWCookie * sessionIDCookie; NSCalendarDate *dateInThePast = [today dateByAddingYears:0 months:-1 days:0 hours:0 minutes:0 seconds:0]; sessionIDCookie = [GSWCookie cookieWithName:[GSWApp sessionIdKey] value:_sessionID path:cookiePath domain:nil expires:dateInThePast isSecure:NO]; [aResponse addCookie:sessionIDCookie]; instanceCookie = [GSWCookie cookieWithName:[GSWApp instanceIdKey] value:@"-1" path:cookiePath domain:nil expires:dateInThePast isSecure:NO]; [aResponse addCookie:instanceCookie]; } //-------------------------------------------------------------------- -(BOOL)validateEventsLogin:(NSString*)login withPassword:(NSString*)password { //TODO wait for gdl2 implementation /* NSString* eventCenterPassword=[EOEventCenter password]; if ([eventCenterPassword isEqual:password]) [self _allowToViewEvents]; */ return [self _allowedToViewEvents]; } //-------------------------------------------------------------------- -(GSWApplication*)application { return [GSWApplication application]; } @end