# HG changeset patch # User Goffi <goffi@goffi.org> # Date 1516536846 -3600 # Node ID 34240d08f682c1062a2bb9c7c27c27618dc39809 # Parent 78af5457d3f80ecdffd37e89f8091cdc9068dc80 pages: HTTP cache headers handling: when checkCache is used, HTTP headers handling cache are now used: - ETag is first checked, using a hash of the rendered content - Last-Modified is used as a fallback is client is not handling ETag When suitable, a HTTP 304 code (Not Modified) wihtout body is returned instead of the whole page. diff -r 78af5457d3f8 -r 34240d08f682 src/server/constants.py --- a/src/server/constants.py Sun Jan 21 13:08:54 2018 +0100 +++ b/src/server/constants.py Sun Jan 21 13:14:06 2018 +0100 @@ -72,6 +72,7 @@ ## HTTP codes ## HTTP_SEE_OTHER = 303 + HTTP_NOT_MODIFIED = 304 HTTP_BAD_REQUEST = 400 HTTP_UNAUTHORIZED = 401 HTTP_NOT_FOUND = 404 @@ -80,3 +81,7 @@ ## Cache ## CACHE_PUBSUB = 0 + + ## Date/Time ## + HTTP_DAYS = ("Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun") + HTTP_MONTH = ("Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec") diff -r 78af5457d3f8 -r 34240d08f682 src/server/pages.py --- a/src/server/pages.py Sun Jan 21 13:08:54 2018 +0100 +++ b/src/server/pages.py Sun Jan 21 13:14:06 2018 +0100 @@ -26,6 +26,7 @@ from sat.core.i18n import _ from sat.core import exceptions from sat.tools.common import uri as common_uri +from sat.tools import utils from sat.core.log import getLogger log = getLogger(__name__) from libervia.server.constants import Const as C @@ -38,6 +39,7 @@ import os.path import urllib import time +import hashlib WebsocketMeta = namedtuple("WebsocketMeta", ('url', 'token', 'debug')) @@ -69,11 +71,15 @@ self._created = time.time() self._last_access = self._created self._rendered = rendered + self._hash = hashlib.sha256(rendered).hexdigest() @property def rendered(self): return self._rendered + @property + def hash(self): + return self._hash class CacheURL(CacheBase): @@ -488,6 +494,30 @@ request.prepath.append(pathElement) return urllib.unquote(pathElement).decode('utf-8') + ## Cache handling ## + + def _setCacheHeaders(self, request, cache): + """Set ETag and Last-Modified HTTP headers, used for caching""" + request.setHeader('ETag', cache.hash) + last_modified = self.host.getHTTPDate(cache.created) + request.setHeader('Last-Modified', last_modified) + + def _checkCacheHeaders(self, request, cache): + """Check if a cache condition is set on the request + + if condition is valid, C.HTTP_NOT_MODIFIED is returned + """ + etag_match = request.getHeader('If-None-Match') + if etag_match is not None: + if cache.hash == etag_match: + self.pageError(request, C.HTTP_NOT_MODIFIED, no_body=True) + else: + modified_match = request.getHeader('If-Modified-Since') + if modified_match is not None: + modified = utils.date_parse(modified_match) + if modified >= int(cache.created): + self.pageError(request, C.HTTP_NOT_MODIFIED, no_body=True) + def checkCacheSubscribeCb(self, sub_id, service, node): self.cache_pubsub_sub.add((service, node, sub_id)) @@ -509,7 +539,7 @@ C.CACHE_PUBSUB: service: pubsub service node: pubsub node - short: short name of feature (needed if node is empty) + short: short name of feature (needed if node is empty to find namespace) """ if request.postpath: @@ -547,6 +577,8 @@ raise exceptions.InternalError(u'Unknown cache_type') log.debug(u'using cache for {page}'.format(page=self)) cache.last_access = time.time() + self._setCacheHeaders(request, cache) + self._checkCacheHeaders(request, cache) request.write(cache.rendered) request.finish() raise failure.Failure(exceptions.CancelError(u'cache is used')) @@ -696,23 +728,26 @@ redirect_page.renderPage(request, skip_parse_url=skip_parse_url) raise failure.Failure(exceptions.CancelError(u'page redirection is used')) - def pageError(self, request, code=C.HTTP_NOT_FOUND): + def pageError(self, request, code=C.HTTP_NOT_FOUND, no_body=False): """generate an error page and terminate the request @param request(server.Request): HTTP request @param core(int): error code to use + @param no_body: don't write body if True """ - template = u'error/' + unicode(code) + '.html' - request.setResponseCode(code) + if no_body: + request.finish() + else: + template = u'error/' + unicode(code) + '.html' - rendered = self.host.renderer.render( - template, - root_path = '/templates/', - error_code = code, - **request.template_data) + rendered = self.host.renderer.render( + template, + root_path = '/templates/', + error_code = code, + **request.template_data) - self.writeData(rendered, request) + self.writeData(rendered, request) raise failure.Failure(exceptions.CancelError(u'error page is used')) def writeData(self, data, request): @@ -720,15 +755,19 @@ if data is None: self.pageError(request) data_encoded = data.encode('utf-8') - request.write(data_encoded) - request.finish() + if self._do_cache is not None: cache = reduce(lambda d, k: d.setdefault(k, {}), self._do_cache, self.cache) - cache[self] = Cache(data_encoded) + page_cache = cache[self] = CachePage(data_encoded) + self._setCacheHeaders(request, page_cache) log.debug(_(u'{page} put in cache for [{profile}]').format( page=self, profile=self._do_cache[0])) self._do_cache = None + self._checkCacheHeaders(request, page_cache) + + request.write(data_encoded) + request.finish() def _subpagesHandler(self, dummy, request): """render subpage if suitable diff -r 78af5457d3f8 -r 34240d08f682 src/server/server.py --- a/src/server/server.py Sun Jan 21 13:08:54 2018 +0100 +++ b/src/server/server.py Sun Jan 21 13:14:06 2018 +0100 @@ -1909,6 +1909,15 @@ def registerWSToken(self, token, page, request): websockets.LiberviaPageWSProtocol.registerToken(token, page, request) + ## Various utils ## + + def getHTTPDate(self, timestamp=None): + now = time.gmtime(timestamp) + fmt_date = u"{day_name}, %d {month_name} %Y %H:%M:%S GMT".format( + day_name = C.HTTP_DAYS[now.tm_wday], + month_name = C.HTTP_MONTH[now.tm_mon-1]) + return time.strftime(fmt_date, now) + ## TLS related methods ## def _TLSOptionsCheck(self):