Go to file
Simon Binder e018f6d725 Implement order-by clauses for select statements 2019-02-19 11:10:27 +01:00
.idea release readiness? 2019-02-18 20:51:37 +01:00
end_to_end_testing Validate data integrity before inserts and updates 2019-02-13 19:30:13 +01:00
sally Implement order-by clauses for select statements 2019-02-19 11:10:27 +01:00
sally_flutter Turns out LIMIT doesn't work for update and delete :O 2019-02-19 10:42:30 +01:00
sally_generator release readiness? 2019-02-18 20:51:37 +01:00
tool release readiness? 2019-02-18 20:51:37 +01:00
.gitignore Add .gitignores 2019-02-03 13:54:45 +01:00
.travis.yml Format, improve travis testing 2019-02-14 17:55:31 +01:00
LICENSE Add MIT license 2019-02-09 21:00:00 +01:00
README.md release readiness? 2019-02-18 20:51:37 +01:00
analysis_options.yaml Specify analysis rules and conform to them 2019-02-04 21:03:41 +01:00

README.md

Sally

Build Status

Sally is an easy to use and safe way to persist data for Flutter apps. It features a fluent Dart DSL to describe tables and will generate matching database code that can be used to easily read and store your app's data. It also features a reactive API that will deliver auto-updating streams for your queries.

Getting started

Adding the dependency

First, let's add sally to your project's pubspec.yaml: TODO: Finish this part of the readme when sally is out on pub.

dependencies:
  sally:
    git:
      url: 
      path: sally/

dev_dependencies:
  sally_generator:
    git:
      url:
      path: sally_generator/
  build_runner:

We're going to use the sally_flutter library to specify tables and access the database. The sally_generator library will take care of generating the necessary code so the library knows how your table structure looks like.

Declaring tables

You can use the DSL included with this library to specify your libraries with simple dart code:

import 'package:sally_flutter/sally_flutter.dart';

// assuming that your file is called filename.dart. This will give an error at first,
// but it's needed for sally to know about the generated code
part 'filename.g.dart'; 

// this will generate a table called "todos" for us. The rows of that table will
// be represented by a class called "Todo".
class Todos extends Table {
  IntColumn get id => integer().autoIncrement()();
  TextColumn get title => text().withLength(min: 6, max: 10)();
  TextColumn get content => text().named('body')();
  IntColumn get category => integer().nullable()();
}

// This will make sally generate a class called "Category" to represent a row in this table.
// By default, "Categorie" would have been used because it only strips away the trailing "s"
// in the table name.
@DataClassName("Category")
class Categories extends Table {
  
  IntColumn get id => integer().autoIncrement()();
  TextColumn get description => text()();
}

// this annotation tells sally to prepare a database class that uses both of the
// tables we just defined. We'll see how to use that database class in a moment.
@UseSally(tables: [Todos, Categories])
class MyDatabase {
  
}

⚠️ Warning: Even though it might look like it, the content of a Table class does not support full Dart code. It can only be used to declare the table name, it's primary keys and columns. The code inside of a table class will never be executed. Instead, the generator will take a look at your table classes to figure out how their structure looks like. This won't work if the body of your tables is not constant. This should not be problem, but please be aware of this as you can't put logic inside these classes.

Generating the code

Sally integrates with the dart build system, so you can generate all the code needed with flutter packages pub run build_runner build. If you want to continously rebuild the code whever you change your code, run flutter packages pub run build_runner watch instead. After running either command once, sally generator will have created a class for your database and data classes for your entities. To use it, change the MyDatabase class as follows:

@UseSally(tables: [Todos, Categories])
class MyDatabase extends _$MyDatabase {
  // we tell the database where to store the data with this constructor
  MyDatabase() : super(FlutterQueryExecutor.inDatabaseFolder(path: 'db.sqlite'));

  // you should bump this number whenever you change or add a table definition. Migrations
  // are covered later in this readme.
  @override
  int get schemaVersion => 1; 
}

You can ignore the schemaVersion at the moment, the important part is that you can now run your queries with fluent Dart code:

Writing queries

class MyDatabase extends _$MyDatabase {
  // .. the versionCode getter still needs to be here

  // loads all todo entries
  Future<List<Todo>> get allTodoEntries => select(todos).get();

