Google Analytics SDK 创建 SQLite 错误

Google Analytics SDK creates SQLite errors

更新

2015 年 8 月 19 日 - Bug 现在似乎已在 3.13 更新中修复,尽管他们在更新日志中列出的唯一内容是 "Fixed an issue which caused crashes when using setCampaignParametersFromUrl".随你便吧。

2015 年 6 月 8 日 - 仍然遇到这个问题。如果我禁用自动发送事件 ([GAI sharedInstance].dispatchInterval = -1;),那么我仍然会收到错误消息。因此,我假设问题在于将事件插入 Google Analytics SQLite 数据库,不知何故我自己的数据库语句正在变成 void.

2015 年 6 月 10 日 - 仍然遇到崩溃。尝试删除我的控制器扩展 GAITrackedViewController 并手动发送 createScreenView 轨道,但崩溃频率没有变化。

2015 年 6 月 25 日 - 仍然遇到崩溃。


简介

我已将 Google Analytics SDK 3.12 添加到我的 iPhone 应用程序,一切都按预期工作 - 我 运行 该应用程序并且可以看到我设置的所有点击和事件通过 Web 界面。

我正在 didFinishLaunchingWithOptions 顶部的 AppDelegate 中初始化 SDK,如下所示:

[[GAI sharedInstance] trackerWithTrackingId:GOOGLE_ANALYTICS_ID];

问题

但是,当我尝试自己使用 SQLite 时,我发现 运行ning Google Analytics 会产生错误。它们可能表现为严重错误,例如:

它们还会导致我自己的 SQLite 查询失败,例如:

if (! sqlite3_prepare_v2(_db, [sql UTF8String], -1, &_statement, NULL) == SQLITE_OK) {`
    // ..
    // ..

    if (sqlite3_step(_statement) == SQLITE_ROW) {

将随机产生以下错误:

sqlite3_prepare_v2 EXC_BAD_ACCESS (code=1, address=0x6800000000)

如果我注释掉 SDK 初始化,那么一切都会恢复到令人难以置信的稳定状态。再次取消注释,它会在一分钟内使应用程序崩溃。


抢先答题

  1. 运行在 iPhone 6 运行ning 8.3 (12F70).

  2. 已尝试卸载并重新安装应用程序。

  3. 我已经添加了 Google Analytics 工作的所有先决条件;所有 .m 文件到库,libGoogleAnalyticsServices.a 文件,还有 Linked Frameworks and Libraries.

  4. 我也有 Crashlytics,但已尝试从代码 ([Fabric with:@[CrashlyticsKit]];) 中将其注释掉并从 Linked Frameworks and Libraries 中删除其库,结果完全相同。


代码

正在设置 class

// In didFinishLaunchingWithOptions
[Db setup];
[Db connect];

访问 class

Db * db = [[Db alloc] init];

if ([db prepare:@"SELECT * FROM `table` WHERE `id` = ?" withBindings:@[@"123"]]) {
    while ([db stepThrough]) {
        // ..
    }
}

[db finalise];

class

(错误之处已用注释指出)

@implementation Db

static sqlite3  * _db;
static NSString * _dbPath;

#pragma mark - Setup

+ (BOOL)setup {
    NSString      * sqlBundlePath   = [[NSBundle mainBundle] pathForResource:@"db" ofType:@"sqlite"];
    NSString      * documentsFolder = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)[0];
    NSString      * sqlDocumentPath = [[documentsFolder stringByAppendingPathComponent:@"db"] stringByAppendingPathExtension:@"sqlite"];
    NSFileManager * fileManager     = [NSFileManager defaultManager];

    if (! [fileManager fileExistsAtPath:sqlDocumentPath]) {
        NSError * error;
        BOOL success = [fileManager copyItemAtPath:sqlBundlePath toPath:sqlDocumentPath error:&error];

        if (! success) {
            return NO;
        }
    }

    _dbPath = sqlDocumentPath;
    return YES;
}

+ (BOOL)connect {
    sqlite3_config(SQLITE_CONFIG_SERIALIZED);
    return sqlite3_open([_dbPath UTF8String], &_db);
}

#pragma mark - Querying

- (BOOL)prepare:(NSString *)sql withBindings:(NSArray *)bindings {
    // ERROR CAN OCCUR ON THE FOLLOWING LINE
    if (! sqlite3_prepare_v2(_db, [sql UTF8String], -1, &_statement, NULL) == SQLITE_OK) {
        NSLog(@"Error whilst preparing query: %s", sqlite3_errmsg(_db));
        sqlite3_finalize(_statement);
        return NO;
    }

    for (int i = 0; i < [bindings count]; i++) {
        sqlite3_bind_text(_statement,
                          i + 1,
                          [bindings[i] isKindOfClass:[NSNull class]] ? [@"" UTF8String] : [bindings[i] UTF8String],
                          -1,
                          SQLITE_TRANSIENT);
    }

    return YES;
}

- (BOOL)stepThrough {
    // ERROR CAN OCCUR ON THE FOLLOWING LINE
    if (sqlite3_step(_statement) == SQLITE_ROW) {
        return YES;
    }

    sqlite3_finalize(_statement);
    return NO;
}

- (void)finalise {
    sqlite3_finalize(_statement);
}

@end

升级到新版本的 SDK (3.13) 解决了这个问题(至少对我来说是这样),尽管他们的更新日志没有特别提到它。