import 'dart:async'; import 'package:angel3_framework/angel3_framework.dart'; import 'package:sembast/sembast.dart'; class SembastService extends Service> { final Database database; final StoreRef> 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}) : store = intMapStoreFactory.store(store), super(); Finder? _makeQuery([Map? params]) { params = Map.from(params ?? {}); Filter? out; var sort = []; // 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; 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 _withId(Map data, String id) => Map.from(data)..['id'] = id; @override Future> findOne( [Map? params, String errorMessage = 'No record was found matching the given query.']) async { var result = (await store.findFirst(database, finder: _makeQuery(params)))?.value; if (result == null) { return {}; } return result; } @override Future>> index( [Map? 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> read(String id, [Map? 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> create(Map data, [Map? 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> modify(String id, Map data, [Map? 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> update(String id, Map data, [Map? 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> remove(String? id, [Map? 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); }); } }