  // watches all todo entries in a given category. The stream will automatically
  // emit new items whenever the underlying data changes.
  Stream<List<TodoEntry>> watchEntriesInCategory(Category c) {
    return (select(todos)..where((t) => t.category.equals(c.id))).watch();
  }
}

Select statements

You can create select statements by starting them with select(tableName), where the table name is a field generated for you by sally. Each table used in a database will have a matching field to run queries against. A query can be run once with get() or be turned into an auto-updating stream using watch().

Where

You can apply filters to a query by calling where(). The where method takes a function that should map the given table to an Expression of boolean. A common way to create such expression is by using equals on expressions. Integer columns can also be compared with isBiggerThan and isSmallerThan. You can compose expressions using and(a, b), or(a, b) and not(a).

Limit

You can limit the amount of results returned by calling limit on queries. The method accepts the amount of rows to return and an optional offset.

Updates and deletes

You can use the generated row class to update individual fields of any row:

Future moveImportantTasksIntoCategory(Category target) {
  return (update(todos)
      ..where((t) => t.title.like('%Important%'))
    ).write(TodoEntry(
      category: target.id
    ),
  );
}

Future feelingLazy() {
  // delete 10 todo entries just cause
  return (delete(todos)..limit(10)).go();
}

⚠️ Caution: If you don't explicitly add a where or limit clause on updates or deletes, the statement will affect all rows in the table!

Inserts

You can very easily insert any valid object into tables:

// returns the generated id
Future<int> addTodoEntry(Todo entry) {
  return into(todos).insert(entry);
}

All row classes generated will have a constructor that can be used to create objects:

addTodoEntry(
  Todo(
    title: 'Important task',
    content: 'Refactor persistence code',
  ),
);

If a column is nullable or has a default value (this includes auto-increments), the field can be omitted. All other fields must be set and non-null. The insert method will throw otherwise.

Migrations

Sally provides a migration API that can be used to gradually apply schema changes after bumping the schemaVersion getter inside the Database class. To use it, override the migration getter. Here's an example: Let's say you wanted to add a due date to your todo entries:

class Todos extends Table {
  IntColumn get id => integer().autoIncrement()();
  TextColumn get title => text().withLength(min: 6, max: 10)();
  TextColumn get content => text().named('body')();
  IntColumn get category => integer().nullable()();
  DateTimeColumn get dueDate => dateTime().nullable()(); // we just added this column
}

We can now change the database class like this:

  @override
  int get schemaVersion => 1; // bump because the tables have changed

  @override
  MigrationStrategy get migration => MigrationStrategy(
    onCreate: (Migrator m) {
      return m.createAllTables();
    },
    onUpgrade: (Migrator m, int from, int to) async {
      if (from == 1) {
        // we added the dueDate propery in the change from version 1
        await m.addColumn(todos, todos.dueDate);
      }
    }
  );

  // rest of class can stay the same

You can also add individual tables or drop them.

TODO-List and current limitations

Limitions (at the moment)

  • No joins
  • No group by or window functions
  • Custom primary key support is very limited
  • No ORDER BY

Planned for the future

These aren't sorted by priority. If you have more ideas or want some features happening soon, let us know by creating an issue!

  • Refactor comparison API
    1. Instead of defining them in IntColumn, move isBiggerThan and isSmallerThan into a new class (comparable expression?)
    2. Support for non-strict comparisons (<=, >=)
    3. Support ORDER BY clauses.
  • Specify primary keys
  • Simple COUNT(*) operations (group operations will be much more complicated)
  • Support default values and expressions
  • Allow using DAOs or some other mechanism instead of having to put everything in the main database class.
  • Support more Datatypes: We should at least support Uint8List out of the box, supporting floating / fixed point numbers as well would be awesome
  • Nullable / non-nullable datatypes
    • DSL API ✔️
    • Support in generator ✔️
    • Use in queries (IS NOT NULL) ✔️
    • Setting fields to null during updates
  • Support Dart VM apps
  • References
    • DSL API
    • Support in generator
    • Validation
  • Table joins
  • Bulk inserts
  • Transactions

Interesting stuff that would be nice to have

Implementing this will very likely result in backwards-incompatible changes.

  • Find a way to hide implementation details from users while still making them accessible for the generated code
  • GROUP BY grouping functions
  • Support for different database engines
    • Support webapps via AlaSQL or a different engine