/// Support for using `angel_validate` with the Angel Framework. library platform_validation.server; import 'dart:async'; import 'package:platform_foundation/core.dart'; import 'src/async.dart'; import 'platform_validation.dart'; export 'src/async.dart'; export 'platform_validation.dart'; /// Auto-parses numbers in `req.bodyAsMap`. RequestHandler autoParseBody(List fields) { return (RequestContext req, res) async { await req.parseBody(); req.bodyAsMap.addAll(autoParse(req.bodyAsMap, fields)); return true; }; } /// Auto-parses numbers in `req.queryParameters`. RequestHandler autoParseQuery(List fields) { return (RequestContext req, res) async { req.queryParameters.addAll(autoParse(req.queryParameters, fields)); return true; }; } /// Filters unwanted data out of `req.bodyAsMap`. RequestHandler filterBody(Iterable only) { return (RequestContext req, res) async { await req.parseBody(); var filtered = filter(req.bodyAsMap, only); req.bodyAsMap ..clear() ..addAll(filtered); return true; }; } /// Filters unwanted data out of `req.queryParameters`. RequestHandler filterQuery(Iterable only) { return (RequestContext req, res) async { var filtered = filter(req.queryParameters, only); req.queryParameters ..clear() ..addAll(filtered); return true; }; } /// Validates the data in `req.bodyAsMap`, and sets the body to /// filtered data before continuing the response. RequestHandler validate(Validator validator, {String errorMessage = 'Invalid data.'}) { return (RequestContext req, res) async { await req.parseBody(); var app = req.app; if (app != null) { var result = await asyncApplyValidator(validator, req.bodyAsMap, app); if (result.errors.isNotEmpty) { throw PlatformHttpException.badRequest( message: errorMessage, errors: result.errors); } req.bodyAsMap ..clear() ..addAll(result.data); } return true; }; } /// Validates the data in `req.queryParameters`, and sets the query to /// filtered data before continuing the response. RequestHandler validateQuery(Validator validator, {String errorMessage = 'Invalid data.'}) { return (RequestContext req, res) async { var app = req.app; if (app != null) { var result = await asyncApplyValidator(validator, req.queryParameters, app); if (result.errors.isNotEmpty) { throw PlatformHttpException.badRequest( message: errorMessage, errors: result.errors); } req.queryParameters ..clear() ..addAll(result.data); } return true; }; } /// Validates the data in `e.data`, and sets the data to /// filtered data before continuing the service event. HookedServiceEventListener validateEvent(Validator validator, {String errorMessage = 'Invalid data.'}) { return (HookedServiceEvent e) async { var app = e.request?.app ?? e.service.app; var result = await asyncApplyValidator(validator, e.data as Map, app); if (result.errors.isNotEmpty) { throw PlatformHttpException.badRequest( message: errorMessage, errors: result.errors); } e.data ..clear() ..addAll(result.data); }; } /// Asynchronously apply a [validator], running any [AngelMatcher]s. Future asyncApplyValidator( Validator validator, Map data, Application app) async { var result = validator.check(data); if (result.errors.isNotEmpty) return result; var errantKeys = [], errors = []; for (var key in result.data.keys) { var value = result.data[key]; var description = StringDescription("'$key': expected "); for (var rule in validator.rules[key]!) { if (rule is AngelMatcher) { var r = await rule.matchesWithAngel(value, key, result.data, {}, app); if (!r) { errors.add(rule.describe(description).toString().trim()); errantKeys.add(key); break; } } } } var m = Map.from(result.data); for (var key in errantKeys) { m.remove(key); } return result.withData(m).withErrors(errors); }