Initial support for sql expressions in companions

This commit is contained in:
Simon Binder 2020-04-17 20:48:22 +02:00
parent 62a363105a
commit 84bac1bf1d
No known key found for this signature in database
GPG Key ID: 7891917E4147B8C0
8 changed files with 270 additions and 0 deletions

View File

@ -30,6 +30,7 @@
- New feature: Nested results for compiled queries ([#288](https://github.com/simolus3/moor/issues/288)) - New feature: Nested results for compiled queries ([#288](https://github.com/simolus3/moor/issues/288))
See the [documentation](https://moor.simonbinder.eu/docs/using-sql/moor_files/#nested-results) for See the [documentation](https://moor.simonbinder.eu/docs/using-sql/moor_files/#nested-results) for
details on how and when to use this feature. details on how and when to use this feature.
- New feature: Use sql expressions for inserts with `Companion.custom`.
## 2.4.1 ## 2.4.1

View File

@ -90,6 +90,16 @@ class CategoriesCompanion extends UpdateCompanion<Category> {
this.id = const Value.absent(), this.id = const Value.absent(),
this.description = const Value.absent(), this.description = const Value.absent(),
}); });
static Insertable<Category> custom({
Expression<int> id,
Expression<String> description,
}) {
return RawValuesInsertable({
if (id != null) 'id': id,
if (description != null) 'description': description,
});
}
CategoriesCompanion copyWith({Value<int> id, Value<String> description}) { CategoriesCompanion copyWith({Value<int> id, Value<String> description}) {
return CategoriesCompanion( return CategoriesCompanion(
id: id ?? this.id, id: id ?? this.id,
@ -294,6 +304,20 @@ class RecipesCompanion extends UpdateCompanion<Recipe> {
this.category = const Value.absent(), this.category = const Value.absent(),
}) : title = Value(title), }) : title = Value(title),
instructions = Value(instructions); instructions = Value(instructions);
static Insertable<Recipe> custom({
Expression<int> id,
Expression<String> title,
Expression<String> instructions,
Expression<int> category,
}) {
return RawValuesInsertable({
if (id != null) 'id': id,
if (title != null) 'title': title,
if (instructions != null) 'instructions': instructions,
if (category != null) 'category': category,
});
}
RecipesCompanion copyWith( RecipesCompanion copyWith(
{Value<int> id, {Value<int> id,
Value<String> title, Value<String> title,
@ -523,6 +547,18 @@ class IngredientsCompanion extends UpdateCompanion<Ingredient> {
@required int caloriesPer100g, @required int caloriesPer100g,
}) : name = Value(name), }) : name = Value(name),
caloriesPer100g = Value(caloriesPer100g); caloriesPer100g = Value(caloriesPer100g);
static Insertable<Ingredient> custom({
Expression<int> id,
Expression<String> name,
Expression<int> caloriesPer100g,
}) {
return RawValuesInsertable({
if (id != null) 'id': id,
if (name != null) 'name': name,
if (caloriesPer100g != null) 'calories': caloriesPer100g,
});
}
IngredientsCompanion copyWith( IngredientsCompanion copyWith(
{Value<int> id, Value<String> name, Value<int> caloriesPer100g}) { {Value<int> id, Value<String> name, Value<int> caloriesPer100g}) {
return IngredientsCompanion( return IngredientsCompanion(
@ -741,6 +777,18 @@ class IngredientInRecipesCompanion extends UpdateCompanion<IngredientInRecipe> {
}) : recipe = Value(recipe), }) : recipe = Value(recipe),
ingredient = Value(ingredient), ingredient = Value(ingredient),
amountInGrams = Value(amountInGrams); amountInGrams = Value(amountInGrams);
static Insertable<IngredientInRecipe> custom({
Expression<int> recipe,
Expression<int> ingredient,
Expression<int> amountInGrams,
}) {
return RawValuesInsertable({
if (recipe != null) 'recipe': recipe,
if (ingredient != null) 'ingredient': ingredient,
if (amountInGrams != null) 'amount': amountInGrams,
});
}
IngredientInRecipesCompanion copyWith( IngredientInRecipesCompanion copyWith(
{Value<int> recipe, Value<int> ingredient, Value<int> amountInGrams}) { {Value<int> recipe, Value<int> ingredient, Value<int> amountInGrams}) {
return IngredientInRecipesCompanion( return IngredientInRecipesCompanion(

View File

@ -61,6 +61,28 @@ abstract class UpdateCompanion<D extends DataClass> implements Insertable<D> {
const UpdateCompanion(); const UpdateCompanion();
} }
/// An [Insertable] implementation based on raw column expressions.
///
/// Mostly used in generated code.
class RawValuesInsertable<D extends DataClass> implements Insertable<D> {
/// A map from column names to a value that should be inserted or updated.
///
/// See also:
/// - [toColumns], which returns [data] in a [RawValuesInsertable]
final Map<String, Expression> data;
/// Creates a [RawValuesInsertable] based on the [data] to insert or update.
const RawValuesInsertable(this.data);
@override
Map<String, Expression> toColumns(bool nullToAbsent) => data;
@override
String toString() {
return 'RawValuesInsertable($data)';
}
}
/// A wrapper around arbitrary data [T] to indicate presence or absence /// A wrapper around arbitrary data [T] to indicate presence or absence
/// explicitly. We can use [Value]s in companions to distinguish between null /// explicitly. We can use [Value]s in companions to distinguish between null
/// and absent values. /// and absent values.

View File

@ -65,6 +65,9 @@ class Variable<T> extends Expression<T> {
context.buffer.write('NULL'); context.buffer.write('NULL');
} }
} }
@override
String toString() => 'Variable($value)';
} }
/// An expression that represents the value of a dart object encoded to sql /// An expression that represents the value of a dart object encoded to sql
@ -98,4 +101,7 @@ class Constant<T> extends Expression<T> {
// ignore: test_types_in_equals // ignore: test_types_in_equals
(other as Constant<T>).value == value; (other as Constant<T>).value == value;
} }
@override
String toString() => 'Constant($value)';
} }

