platform/packages/sembast/lib/angel_sembast.dart
Tobe O b1a6f262ea Add 'packages/sembast/' from commit 'eda2acc36a69cefbe038c0a362a03b23aa4f13de'
git-subtree-dir: packages/sembast
git-subtree-mainline: 567ddd3897
git-subtree-split: eda2acc36a
2020-02-15 18:43:59 -05:00

166 lines
5 KiB
Dart

import 'dart:async';
import 'package:angel_framework/angel_framework.dart';
import 'package:sembast/sembast.dart';
class SembastService extends Service<String, Map<String, dynamic>> {
final Database database;
final StoreRef<int, Map<String, dynamic>> store;
/// If set to `true`, clients can remove all items by passing a `null` `id` to `remove`.
///
/// `false` by default.
final bool allowRemoveAll;
/// If set to `true`, parameters in `req.query` are applied to the database query.
final bool allowQuery;
SembastService(this.database,
{String store, this.allowRemoveAll = false, this.allowQuery = true})
: this.store = intMapStoreFactory.store(store),
super();
Finder _makeQuery([Map<String, dynamic> params]) {
params = Map<String, dynamic>.from(params ?? {});
Filter out;
var sort = <SortOrder>[];
// You can pass a Finder as 'query':
if (params['query'] is Finder) {
return params['query'] as Finder;
}
for (var key in params.keys) {
if (key == r'$sort' &&
(allowQuery == true || !params.containsKey('provider'))) {
var v = params[key];
if (v is! Map) {
sort.add(SortOrder(v.toString(), true));
} else {
var m = v as Map;
m.forEach((k, sorter) {
if (sorter is SortOrder) {
sort.add(sorter);
} else if (sorter is String) {
sort.add(SortOrder(k.toString(), sorter == "-1"));
} else if (sorter is num) {
sort.add(SortOrder(k.toString(), sorter == -1));
}
});
}
} else if (key == 'query' &&
(allowQuery == true || !params.containsKey('provider'))) {
var queryObj = params[key];
if (queryObj is Map) {
queryObj.forEach((k, v) {
if (k != 'provider' &&
!const ['__requestctx', '__responsectx'].contains(k)) {
var filter = Filter.equals(k.toString(), v);
if (out == null) {
out = filter;
} else {
out = Filter.or([out, filter]);
}
}
});
}
}
}
return Finder(filter: out, sortOrders: sort);
}
Map<String, dynamic> _withId(Map<String, dynamic> data, String id) =>
Map<String, dynamic>.from(data ?? {})..['id'] = id;
@override
Future<Map<String, dynamic>> findOne(
[Map<String, dynamic> params,
String errorMessage =
'No record was found matching the given query.']) async {
return (await store.findFirst(database, finder: _makeQuery(params)))?.value;
}
@override
Future<List<Map<String, dynamic>>> index(
[Map<String, dynamic> params]) async {
var records = await store.find(database, finder: _makeQuery(params));
return records
.where((r) => r.value != null)
.map((r) => _withId(r.value, r.key.toString()))
.toList();
}
@override
Future<Map<String, dynamic>> read(String id,
[Map<String, dynamic> params]) async {
var record = await store.record(int.parse(id)).getSnapshot(database);
if (record == null) {
throw AngelHttpException.notFound(message: 'No record found for ID $id');
}
return _withId(record.value, id);
}
@override
Future<Map<String, dynamic>> create(Map<String, dynamic> data,
[Map<String, dynamic> params]) async {
return await database.transaction((txn) async {
var key = await store.add(txn, data);
var id = key.toString();
return _withId(data, id);
});
}
@override
Future<Map<String, dynamic>> modify(String id, Map<String, dynamic> data,
[Map<String, dynamic> params]) async {
return await database.transaction((txn) async {
var record = store.record(int.parse(id));
data = await record.put(txn, data, merge: true);
return _withId(data, id);
});
}
@override
Future<Map<String, dynamic>> update(String id, Map<String, dynamic> data,
[Map<String, dynamic> params]) async {
return await database.transaction((txn) async {
var record = store.record(int.parse(id));
data = await record.put(txn, data);
return _withId(data, id);
});
}
@override
Future<Map<String, dynamic>> remove(String id,
[Map<String, dynamic> params]) async {
if (id == null || id == 'null') {
// Remove everything...
if (!(allowRemoveAll == true ||
params?.containsKey('provider') != true)) {
throw AngelHttpException.forbidden(
message: 'Clients are not allowed to delete all items.');
} else {
await store.delete(database);
return {};
}
}
return database.transaction((txn) async {
var record = store.record(int.parse(id));
var snapshot = await record.getSnapshot(txn);
if (snapshot == null) {
throw AngelHttpException.notFound(
message: 'No record found for ID $id');
} else {
await record.delete(txn);
}
return _withId(snapshot.value, id);
});
}
}