Integration tests for type converters

This commit is contained in:
Simon Binder 2019-07-28 11:43:02 +02:00
parent 7cddf6f5d7
commit f4bc8e2121
No known key found for this signature in database
GPG Key ID: 7891917E4147B8C0
6 changed files with 120 additions and 12 deletions

View File

@ -14,7 +14,7 @@ database tables in pure Dart and enjoy a fluent query API, auto-updating streams
and more!
{: .fs-6 .fw-300 }
[![Build Status](https://travis-ci.com/simolus3/moor.svg?token=u4VnFEE5xnWVvkE6QsqL&branch=master)](https://travis-ci.com/simolus3/moor)
[![Build Status](https://api.cirrus-ci.com/github/simolus3/moor.svg)](https://cirrus-ci.com/github/simolus3/moor)
[![codecov](https://codecov.io/gh/simolus3/moor/branch/master/graph/badge.svg)](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 }

View File

@ -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)));
}
}

View File

@ -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];
}

View File

@ -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();
});
}

View File

@ -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();
});
}

View File

@ -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);
}