Frequently asked questions
Using the database
If you've created a MyDatabase
class by following the getting started guide, you still need to somehow obtain an instance of it. It's recommended to only have one (singleton) instance of your database, so you could store that instance in a global variable:
Vanilla flutter
MyDatabase database;
void main() {
database = MyDatabase();
runApp(MyFlutterApp());
}
It would be cleaner to use InheritedWidgets
for that, and the provider
package helps here:
Provider
If you're using the provider package, you can wrap your top-level widget in a provider that manages the database instance:
void main() {
runApp(
Provider<MyDatabase>(
create: (context) => MyDatabase(),
child: MyFlutterApp(),
dispose: (context, db) => db.close(),
),
);
}
Your widgets would then have access to the database using Provider.of<MyDatabase>(context)
.
GetX
If you're using the GetX package, you can add it as a service that manages the database instance:.
void main() {
Get.put(MyDatabase());
runApp(MyFlutterApp());
}
Your widgets would then have access to the database using Get.find<MyDatabase>().your_method
.
A more complex architecture
If you're strict on keeping your business logic out of the widget layer, you probably use some dependency injection framework like kiwi
or get_it
to instantiate services and view models. Creating a singleton instance of MyDatabase
in your favorite dependency injection framework for flutter hence solves this problem for you.
Why am I getting no such table errors?
If you add another table after your app has already been installed, you need to write a migration that covers creating that table. If you're in the process of developing your app and want to use un- and reinstall your app instead of writing migrations, that's fine too. Please note that your apps data might be backed up on Android, so manually deleting your app's data instead of a reinstall is necessary on some devices.
How do I fix lints in generated files?
Based on your linter settings, you might see some warnings in the .g.dart
files generated by drift. Since lints are mainly used for human-written code, we recommend to disable static analysis on generated files. For that, generate a top-level file called analysis_options.yaml
in your project and add this content:
analyzer:
exclude:
- "**/*.g.dart"
You might have to restart your IDE for the changes to apply.
How can I inspect generated SQL?
All database implementations (NativeDatabase
, FlutterQueryExecutor
, ...) have a logStatements
parameter that you can set to true
. When enabled, drift will print the statements it runs.
How do I insert data on the first app start?
You can populate the database on the first start of your app with a custom migration strategy. To insert data when the database is created (which usually happens when the app is first run), you can use this:
MigrationStrategy(
onCreate: (m) async {
await m.createAll(); // create all tables
await into(myTable).insert(...); // insert on first run.
}
)
You can even use transactions or batches in the onCreate
callback.
Another approach is to include a pre-populated database in your app's asset and use that one:
QueryExecutor databaseWithDefaultAsset(File file, String asset) {
// A LazyDatabase lets us do async initialization work.
return LazyDatabase(() async {
if (!await file.exists()) {
// Database does not exist yet, use default from asset
final content = await rootBundle.load(asset);
await file.parent.create(recursive: true);
await file.writeAsBytes(content.buffer.asUint8List(0));
}
});
}
My generated code is using another class with the same name
When you have an imported a class with the same name as the should-be generated ones on your database.dart
file and your run build_runner
there is a known problem about it using the imported class instead of the generated ones.
To solve that, if you can, you can enable modular code generation. It slightly changes how drift is used, but you probably will only have to update a few files to change parts to imports. The big difference is that it allows drift to emit a standalone library file instead of a part - that can have its own imports, so this is much easier and we'll use the correct imports even when the same name classes are imported in your database.dart
file.
How does drift compare to X?
There are a variety of good persistence libraries for Dart and Flutter.
That said, here's an incomplete (and obviously biased) list of great libraries and how drift compares to them. If you have experience with any of these (or other) libraries and want to share how they compare to drift, please feel invited to contribute to this page.
sqflite, sqlite3
Sqflite is a Flutter package that provides bindings to the sqlite api for both iOS and Android. It's well maintained and has stable api. In fact, the moor_flutter
or drift_sqflite
variants are built on top of sqflite. But even though sqflite has an api to construct some simple queries in Dart, drift goes a bit further by
- Generating type-safe mapping code for your queries
- Providing auto-updating streams for queries
- Managing
CREATE TABLE
statements and most schema migrations - A more fluent api to compose queries
Still, for most apps that don't need these features, sqflite can be a very fitting persistence library.
The same thing applies to the sqlite3
package - package:drift/native.dart
uses that library, but provides additional services on top.
sqlcool
Sqlcool is a lightweight library around sqflite that makes writing queries and schema management easier, it also has auto-updating streams. It can be a decent alternative to drift if you don't want/need generated code to parse the result of your queries.
floor
Floor also has a lot of convenience features like auto-updating queries and schema migrations. Similar to drift, you define the structure of your database in Dart. Then, you have write queries in sql - the mapping code if generated by floor. Drift has a similar feature, but it can also verify that your queries are valid at compile time. Drift additionally has an api that lets you write some queries in Dart instead of sql.
A difference between these two is that Floor lets you write your own classes and generates mapping code around that. By default, drift generates most classes for you, which can make it easier to use, but makes the api less flexible in some instances. Drift can also be used with custom row classes though.
firebase
Both the Realtime Database and Cloud Datastore are easy to use persistence libraries that can sync across devices while still working offline. Both of them feature auto-updating streams and a simple query api. However, neither of them is a relational database, so they don't support useful sql features like aggregate functions, joins, or complex filters.
Firebase is a very good option when
- your data model can be expressed as documents instead of relations
- you don't have your own backend, but still need to synchronize data
Can I view a drift database?
Yes! Drift stores its data in a sqlite3 database file that can be extracted from the device and inspected locally.
To inspect a drift database directly in your app, you can use the drift_db_viewer
package by Koen Van Looveren.
There is also an under-development DevTools Extension that comes when you add drift
to your app. Open DevTools and try it out! Contributions and feedback are definitely welcome!