platform/lib/src/cache.dart

99 lines
3.1 KiB
Dart
Raw Normal View History

2018-04-02 01:05:35 +00:00
import 'dart:async';
import 'package:angel_framework/angel_framework.dart';
import 'package:intl/intl.dart';
2018-04-02 01:37:34 +00:00
import 'package:pool/pool.dart';
2018-04-02 01:05:35 +00:00
final DateFormat _fmt = new DateFormat('EEE, d MMM yyyy HH:mm:ss');
/// Formats a date (converted to UTC), ex: `Sun, 03 May 2015 23:02:37 GMT`.
String _formatDateForHttp(DateTime dt) => _fmt.format(dt.toUtc()) + ' GMT';
class ResponseCache {
final List<Pattern> patterns = [];
final Duration timeout;
final Map<String, _CachedResponse> _cache = {};
ResponseCache({this.timeout});
2018-04-02 01:37:34 +00:00
/// Removes an entry from the response cache.
void invalidate(String path) => _cache.remove(path);
2018-04-02 01:24:13 +00:00
/// A middleware that handles requests with an `If-Modified-Since` header.
///
/// This prevents the server from even having to access the cache, and plays very well with static assets.
Future<bool> ifModifiedSince(RequestContext req, ResponseContext res) async {
if (req.headers.value('if-modified-since') != null) {
2018-04-02 01:37:34 +00:00
var modifiedSince = _fmt
.parse(req.headers.value('if-modified-since').replaceAll('GMT', ''));
2018-04-02 01:05:35 +00:00
2018-04-02 01:24:13 +00:00
// Check if there is a cache entry.
for (var pattern in patterns) {
if (pattern.allMatches(req.uri.path).isNotEmpty &&
_cache.containsKey(req.uri.path)) {
var response = _cache[req.uri.path];
2018-04-02 01:05:35 +00:00
2018-04-02 01:27:56 +00:00
if (response.timestamp.compareTo(modifiedSince) <= 0) {
2018-04-02 01:24:13 +00:00
res.statusCode = 304;
return false;
}
2018-04-02 01:05:35 +00:00
}
}
}
return true;
}
2018-04-02 01:37:34 +00:00
/// A response finalizer that saves responses to the cache.
2018-04-02 01:24:13 +00:00
Future<bool> responseFinalizer(
RequestContext req, ResponseContext res) async {
if (res.statusCode == 304) return true;
2018-04-02 01:37:34 +00:00
// Check if there is a cache entry.
for (var pattern in patterns) {
if (pattern.allMatches(req.uri.path).isNotEmpty) {
var now = new DateTime.now().toUtc();
// Invalidate the response, if need be.
if (_cache.containsKey(req.uri.path)) {
// If there is no timeout, don't invalidate.
if (timeout == null) return true;
// Otherwise, don't invalidate unless the timeout has been exceeded.
var response = _cache[req.uri.path];
if (now.difference(response.timestamp) < timeout) return true;
// If the cache entry should be invalidated, then invalidate it.
invalidate(req.uri.path);
}
2018-04-02 01:40:08 +00:00
// Save the response.
_cache[req.uri.path] = new _CachedResponse(res.headers, res.buffer.toBytes(), now);
2018-04-02 01:37:34 +00:00
}
2018-04-02 01:40:08 +00:00
return true;
2018-04-02 01:37:34 +00:00
}
2018-04-02 01:05:35 +00:00
}
void setCachedHeaders(
DateTime modified, RequestContext req, ResponseContext res) {
var privacy = 'public';
res.headers
..['cache-control'] = '$privacy, max-age=${timeout?.inSeconds ?? 0}'
..['last-modified'] = _formatDateForHttp(modified);
if (timeout != null) {
var expiry = new DateTime.now().add(timeout);
res.headers['expires'] = _formatDateForHttp(expiry);
}
}
}
class _CachedResponse {
final Map<String, String> headers;
final List<int> body;
final DateTime timestamp;
2018-04-02 01:40:08 +00:00
_CachedResponse(this.headers, this.body, this.timestamp);
2018-04-02 01:24:13 +00:00
}