mirror of https://github.com/AMT-Cheif/drift.git
Integration tests for type converters
This commit is contained in:
parent
7cddf6f5d7
commit
f4bc8e2121
|
@ -14,7 +14,7 @@ database tables in pure Dart and enjoy a fluent query API, auto-updating streams
|
|||
and more!
|
||||
{: .fs-6 .fw-300 }
|
||||
|
||||
[](https://travis-ci.com/simolus3/moor)
|
||||
[](https://cirrus-ci.com/github/simolus3/moor)
|
||||
[](https://codecov.io/gh/simolus3/moor)
|
||||
|
||||
[Get started now]({{ site.common_links.getting_started | absolute_url }}){: .btn .btn-green .fs-5 .mb-4 .mb-md-0 .mr-2 }
|
||||
|
|
|
@ -15,9 +15,8 @@ class Users extends Table {
|
|||
|
||||
BlobColumn get profilePicture => blob().nullable()();
|
||||
|
||||
// todo enable custom column example. The feature isn't stable yet.
|
||||
//TextColumn get preferences =>
|
||||
// text().map(const PreferenceConverter()).nullable()();
|
||||
TextColumn get preferences =>
|
||||
text().map(const PreferenceConverter()).nullable()();
|
||||
}
|
||||
|
||||
class Friendships extends Table {
|
||||
|
@ -59,7 +58,7 @@ class PreferenceConverter extends TypeConverter<Preferences, String> {
|
|||
return null;
|
||||
}
|
||||
|
||||
return json.encode(json.encode(value.toJson()));
|
||||
return json.encode(value.toJson());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -71,6 +70,7 @@ class PreferenceConverter extends TypeConverter<Preferences, String> {
|
|||
'amountOfGoodFriends':
|
||||
'SELECT COUNT(*) FROM friendships f WHERE f.really_good_friends AND (f.first_user = :user OR f.second_user = :user)',
|
||||
'userCount': 'SELECT COUNT(id) FROM users',
|
||||
'settingsFor': 'SELECT preferences FROM users WHERE id = :user',
|
||||
},
|
||||
)
|
||||
class Database extends _$Database {
|
||||
|
@ -139,4 +139,9 @@ class Database extends _$Database {
|
|||
|
||||
await into(friendships).insert(companion, orReplace: true);
|
||||
}
|
||||
|
||||
Future<void> updateSettings(int userId, Preferences c) async {
|
||||
await (update(users)..where((u) => u.id.equals(userId)))
|
||||
.write(UsersCompanion(preferences: Value(c)));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -27,11 +27,13 @@ class User extends DataClass implements Insertable<User> {
|
|||
final String name;
|
||||
final DateTime birthDate;
|
||||
final Uint8List profilePicture;
|
||||
final Preferences preferences;
|
||||
User(
|
||||
{@required this.id,
|
||||
@required this.name,
|
||||
@required this.birthDate,
|
||||
this.profilePicture});
|
||||
this.profilePicture,
|
||||
this.preferences});
|
||||
factory User.fromData(Map<String, dynamic> data, GeneratedDatabase db,
|
||||
{String prefix}) {
|
||||
final effectivePrefix = prefix ?? '';
|
||||
|
@ -46,6 +48,8 @@ class User extends DataClass implements Insertable<User> {
|
|||
.mapFromDatabaseResponse(data['${effectivePrefix}birth_date']),
|
||||
profilePicture: uint8ListType
|
||||
.mapFromDatabaseResponse(data['${effectivePrefix}profile_picture']),
|
||||
preferences: $UsersTable.$converter0.mapToDart(stringType
|
||||
.mapFromDatabaseResponse(data['${effectivePrefix}preferences'])),
|
||||
);
|
||||
}
|
||||
factory User.fromJson(Map<String, dynamic> json,
|
||||
|
@ -55,6 +59,7 @@ class User extends DataClass implements Insertable<User> {
|
|||
name: serializer.fromJson<String>(json['name']),
|
||||
birthDate: serializer.fromJson<DateTime>(json['born_on']),
|
||||
profilePicture: serializer.fromJson<Uint8List>(json['profilePicture']),
|
||||
preferences: serializer.fromJson<Preferences>(json['preferences']),
|
||||
);
|
||||
}
|
||||
@override
|
||||
|
@ -65,6 +70,7 @@ class User extends DataClass implements Insertable<User> {
|
|||
'name': serializer.toJson<String>(name),
|
||||
'born_on': serializer.toJson<DateTime>(birthDate),
|
||||
'profilePicture': serializer.toJson<Uint8List>(profilePicture),
|
||||
'preferences': serializer.toJson<Preferences>(preferences),
|
||||
};
|
||||
}
|
||||
|
||||
|
@ -79,6 +85,9 @@ class User extends DataClass implements Insertable<User> {
|
|||
profilePicture: profilePicture == null && nullToAbsent
|
||||
? const Value.absent()
|
||||
: Value(profilePicture),
|
||||
preferences: preferences == null && nullToAbsent
|
||||
? const Value.absent()
|
||||
: Value(preferences),
|
||||
) as T;
|
||||
}
|
||||
|
||||
|
@ -86,12 +95,14 @@ class User extends DataClass implements Insertable<User> {
|
|||
{int id,
|
||||
String name,
|
||||
DateTime birthDate,
|
||||
Uint8List profilePicture}) =>
|
||||
Uint8List profilePicture,
|
||||
Preferences preferences}) =>
|
||||
User(
|
||||
id: id ?? this.id,
|
||||
name: name ?? this.name,
|
||||
birthDate: birthDate ?? this.birthDate,
|
||||
profilePicture: profilePicture ?? this.profilePicture,
|
||||
preferences: preferences ?? this.preferences,
|
||||
);
|
||||
@override
|
||||
String toString() {
|
||||
|
@ -99,7 +110,8 @@ class User extends DataClass implements Insertable<User> {
|
|||
..write('id: $id, ')
|
||||
..write('name: $name, ')
|
||||
..write('birthDate: $birthDate, ')
|
||||
..write('profilePicture: $profilePicture')
|
||||
..write('profilePicture: $profilePicture, ')
|
||||
..write('preferences: $preferences')
|
||||
..write(')'))
|
||||
.toString();
|
||||
}
|
||||
|
@ -108,7 +120,9 @@ class User extends DataClass implements Insertable<User> {
|
|||
int get hashCode => $mrjf($mrjc(
|
||||
id.hashCode,
|
||||
$mrjc(
|
||||
name.hashCode, $mrjc(birthDate.hashCode, profilePicture.hashCode))));
|
||||
name.hashCode,
|
||||
$mrjc(birthDate.hashCode,
|
||||
$mrjc(profilePicture.hashCode, preferences.hashCode)))));
|
||||
@override
|
||||
bool operator ==(other) =>
|
||||
identical(this, other) ||
|
||||
|
@ -116,7 +130,8 @@ class User extends DataClass implements Insertable<User> {
|
|||
other.id == id &&
|
||||
other.name == name &&
|
||||
other.birthDate == birthDate &&
|
||||
other.profilePicture == profilePicture);
|
||||
other.profilePicture == profilePicture &&
|
||||
other.preferences == preferences);
|
||||
}
|
||||
|
||||
class UsersCompanion extends UpdateCompanion<User> {
|
||||
|
@ -124,11 +139,13 @@ class UsersCompanion extends UpdateCompanion<User> {
|
|||
final Value<String> name;
|
||||
final Value<DateTime> birthDate;
|
||||
final Value<Uint8List> profilePicture;
|
||||
final Value<Preferences> preferences;
|
||||
const UsersCompanion({
|
||||
this.id = const Value.absent(),
|
||||
this.name = const Value.absent(),
|
||||
this.birthDate = const Value.absent(),
|
||||
this.profilePicture = const Value.absent(),
|
||||
this.preferences = const Value.absent(),
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -182,8 +199,23 @@ class $UsersTable extends Users with TableInfo<$UsersTable, User> {
|
|||
);
|
||||
}
|
||||
|
||||
final VerificationMeta _preferencesMeta =
|
||||
const VerificationMeta('preferences');
|
||||
GeneratedTextColumn _preferences;
|
||||
@override
|
||||
List<GeneratedColumn> get $columns => [id, name, birthDate, profilePicture];
|
||||
GeneratedTextColumn get preferences =>
|
||||
_preferences ??= _constructPreferences();
|
||||
GeneratedTextColumn _constructPreferences() {
|
||||
return GeneratedTextColumn(
|
||||
'preferences',
|
||||
$tableName,
|
||||
true,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
List<GeneratedColumn> get $columns =>
|
||||
[id, name, birthDate, profilePicture, preferences];
|
||||
@override
|
||||
$UsersTable get asDslTable => this;
|
||||
@override
|
||||
|
@ -219,6 +251,7 @@ class $UsersTable extends Users with TableInfo<$UsersTable, User> {
|
|||
} else if (profilePicture.isRequired && isInserting) {
|
||||
context.missing(_profilePictureMeta);
|
||||
}
|
||||
context.handle(_preferencesMeta, const VerificationResult.success());
|
||||
return context;
|
||||
}
|
||||
|
||||
|
@ -246,6 +279,11 @@ class $UsersTable extends Users with TableInfo<$UsersTable, User> {
|
|||
map['profile_picture'] =
|
||||
Variable<Uint8List, BlobType>(d.profilePicture.value);
|
||||
}
|
||||
if (d.preferences.present) {
|
||||
final converter = $UsersTable.$converter0;
|
||||
map['preferences'] =
|
||||
Variable<String, StringType>(converter.mapToSql(d.preferences.value));
|
||||
}
|
||||
return map;
|
||||
}
|
||||
|
||||
|
@ -253,6 +291,8 @@ class $UsersTable extends Users with TableInfo<$UsersTable, User> {
|
|||
$UsersTable createAlias(String alias) {
|
||||
return $UsersTable(_db, alias);
|
||||
}
|
||||
|
||||
static PreferenceConverter $converter0 = const PreferenceConverter();
|
||||
}
|
||||
|
||||
class Friendship extends DataClass implements Insertable<Friendship> {
|
||||
|
@ -470,6 +510,13 @@ class UserCountResult {
|
|||
});
|
||||
}
|
||||
|
||||
class SettingsForResult {
|
||||
final Preferences preferences;
|
||||
SettingsForResult({
|
||||
this.preferences,
|
||||
});
|
||||
}
|
||||
|
||||
abstract class _$Database extends GeneratedDatabase {
|
||||
_$Database(QueryExecutor e) : super(const SqlTypeSystem.withDefaults(), e);
|
||||
$UsersTable _users;
|
||||
|
@ -482,6 +529,8 @@ abstract class _$Database extends GeneratedDatabase {
|
|||
name: row.readString('name'),
|
||||
birthDate: row.readDateTime('birth_date'),
|
||||
profilePicture: row.readBlob('profile_picture'),
|
||||
preferences:
|
||||
$UsersTable.$converter0.mapToDart(row.readString('preferences')),
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -555,6 +604,34 @@ abstract class _$Database extends GeneratedDatabase {
|
|||
.map((rows) => rows.map(_rowToUserCountResult).toList());
|
||||
}
|
||||
|
||||
SettingsForResult _rowToSettingsForResult(QueryRow row) {
|
||||
return SettingsForResult(
|
||||
preferences:
|
||||
$UsersTable.$converter0.mapToDart(row.readString('preferences')),
|
||||
);
|
||||
}
|
||||
|
||||
Future<List<SettingsForResult>> settingsFor(
|
||||
int user,
|
||||
{@Deprecated('No longer needed with Moor 1.6 - see the changelog for details')
|
||||
QueryEngine operateOn}) {
|
||||
return (operateOn ?? this).customSelect(
|
||||
'SELECT preferences FROM users WHERE id = :user',
|
||||
variables: [
|
||||
Variable.withInt(user),
|
||||
]).then((rows) => rows.map(_rowToSettingsForResult).toList());
|
||||
}
|
||||
|
||||
Stream<List<SettingsForResult>> watchSettingsFor(int user) {
|
||||
return customSelectStream('SELECT preferences FROM users WHERE id = :user',
|
||||
variables: [
|
||||
Variable.withInt(user),
|
||||
],
|
||||
readsFrom: {
|
||||
users
|
||||
}).map((rows) => rows.map(_rowToSettingsForResult).toList());
|
||||
}
|
||||
|
||||
@override
|
||||
List<TableInfo> get allTables => [users, friendships];
|
||||
}
|
||||
|
|
|
@ -0,0 +1,20 @@
|
|||
import 'package:test/test.dart';
|
||||
import 'package:tests/database/database.dart';
|
||||
|
||||
import 'suite.dart';
|
||||
|
||||
void customObjectTests(TestExecutor executor) {
|
||||
test('custom objects', () async {
|
||||
final db = Database(executor.createExecutor());
|
||||
|
||||
var preferences = await db.settingsFor(1);
|
||||
expect(preferences.single.preferences, isNull);
|
||||
|
||||
await db.updateSettings(1, Preferences(true));
|
||||
preferences = await db.settingsFor(1);
|
||||
|
||||
expect(preferences.single.preferences.receiveEmails, true);
|
||||
|
||||
await db.close();
|
||||
});
|
||||
}
|
|
@ -11,6 +11,8 @@ void migrationTests(TestExecutor executor) {
|
|||
// we write 3 users when the database is created
|
||||
final count = await database.userCount();
|
||||
expect(count.single.cOUNTid, 3);
|
||||
|
||||
await database.close();
|
||||
});
|
||||
|
||||
test('saves and restores database', () async {
|
||||
|
@ -23,5 +25,7 @@ void migrationTests(TestExecutor executor) {
|
|||
// the 3 initial users plus People.florian
|
||||
final count = await database.userCount();
|
||||
expect(count.single.cOUNTid, 4);
|
||||
|
||||
await database.close();
|
||||
});
|
||||
}
|
||||
|
|
|
@ -1,7 +1,8 @@
|
|||
import 'package:moor/moor.dart';
|
||||
import 'package:test/test.dart';
|
||||
|
||||
import 'package:tests/suite/migrations.dart';
|
||||
import 'custom_objects.dart';
|
||||
import 'migrations.dart';
|
||||
|
||||
abstract class TestExecutor {
|
||||
QueryExecutor createExecutor();
|
||||
|
@ -16,4 +17,5 @@ void runAllTests(TestExecutor executor) {
|
|||
});
|
||||
|
||||
migrationTests(executor);
|
||||
customObjectTests(executor);
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue