Redis example

This commit is contained in:
Tobe O 2019-08-16 11:37:34 -04:00
parent ee92a33ede
commit 92e40d88b8
2 changed files with 22 additions and 18 deletions

View file

@ -1,24 +1,28 @@
import 'package:angel_redis/angel_redis.dart';
import 'package:angel_framework/angel_framework.dart'; import 'package:angel_framework/angel_framework.dart';
import 'package:angel_framework/http.dart'; import 'package:angel_production/angel_production.dart';
import 'package:angel_security/angel_security.dart'; import 'package:angel_security/angel_security.dart';
import 'package:logging/logging.dart'; import 'package:resp_client/resp_client.dart';
import 'package:pretty_logging/pretty_logging.dart'; import 'package:resp_client/resp_commands.dart';
main() async { // We run this through angel_production, so that we can have
// Logging boilerplate. // multiple instances, all using the same Redis queue.
Logger.root.onRecord.listen(prettyLog); main(List<String> args) =>
Runner('rate_limit_redis', configureServer).run(args);
// Create an app, and HTTP driver. configureServer(Angel app) async {
var app = Angel(logger: Logger('rate_limit_redis')), http = AngelHttp(app); // Create a simple rate limiter that limits users to 10
// Create a simple rate limiter that limits users to 5
// queries per 30 seconds. // queries per 30 seconds.
// //
// In this case, we rate limit users by IP address. // In this case, we rate limit users by IP address.
// //
// Our Redis store will be used to manage windows. // Our Redis store will be used to manage windows.
var rateLimiter = var connection = await connectSocket('localhost');
InMemoryRateLimiter(5, Duration(seconds: 30), (req, res) => req.ip); var client = RespClient(connection);
var service =
RedisService(RespCommands(client), prefix: 'rate_limit_redis_example');
var rateLimiter = ServiceRateLimiter(
10, Duration(seconds: 30), service, (req, res) => req.ip);
// `RateLimiter.handleRequest` is a middleware, and can be used anywhere // `RateLimiter.handleRequest` is a middleware, and can be used anywhere
// a middleware can be used. In this case, we apply the rate limiter to // a middleware can be used. In this case, we apply the rate limiter to
@ -27,10 +31,9 @@ main() async {
// Basic routes. // Basic routes.
app app
..get('/', (req, res) => 'Hello!') ..get('/', (req, res) {
var instance = req.container.make<InstanceInfo>();
res.writeln('This is instance ${instance.id}.');
})
..fallback((req, res) => throw AngelHttpException.notFound()); ..fallback((req, res) => throw AngelHttpException.notFound());
// Start the server.
await http.startServer('127.0.0.1', 3000);
print('Rate limiting (redis) example listening at ${http.uri}');
} }

View file

@ -21,7 +21,8 @@ class ServiceRateLimiter<Id> extends RateLimiter<Id> {
FutureOr<RateLimitingWindow<Id>> getCurrentWindow( FutureOr<RateLimitingWindow<Id>> getCurrentWindow(
RequestContext req, ResponseContext res, DateTime currentTime) async { RequestContext req, ResponseContext res, DateTime currentTime) async {
var id = await getId(req, res); var id = await getId(req, res);
var existing = await service.read(id); var existing = await service.read(id).catchError((_) => null,
test: (e) => e is AngelHttpException && e.statusCode == 404);
if (existing != null) { if (existing != null) {
return RateLimitingWindow.fromJson(existing); return RateLimitingWindow.fromJson(existing);
} }