part of angel_mongo.services; /// Manipulates data from MongoDB as Maps. class MongoService extends Service> { DbCollection collection; /// 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; /// No longer used. Will be removed by `2.1.0`. @deprecated final bool debug; MongoService(DbCollection this.collection, {this.allowRemoveAll = false, this.allowQuery = true, this.debug = true}) : super(); SelectorBuilder _makeQuery([Map params_]) { Map params = new Map.from(params_ ?? {}); params = params..remove('provider'); SelectorBuilder result = where.exists('_id'); // You can pass a SelectorBuilder as 'query'; if (params['query'] is SelectorBuilder) { return params['query'] as SelectorBuilder; } for (var key in params.keys) { if (key == r'$sort' || key == r'$query' && (allowQuery == true || !params.containsKey('provider'))) { if (params[key] is Map) { // If they send a map, then we'll sort by every key in the map for (String fieldName in params[key].keys.where((x) => x is String)) { var sorter = params[key][fieldName]; if (sorter is num) { result = result.sortBy(fieldName, descending: sorter == -1); } else if (sorter is String) { result = result.sortBy(fieldName, descending: sorter == "-1"); } else if (sorter is SelectorBuilder) { result = result.and(sorter); } } } else if (params[key] is String && key == r'$sort') { // If they send just a string, then we'll sort // by that, ascending result = result.sortBy(params[key] as String); } } else if (key == 'query' && (allowQuery == true || !params.containsKey('provider'))) { var query = params[key] as Map; query.forEach((key, v) { var value = v is Map ? _filterNoQuery(v) : v; if (!_NO_QUERY.contains(key) && value is! RequestContext && value is! ResponseContext) { result = result.and(where.eq(key as String, value)); } }); } } return result; } Map _jsonify(Map doc, [Map params]) { var result = {}; for (var key in doc.keys) { var value = doc[key]; if (value is ObjectId) { result[key] = value.toHexString(); } else if (value is! RequestContext && value is! ResponseContext) { result[key] = value; } } return _transformId(result); } @override Future>> index( [Map params]) async { return await (await collection.find(_makeQuery(params))) .map((x) => _jsonify(x, params)) .toList(); } static const String _NONCE_KEY = '__angel__mongo__nonce__key__'; @override Future> create(Map data, [Map params]) async { var item = _removeSensitive(data); try { var nonce = (await collection.db.getNonce())['nonce'] as String; var result = await collection.findAndModify( query: where.eq(_NONCE_KEY, nonce), update: item, returnNew: true, upsert: true); return _jsonify(result); } catch (e, st) { throw new AngelHttpException(e, stackTrace: st); } } @override Future> findOne( [Map params, String errorMessage = 'No record was found matching the given query.']) async { var found = await collection.findOne(_makeQuery(params)); if (found == null) { throw new AngelHttpException.notFound(message: errorMessage); } return _jsonify(found, params); } @override Future> read(String id, [Map params]) async { ObjectId _id = _makeId(id); var found = await collection.findOne(where.id(_id).and(_makeQuery(params))); if (found == null) { throw new AngelHttpException.notFound( message: 'No record found for ID ${_id.toHexString()}'); } return _jsonify(found, params); } @override Future>> readMany(List ids, [Map params]) async { var q = _makeQuery(params); q = ids.fold(q, (q, id) => q.or(where.id(_makeId(id)))); return await (await collection.find(q)) .map((x) => _jsonify(x, params)) .toList(); } @override Future> modify(String id, data, [Map params]) async { Map target; try { target = await read(id, params); } on AngelHttpException catch (e) { if (e.statusCode == 404) return await create(data, params); else rethrow; } var result = mergeMap([target, _removeSensitive(data)]); //result['updatedAt'] = new DateTime.now().toIso8601String(); try { var modified = await collection.findAndModify( query: where.id(_makeId(id)), update: result, returnNew: true); result = _jsonify(modified, params); result['id'] = _makeId(id).toHexString(); return result; } catch (e, st) { //printDebug(e, st, 'MODIFY'); throw new AngelHttpException(e, stackTrace: st); } } @override Future> update(String id, Map data, [Map params]) async { var result = _removeSensitive(data); result['_id'] = _makeId(id); /*result['createdAt'] = target is Map ? target['createdAt'] : target.createdAt; if (result['createdAt'] is DateTime) result['createdAt'] = result['createdAt'].toIso8601String(); result['updatedAt'] = new DateTime.now().toIso8601String();*/ try { var updated = await collection.findAndModify( query: where.id(_makeId(id)), update: result, returnNew: true, upsert: true); result = _jsonify(updated, params); result['id'] = _makeId(id).toHexString(); return result; } catch (e, st) { //printDebug(e, st, 'UPDATE'); throw new AngelHttpException(e, stackTrace: st); } } @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 collection.remove(null); return {}; } } // var result = await read(id, params); try { var result = await collection.findAndModify( query: where.id(_makeId(id)), remove: true); return _jsonify(result); } catch (e, st) { //printDebug(e, st, 'REMOVE'); throw new AngelHttpException(e, stackTrace: st); } } }