View File

@ -107,6 +107,18 @@ class ConfigCompanion extends UpdateCompanion<Config> {
this.configValue = const Value.absent(), this.configValue = const Value.absent(),
this.syncState = const Value.absent(), this.syncState = const Value.absent(),
}) : configKey = Value(configKey); }) : configKey = Value(configKey);
static Insertable<Config> custom({
Expression<String> configKey,
Expression<String> configValue,
Expression<int> syncState,
}) {
return RawValuesInsertable({
if (configKey != null) 'config_key': configKey,
if (configValue != null) 'config_value': configValue,
if (syncState != null) 'sync_state': syncState,
});
}
ConfigCompanion copyWith( ConfigCompanion copyWith(
{Value<String> configKey, {Value<String> configKey,
Value<String> configValue, Value<String> configValue,
@ -292,6 +304,16 @@ class WithDefaultsCompanion extends UpdateCompanion<WithDefault> {
this.a = const Value.absent(), this.a = const Value.absent(),
this.b = const Value.absent(), this.b = const Value.absent(),
}); });
static Insertable<WithDefault> custom({
Expression<String> a,
Expression<int> b,
}) {
return RawValuesInsertable({
if (a != null) 'a': a,
if (b != null) 'b': b,
});
}
WithDefaultsCompanion copyWith({Value<String> a, Value<int> b}) { WithDefaultsCompanion copyWith({Value<String> a, Value<int> b}) {
return WithDefaultsCompanion( return WithDefaultsCompanion(
a: a ?? this.a, a: a ?? this.a,
@ -437,6 +459,14 @@ class NoIdsCompanion extends UpdateCompanion<NoId> {
NoIdsCompanion.insert({ NoIdsCompanion.insert({
@required Uint8List payload, @required Uint8List payload,
}) : payload = Value(payload); }) : payload = Value(payload);
static Insertable<NoId> custom({
Expression<Uint8List> payload,
}) {
return RawValuesInsertable({
if (payload != null) 'payload': payload,
});
}
NoIdsCompanion copyWith({Value<Uint8List> payload}) { NoIdsCompanion copyWith({Value<Uint8List> payload}) {
return NoIdsCompanion( return NoIdsCompanion(
payload: payload ?? this.payload, payload: payload ?? this.payload,
@ -603,6 +633,18 @@ class WithConstraintsCompanion extends UpdateCompanion<WithConstraint> {
@required int b, @required int b,
this.c = const Value.absent(), this.c = const Value.absent(),
}) : b = Value(b); }) : b = Value(b);
static Insertable<WithConstraint> custom({
Expression<String> a,
Expression<int> b,
Expression<double> c,
}) {
return RawValuesInsertable({
if (a != null) 'a': a,
if (b != null) 'b': b,
if (c != null) 'c': c,
});
}
WithConstraintsCompanion copyWith( WithConstraintsCompanion copyWith(
{Value<String> a, Value<int> b, Value<double> c}) { {Value<String> a, Value<int> b, Value<double> c}) {
return WithConstraintsCompanion( return WithConstraintsCompanion(
@ -819,6 +861,20 @@ class MytableCompanion extends UpdateCompanion<MytableData> {
this.somebool = const Value.absent(), this.somebool = const Value.absent(),
this.somedate = const Value.absent(), this.somedate = const Value.absent(),
}); });
static Insertable<MytableData> custom({
Expression<int> someid,
Expression<String> sometext,
Expression<bool> somebool,
Expression<DateTime> somedate,
}) {
return RawValuesInsertable({
if (someid != null) 'someid': someid,
if (sometext != null) 'sometext': sometext,
if (somebool != null) 'somebool': somebool,
if (somedate != null) 'somedate': somedate,
});
}
MytableCompanion copyWith( MytableCompanion copyWith(
{Value<int> someid, {Value<int> someid,
Value<String> sometext, Value<String> sometext,
@ -1034,6 +1090,18 @@ class EmailCompanion extends UpdateCompanion<EMail> {
}) : sender = Value(sender), }) : sender = Value(sender),
title = Value(title), title = Value(title),
body = Value(body); body = Value(body);
static Insertable<EMail> custom({
Expression<String> sender,
Expression<String> title,
Expression<String> body,
}) {
return RawValuesInsertable({
if (sender != null) 'sender': sender,
if (title != null) 'title': title,
if (body != null) 'body': body,
});
}
EmailCompanion copyWith( EmailCompanion copyWith(
{Value<String> sender, Value<String> title, Value<String> body}) { {Value<String> sender, Value<String> title, Value<String> body}) {
return EmailCompanion( return EmailCompanion(

View File

@ -149,6 +149,22 @@ class TodosTableCompanion extends UpdateCompanion<TodoEntry> {
this.targetDate = const Value.absent(), this.targetDate = const Value.absent(),
this.category = const Value.absent(), this.category = const Value.absent(),
}) : content = Value(content); }) : content = Value(content);
static Insertable<TodoEntry> custom({
Expression<int> id,
Expression<String> title,
Expression<String> content,
Expression<DateTime> targetDate,
Expression<int> category,
}) {
return RawValuesInsertable({
if (id != null) 'id': id,
if (title != null) 'title': title,
if (content != null) 'content': content,
if (targetDate != null) 'target_date': targetDate,
if (category != null) 'category': category,
});
}
TodosTableCompanion copyWith( TodosTableCompanion copyWith(
{Value<int> id, {Value<int> id,
Value<String> title, Value<String> title,
@ -383,6 +399,16 @@ class CategoriesCompanion extends UpdateCompanion<Category> {
this.id = const Value.absent(), this.id = const Value.absent(),
@required String description, @required String description,
}) : description = Value(description); }) : description = Value(description);
static Insertable<Category> custom({
Expression<int> id,
Expression<String> description,
}) {
return RawValuesInsertable({
if (id != null) 'id': id,
if (description != null) 'desc': description,
});
}
CategoriesCompanion copyWith({Value<int> id, Value<String> description}) { CategoriesCompanion copyWith({Value<int> id, Value<String> description}) {
return CategoriesCompanion( return CategoriesCompanion(
id: id ?? this.id, id: id ?? this.id,
@ -610,6 +636,22 @@ class UsersCompanion extends UpdateCompanion<User> {
this.creationTime = const Value.absent(), this.creationTime = const Value.absent(),
}) : name = Value(name), }) : name = Value(name),
profilePicture = Value(profilePicture); profilePicture = Value(profilePicture);
static Insertable<User> custom({
Expression<int> id,
Expression<String> name,
Expression<bool> isAwesome,
Expression<Uint8List> profilePicture,
Expression<DateTime> creationTime,
}) {
return RawValuesInsertable({
if (id != null) 'id': id,
if (name != null) 'name': name,
if (isAwesome != null) 'is_awesome': isAwesome,
if (profilePicture != null) 'profile_picture': profilePicture,
if (creationTime != null) 'creation_time': creationTime,
});
}
UsersCompanion copyWith( UsersCompanion copyWith(
{Value<int> id, {Value<int> id,
Value<String> name, Value<String> name,
@ -843,6 +885,16 @@ class SharedTodosCompanion extends UpdateCompanion<SharedTodo> {
@required int user, @required int user,
}) : todo = Value(todo), }) : todo = Value(todo),
user = Value(user); user = Value(user);
static Insertable<SharedTodo> custom({
Expression<int> todo,
Expression<int> user,
}) {
return RawValuesInsertable({
if (todo != null) 'todo': todo,
if (user != null) 'user': user,
});
}
SharedTodosCompanion copyWith({Value<int> todo, Value<int> user}) { SharedTodosCompanion copyWith({Value<int> todo, Value<int> user}) {
return SharedTodosCompanion( return SharedTodosCompanion(
todo: todo ?? this.todo, todo: todo ?? this.todo,
@ -1043,6 +1095,18 @@ class TableWithoutPKCompanion extends UpdateCompanion<TableWithoutPKData> {
this.custom = const Value.absent(), this.custom = const Value.absent(),
}) : notReallyAnId = Value(notReallyAnId), }) : notReallyAnId = Value(notReallyAnId),
someFloat = Value(someFloat); someFloat = Value(someFloat);
static Insertable<TableWithoutPKData> createCustom({
Expression<int> notReallyAnId,
Expression<double> someFloat,
Expression<String> custom,
}) {
return RawValuesInsertable({
if (notReallyAnId != null) 'not_really_an_id': notReallyAnId,
if (someFloat != null) 'some_float': someFloat,
if (custom != null) 'custom': custom,
});
}
TableWithoutPKCompanion copyWith( TableWithoutPKCompanion copyWith(
{Value<int> notReallyAnId, {Value<int> notReallyAnId,
Value<double> someFloat, Value<double> someFloat,
@ -1242,6 +1306,16 @@ class PureDefaultsCompanion extends UpdateCompanion<PureDefault> {
this.id = const Value.absent(), this.id = const Value.absent(),
this.txt = const Value.absent(), this.txt = const Value.absent(),
}); });
static Insertable<PureDefault> custom({
Expression<int> id,
Expression<String> txt,
}) {
return RawValuesInsertable({
if (id != null) 'id': id,
if (txt != null) 'insert': txt,
});
}
PureDefaultsCompanion copyWith({Value<int> id, Value<String> txt}) { PureDefaultsCompanion copyWith({Value<int> id, Value<String> txt}) {
return PureDefaultsCompanion( return PureDefaultsCompanion(
id: id ?? this.id, id: id ?? this.id,

View File

@ -129,4 +129,20 @@ void main() {
verify(executor verify(executor
.runInsert('INSERT INTO pure_defaults (`insert`) VALUES (?)', ['foo'])); .runInsert('INSERT INTO pure_defaults (`insert`) VALUES (?)', ['foo']));
}); });
test('can insert custom companions', () async {
await db.into(db.users).insert(UsersCompanion.custom(
isAwesome: const Constant(true),
name: const Variable('User name'),
profilePicture: const CustomExpression('_custom_'),
creationTime: currentDateAndTime));
verify(
executor.runInsert(
'INSERT INTO users (name, is_awesome, profile_picture, creation_time) '
"VALUES (?, 1, _custom_, strftime('%s', CURRENT_TIMESTAMP))",
['User name'],
),
);
});
} }

View File

@ -16,8 +16,11 @@ class UpdateCompanionWriter {
_buffer.write('class ${table.getNameForCompanionClass(scope.options)} ' _buffer.write('class ${table.getNameForCompanionClass(scope.options)} '
'extends UpdateCompanion<${table.dartTypeName}> {\n'); 'extends UpdateCompanion<${table.dartTypeName}> {\n');
_writeFields(); _writeFields();
_writeConstructor(); _writeConstructor();
_writeInsertConstructor(); _writeInsertConstructor();
_writeCustomConstructor();
_writeCopyWith(); _writeCopyWith();
_writeToColumnsOverride(); _writeToColumnsOverride();
@ -87,6 +90,38 @@ class UpdateCompanionWriter {
_buffer.write(';\n'); _buffer.write(';\n');
} }
void _writeCustomConstructor() {
// Prefer a .custom constructor, unless there already is a field called
// "custom", in which case we'll use createCustom
final constructorName = table.columns
.map((e) => e.dartGetterName)
.any((name) => name == 'custom')
? 'createCustom'
: 'custom';
_buffer
..write('static Insertable<${table.dartTypeName}> $constructorName')
..write('({');
for (final column in table.columns) {
_buffer
..write('Expression<${column.variableTypeName}> ')
..write(column.dartGetterName)
..write(',\n');
}
_buffer..write('}) {\n')..write('return RawValuesInsertable({');
for (final column in table.columns) {
_buffer
..write('if (${column.dartGetterName} != null)')
..write(asDartLiteral(column.name.name))
..write(': ${column.dartGetterName},');
}
_buffer.write('});\n}');
}
void _writeCopyWith() { void _writeCopyWith() {
_buffer _buffer
..write(table.getNameForCompanionClass(scope.options)) ..write(table.getNameForCompanionClass(scope.options))