import 'dart:io'; import 'package:drift_sqflite/drift_sqflite.dart'; import 'package:drift_testcases/tests.dart'; import 'package:flutter/services.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:integration_test/integration_test.dart'; import 'package:path/path.dart'; import 'package:sqflite/sqflite.dart' show DatabaseException, getDatabasesPath; class SqfliteExecutor extends TestExecutor { @override bool get supportsNestedTransactions => true; @override DatabaseConnection createConnection() { return DatabaseConnection( SqfliteQueryExecutor.inDatabaseFolder( path: 'app.db', singleInstance: false, ), ); } @override Future deleteData() async { final folder = await getDatabasesPath(); final file = File(join(folder, 'app.db')); if (await file.exists()) { await file.delete(); } } } Future main() async { IntegrationTestWidgetsFlutterBinding.ensureInitialized(); runAllTests(SqfliteExecutor()); // Additional integration test for flutter: Test loading a database from asset test('can load a database from asset', () async { final databasesPath = await getDatabasesPath(); final dbFile = File(join(databasesPath, 'app_from_asset.db')); if (await dbFile.exists()) { await dbFile.delete(); } var didCallCreator = false; final executor = SqfliteQueryExecutor( path: dbFile.path, singleInstance: true, creator: (file) async { final content = await rootBundle.load('test_asset.db'); await file.writeAsBytes(content.buffer.asUint8List()); didCallCreator = true; }, ); final database = Database.executor(executor); await database.executor.ensureOpen(database); addTearDown(database.close); expect(didCallCreator, isTrue); }); test('can rollback transactions', () async { final executor = SqfliteQueryExecutor(path: ':memory:'); final database = EmptyDb(executor); addTearDown(database.close); final expectedException = Exception('oops'); try { await database .customSelect('select 1') .getSingle(); // ensure database is open/created await database.transaction(() async { await database.customSelect('select 1').watchSingle().first; throw expectedException; }); } catch (e) { expect(e, expectedException); } finally { await database.customSelect('select 1').getSingle().timeout( const Duration(milliseconds: 500), onTimeout: () => fail('deadlock?'), ); } }); test('handles failing commits', () async { final executor = SqfliteQueryExecutor(path: ':memory:'); final database = EmptyDb(executor); addTearDown(database.close); await database.customStatement('PRAGMA foreign_keys = ON;'); await database.customStatement('CREATE TABLE x (foo INTEGER PRIMARY KEY);'); await database.customStatement('CREATE TABLE y (foo INTEGER PRIMARY KEY ' 'REFERENCES x (foo) DEFERRABLE INITIALLY DEFERRED);'); // On Android, the failing commit of the transaction rolls it back implicitly, // causing the drift's rollback recovery to fail. This is expected and not // harmful since the end result (transaction not applied) is the same. On // iOS, the transaction stays active despite the failing commit. final expectedException = Platform.isAndroid ? isA() : isA(); await expectLater( database.transaction(() async { await database.customStatement('INSERT INTO y VALUES (2);'); }), throwsA(expectedException), ); }); } class EmptyDb extends GeneratedDatabase { EmptyDb(QueryExecutor q) : super(q); @override final List allTables = const []; @override final schemaVersion = 1; }