iOS sqlcipher fmdb“文件已加密或不是数据库”

时间:2015-08-16 20:38:41

标签: ios sqlite encryption fmdb sqlcipher

这个tutorial可以正常使用以下代码。

pod 'FMDB/SQLCipher'

...

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
...

    NSArray *documentPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
    NSString *documentDir = [documentPaths objectAtIndex:0];
    self.databasePath = [documentDir stringByAppendingPathComponent:@"gameDefault.sqlite"];

    [self createAndCheckDatabase];

...
}

...

-(void) createAndCheckDatabase
{
    BOOL success;

    NSFileManager *fileManager = [NSFileManager defaultManager];
    success = [fileManager fileExistsAtPath:self.databasePath];

    if(success) return; // If file exists, dont do anything

    // if file does not exist, make a copy of the one in the Resources folder
    NSString *databasePathFromApp = [[[NSBundle mainBundle] resourcePath] stringByAppendingPathComponent:@"gameDefault.sqlite"]; // File path

    [fileManager copyItemAtPath:databasePathFromApp toPath:self.databasePath error:nil]; // Make a copy of the file in the Documents folder

    // Set the new encrypted database path to be in the Documents Folder
    NSArray *documentPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
    NSString *documentDir = [documentPaths objectAtIndex:0];
    NSString *ecDB = [documentDir stringByAppendingPathComponent:@"encrypted.sqlite"];

    // SQL Query. NOTE THAT DATABASE IS THE FULL PATH NOT ONLY THE NAME
    const char* sqlQ = [[NSString stringWithFormat:@"ATTACH DATABASE '%@' AS encrypted KEY 'secretKey';",ecDB] UTF8String];

    sqlite3 *unencrypted_DB;    
    if (sqlite3_open([self.databasePath UTF8String], &unencrypted_DB) == SQLITE_OK) {

        // Attach empty encrypted database to unencrypted database
        sqlite3_exec(unencrypted_DB, sqlQ, NULL, NULL, NULL);

        // export database
        sqlite3_exec(unencrypted_DB, "SELECT sqlcipher_export('encrypted');", NULL, NULL, NULL);

        // Detach encrypted database
        sqlite3_exec(unencrypted_DB, "DETACH DATABASE encrypted;", NULL, NULL, NULL);

        sqlite3_close(unencrypted_DB);
    } else {
        sqlite3_close(unencrypted_DB);
        NSAssert1(NO, @"Failed to open database with message '%s'.", sqlite3_errmsg(unencrypted_DB));
    }

    self.databasePath = [documentDir stringByAppendingPathComponent:@"encrypted.sqlite"];
}

...

[db setKey:@"secretKey"]

...

   // FMDatabase
    FMDatabase *db = [FMDatabase databaseWithPath:[self getDatabasePath]];
    [db open];
    [db setKey:@"secretKey"];

    // FMDatabaseQueue
    FMDatabaseQueue *queue = [FMDatabaseQueue databaseQueueWithPath:[self getDatabasePath]];

    [queue inDatabase:^(FMDatabase *db) {
        [db setKey:@"secretKey"];
        ...
    }];

...

当我在我的其他项目中使用相同的代码时,它可以很好地加密现有的sqlite数据库。 但是当我尝试通过sql select query访问表时,我收到错误,"文件已加密或不是数据库"。虽然,教程应用程序没有问题我可以在那里更新/插入/删除/选择记录。任何线索?

2 个答案:

答案 0 :(得分:2)

我已经修好了。我认为FMDB& amp; SQLCipher。我用下面的链接解决了这个问题。现在我的数据库文件使用带FMDB的SQLCipher成功加密.... :-D

How to Use FMDB With SQLCipher?

答案 1 :(得分:1)

我已经修好了。如上所述,相同的代码采取不同的行为,因此设置可能有问题。在我的应用程序中,我添加了SQLCipher和FMDB,如下所示:

pod 'SQLCipher', '~> 3.1'
pod 'FMDB', '~> 2.5'

在该教程中,它们被添加为

pod 'FMDB/SQLCipher'

虽然,这个pod命令按照此处显示的终端输出下载相同版本的SQLCipher和FMDB

Updating local specs repositories
Analyzing dependencies
Downloading dependencies
Installing FMDB (2.5)
Installing SQLCipher (3.1.0)
Generating Pods project
Integrating client project

因此,我更改了教程中的pod命令,并修复了错误。但是,我不知道这些命令有何不同。