// // $Id$ // // SPFavoritesController.m // sequel-pro // // Created by Stuart Connolly (stuconnolly.com) on November 10, 2010. // Copyright (c) 2010 Stuart Connolly. All rights reserved. // // Permission is hereby granted, free of charge, to any person // obtaining a copy of this software and associated documentation // files (the "Software"), to deal in the Software without // restriction, including without limitation the rights to use, // copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following // conditions: // // The above copyright notice and this permission notice shall be // included in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR // OTHER DEALINGS IN THE SOFTWARE. // // More info at <http://code.google.com/p/sequel-pro/> #import "SPFavoritesController.h" #import "SPFavoriteNode.h" #import "SPTreeNode.h" #import "SPGroupNode.h" #import "SPThreadAdditions.h" #import "pthread.h" static SPFavoritesController *sharedFavoritesController = nil; @interface SPFavoritesController () - (void)_loadFavorites; - (void)_constructFavoritesTree; - (void)_saveFavoritesData:(NSDictionary *)data; - (void)_addNode:(SPTreeNode *)node asChildOfNode:(SPTreeNode *)parent; - (SPTreeNode *)_constructBranchForNodeData:(NSDictionary *)nodeData; @end @implementation SPFavoritesController @synthesize favoritesTree; @synthesize favoritesData; #pragma mark - #pragma mark Initialisation + (id)allocWithZone:(NSZone *)zone { @synchronized(self) { return [[self sharedFavoritesController] retain]; } return nil; } - (id)init { if ((self = [super init])) { favoritesTree = nil; favoritesData = nil; pthread_mutex_init(&writeLock, NULL); pthread_mutex_init(&favoritesLock, NULL); [self _loadFavorites]; [self _constructFavoritesTree]; } return self; } #pragma mark - /** * Returns the shared favorites controller. * * @return The shared controller instance. */ + (SPFavoritesController *)sharedFavoritesController { @synchronized(self) { if (sharedFavoritesController == nil) { sharedFavoritesController = [[super allocWithZone:NULL] init]; } } return sharedFavoritesController; } #pragma mark - #pragma mark Favorites data handling /** * Saves the current favorites dictionary in memory to disk. Note that the current favorites data file is moved * rather than overwritten in the event that we can't write the new file, the original can simply be restored. * This method also does a lot of error checking to ensure we don't lose the user's favorites data. * Saves the data in the background so any UI tasks can stay responsive. */ - (void)saveFavorites { pthread_mutex_lock(&favoritesLock); [NSThread detachNewThreadWithName:@"SPFavoritesController background favorite save task" target:self selector:@selector(_saveFavoritesData:) object:[[[favoritesTree childNodes] objectAtIndex:0] dictionaryRepresentation]]; pthread_mutex_unlock(&favoritesLock); } /** * Save the current favorites dictionary in memory to disk, in the foreground, in a blocking manner. */ - (void)saveFavoritesSynchronously { [self _saveFavoritesData:[[[favoritesTree childNodes] objectAtIndex:0] dictionaryRepresentation]]; } /** * Reloads the favorites data from disk with the option to save before doing so. * * @param save Indicates whether the current favorites data in memory should be saved to disk before being * reloaded. Specifying NO effectively discards any changes since the last save operation. */ - (void)reloadFavoritesWithSave:(BOOL)save { if (save) [self saveFavorites]; if (favoritesData) { [self _loadFavorites]; [self _constructFavoritesTree]; } } #pragma mark - #pragma mark Favorites interaction /** * Adds a new group node with the supplied name to the children of the supplied parent node. * * @param name The name of the new group * @param parent * * @return The node instance that was created and added */ - (SPTreeNode *)addGroupNodeWithName:(NSString *)name asChildOfNode:(SPTreeNode *)parent { SPTreeNode *node = [SPTreeNode treeNodeWithRepresentedObject:[SPGroupNode groupNodeWithName:name]]; [node setIsGroup:YES]; [self _addNode:node asChildOfNode:parent]; [[NSNotificationCenter defaultCenter] postNotificationName:SPConnectionFavoritesChangedNotification object:self]; return node; } /** * Adds a new favorite node with the supplied data to the children of the supplied parent node. * * @param data The data for the new favorite * @param * * @return The node instance that was created and added */ - (SPTreeNode *)addFavoriteNodeWithData:(NSMutableDictionary *)data asChildOfNode:(SPTreeNode *)parent { SPTreeNode *node = [SPTreeNode treeNodeWithRepresentedObject:[SPFavoriteNode favoriteNodeWithDictionary:data]]; [self _addNode:node asChildOfNode:parent]; [[NSNotificationCenter defaultCenter] postNotificationName:SPConnectionFavoritesChangedNotification object:self]; return node; } /** * Removes the supplied favorite node by asking the root node to remove it from it's children (i.e. the * entire tree is searched. * * @param The node to be removed */ - (void)removeFavoriteNode:(SPTreeNode *)node { [favoritesTree removeObjectFromChildren:node]; // Save data to disk [self saveFavorites]; [[NSNotificationCenter defaultCenter] postNotificationName:SPConnectionFavoritesChangedNotification object:self]; } #pragma mark - #pragma mark Private API /** * Attempts to load the users connection favorites from ~/Library/Application Support/Sequel Pro/Data/Favorites.plist * If the 'Data' directory doesn't already exist it will be created, as well as an empty favorites plist. */ - (void)_loadFavorites { pthread_mutex_lock(&favoritesLock); NSError *error = nil; NSFileManager *fileManager = [NSFileManager defaultManager]; if (favoritesData) [favoritesData release], favoritesData = nil; NSString *dataPath = [fileManager applicationSupportDirectoryForSubDirectory:SPDataSupportFolder error:&error]; if (error) { NSLog(@"Error retrieving data directory path: %@", [error localizedDescription]); pthread_mutex_unlock(&favoritesLock); return; } NSString *favoritesFile = [dataPath stringByAppendingPathComponent:SPFavoritesDataFile]; // If the favorites data file already exists use it, otherwise create an empty one if ([fileManager fileExistsAtPath:favoritesFile]) { favoritesData = [[NSDictionary alloc] initWithContentsOfFile:favoritesFile]; } else { NSMutableDictionary *newFavorites = [NSMutableDictionary dictionaryWithObject:[NSMutableDictionary dictionaryWithObjectsAndKeys:NSLocalizedString(@"Favorites", @"favorites label"), SPFavoritesGroupNameKey, [NSArray array], SPFavoriteChildrenKey, nil] forKey:SPFavoritesRootKey]; error = nil; NSString *errorString = nil; NSData *plistData = [NSPropertyListSerialization dataFromPropertyList:newFavorites format:NSPropertyListXMLFormat_v1_0 errorDescription:&errorString]; if (plistData) { [plistData writeToFile:favoritesFile options:NSAtomicWrite error:&error]; if (error) { NSLog(@"Error writing default favorites data: %@", [error localizedDescription]); } } else if (errorString) { NSLog(@"Error converting default favorites data to plist format: %@", errorString); [errorString release]; pthread_mutex_unlock(&favoritesLock); return; } favoritesData = newFavorites; } pthread_mutex_unlock(&favoritesLock); } /** * Constructs the favorites tree by initialising an instance of SPFavoriteNode for every favorite and group. */ - (void)_constructFavoritesTree { pthread_mutex_lock(&favoritesLock); if (!favoritesData) { pthread_mutex_unlock(&favoritesLock); return; } NSDictionary *root = [favoritesData objectForKey:SPFavoritesRootKey]; SPGroupNode *rootGroupNode = [[SPGroupNode alloc] init]; SPGroupNode *favoritesGroupNode = [[SPGroupNode alloc] initWithName:[[root objectForKey:SPFavoritesGroupNameKey] uppercaseString]]; [favoritesGroupNode setNodeIsExpanded:[[root objectForKey:SPFavoritesGroupIsExpandedKey] boolValue]]; SPTreeNode *rootNode = [[SPTreeNode alloc] initWithRepresentedObject:rootGroupNode]; SPTreeNode *favoritesNode = [[SPTreeNode alloc] initWithRepresentedObject:favoritesGroupNode]; [rootNode setIsGroup:YES]; [favoritesNode setIsGroup:YES]; for (NSDictionary *favorite in [root objectForKey:SPFavoriteChildrenKey]) { SPTreeNode *node = [self _constructBranchForNodeData:favorite]; [[favoritesNode mutableChildNodes] addObject:node]; } [[rootNode mutableChildNodes] addObject:favoritesNode]; [rootGroupNode release]; [favoritesGroupNode release]; [favoritesNode release]; favoritesTree = rootNode; pthread_mutex_unlock(&favoritesLock); } /** * Constructs the tree branch for the supplied favorites data. Note that depending on the contents of the * branch (i.e. does it contain any groups and their depth) this method will recursively call itself. * * @param nodeData The favorites data dictionary * * @return The root node of the branch */ - (SPTreeNode *)_constructBranchForNodeData:(NSDictionary *)nodeData { id node = nil; SPTreeNode *treeNode = nil; if ([nodeData objectForKey:SPFavoritesGroupNameKey] && [nodeData objectForKey:SPFavoriteChildrenKey]) { node = [[SPGroupNode alloc] initWithName:[nodeData objectForKey:SPFavoritesGroupNameKey]]; [node setNodeIsExpanded:[[nodeData objectForKey:SPFavoritesGroupIsExpandedKey] boolValue]]; treeNode = [[SPTreeNode alloc] initWithRepresentedObject:node]; [node release]; [treeNode setIsGroup:YES]; for (NSDictionary *favorite in [nodeData objectForKey:SPFavoriteChildrenKey]) { SPTreeNode *innerNode = [self _constructBranchForNodeData:favorite]; [[treeNode mutableChildNodes] addObject:innerNode]; } } else { node = [[SPFavoriteNode alloc] initWithDictionary:nodeData]; treeNode = [[SPTreeNode alloc] initWithRepresentedObject:node]; [node release]; } return [treeNode autorelease]; } /** * Saves the supplied favorites data to disk on a background thread. * * @param data The raw plist data (serialized NSDictionary) to be saved */ - (void)_saveFavoritesData:(NSDictionary *)data { NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]; pthread_mutex_lock(&writeLock); if (!favoritesTree) { pthread_mutex_unlock(&writeLock); return; } NSError *error = nil; NSString *errorString = nil; // Before starting the file actions, attempt to create a dictionary // from the current favourites tree and convert it to a dictionary representation // to create the plist data. This is done before file changes as it can sometimes // be terminated during shutdown. NSDictionary *dictionary = [NSDictionary dictionaryWithObject:data forKey:SPFavoritesRootKey]; NSData *plistData = [NSPropertyListSerialization dataFromPropertyList:dictionary format:NSPropertyListXMLFormat_v1_0 errorDescription:&errorString]; if (errorString) { NSLog(@"Error converting favorites data to plist format: %@", errorString); [errorString release]; } NSFileManager *fileManager = [NSFileManager defaultManager]; NSString *dataPath = [fileManager applicationSupportDirectoryForSubDirectory:SPDataSupportFolder error:&error]; if (error) { NSLog(@"Error retrieving data directory path: %@", [error localizedDescription]); pthread_mutex_unlock(&writeLock); return; } NSString *favoritesFile = [dataPath stringByAppendingPathComponent:SPFavoritesDataFile]; NSString *favoritesBackupFile = [dataPath stringByAppendingPathComponent:[NSString stringWithNewUUID]]; // If the favorites data file already exists, attempt to move it to keep as a backup if ([fileManager fileExistsAtPath:favoritesFile]) { [fileManager moveItemAtPath:favoritesFile toPath:favoritesBackupFile error:&error]; } if (error) { NSLog(@"Unable to backup (move) existing favorites data file during save. Deleting instead: %@", [error localizedDescription]); error = nil; // We can't move it so try and delete it if (![fileManager removeItemAtPath:favoritesFile error:&error] && error) { NSLog(@"Unable to delete existing favorites data file during save. Something is wrong, permissions perhaps: %@", [error localizedDescription]); pthread_mutex_unlock(&writeLock); return; } } // Write the converted data to the favourites file [plistData writeToFile:favoritesFile options:NSAtomicWrite error:&error]; if (error) { NSLog(@"Error writing favorites data. Restoring backup if available: %@", [error localizedDescription]); // Restore the original data file error = nil; [fileManager moveItemAtPath:favoritesBackupFile toPath:favoritesFile error:&error]; if (error) { NSLog(@"Could not restore backup; favorites.plist left renamed as %@ due to error (%@)", favoritesBackupFile, [error localizedDescription]); } } else { // Remove the original backup [fileManager removeItemAtPath:favoritesBackupFile error:NULL]; } pthread_mutex_unlock(&writeLock); [pool release]; } /** * Adds the supplied node to the children of the supplied parent and saves the tree to disk. * * @param node The node to be added * @param asChild */ - (void)_addNode:(SPTreeNode *)node asChildOfNode:(SPTreeNode *)parent { if (parent) { [[parent mutableChildNodes] addObject:node]; } else { [[[[favoritesTree mutableChildNodes] objectAtIndex:0] mutableChildNodes] addObject:node]; } [self saveFavorites]; } #pragma mark - - (void)dealloc { if (favoritesTree) [favoritesTree release], favoritesTree = nil; if (favoritesData) [favoritesData release], favoritesData = nil; pthread_mutex_destroy(&writeLock); pthread_mutex_destroy(&favoritesLock); [super dealloc]; } @end