Skip to content

Commit

Permalink
Added cache busting ability to CssAbsoluteFilter filter which appends…
Browse files Browse the repository at this point in the history
… a hash to each image URL (except data: images).
  • Loading branch information
jezdez committed May 7, 2010
1 parent c06e88e commit dbaf995
Show file tree
Hide file tree
Showing 3 changed files with 36 additions and 15 deletions.
29 changes: 23 additions & 6 deletions compressor/filters/css_default.py
Expand Up @@ -4,6 +4,7 @@

from compressor.filters import FilterBase, FilterError
from compressor.conf import settings
from compressor import get_hexdigest

class CssAbsoluteFilter(FilterBase):
def input(self, filename=None, **kwargs):
Expand All @@ -15,6 +16,11 @@ def input(self, filename=None, **kwargs):
self.media_path = filename[len(media_root):]
self.media_path = self.media_path.lstrip('/')
self.media_url = settings.MEDIA_URL.rstrip('/')
try:
mtime = os.path.getmtime(filename)
self.mtime = get_hexdigest(str(int(mtime)))[:12]
except OSError:
self.mtime = None
self.has_http = False
if self.media_url.startswith('http://') or self.media_url.startswith('https://'):
self.has_http = True
Expand All @@ -26,16 +32,27 @@ def input(self, filename=None, **kwargs):
output = url_pattern.sub(self.url_converter, self.content)
return output

def add_mtime(self, url):
if self.mtime is None:
return url
if (url.startswith('http://') or
url.startswith('https://') or
url.startswith('/')):
if "?" in url:
return "%s&%s" % (url, self.mtime)
return "%s?%s" % (url, self.mtime)
return url

def url_converter(self, matchobj):
url = matchobj.group(1)
url = url.strip(' \'"')
if (url.startswith('http://') or
url.startswith('https://') or
url.startswith('/') or
if (url.startswith('http://') or
url.startswith('https://') or
url.startswith('/') or
url.startswith('data:')):
return "url('%s')" % url
return "url('%s')" % self.add_mtime(url)
full_url = '/'.join([str(self.directory_name), url])
full_url = posixpath.normpath(full_url)
if self.has_http:
full_url = "%s%s" % (self.protocol,full_url)
return "url('%s')" % full_url
full_url = "%s%s" % (self.protocol, full_url)
return "url('%s')" % self.add_mtime(full_url)
4 changes: 4 additions & 0 deletions compressor/filters/datauri.py
@@ -1,6 +1,7 @@
import os
import re
import mimetypes
import urlparse
from base64 import b64encode
from compressor.filters import FilterBase
from compressor.conf import settings
Expand All @@ -25,6 +26,9 @@ def input(self, filename=None, **kwargs):
return output

def get_file_path(self, url):
# strip query string of file paths
if "?" in url:
url = url.split("?")[0]
return os.path.join(settings.MEDIA_ROOT, url[len(settings.MEDIA_URL):])

def data_uri_converter(self, matchobj):
Expand Down
18 changes: 9 additions & 9 deletions tests/core/tests.py
Expand Up @@ -122,41 +122,41 @@ def test_css_absolute_filter(self):
from compressor.filters.css_default import CssAbsoluteFilter
filename = os.path.join(settings.MEDIA_ROOT, 'css/url/test.css')
content = "p { background: url('../../images/image.gif') }"
output = "p { background: url('%simages/image.gif') }" % settings.MEDIA_URL
output = "p { background: url('%simages/image.gif?275088b9bcf0') }" % settings.MEDIA_URL
filter = CssAbsoluteFilter(content)
self.assertEqual(output, filter.input(filename=filename))
settings.MEDIA_URL = 'http://media.example.com/'
filename = os.path.join(settings.MEDIA_ROOT, 'css/url/test.css')
output = "p { background: url('%simages/image.gif') }" % settings.MEDIA_URL
output = "p { background: url('%simages/image.gif?275088b9bcf0') }" % settings.MEDIA_URL
self.assertEqual(output, filter.input(filename=filename))

def test_css_absolute_filter_https(self):
from compressor.filters.css_default import CssAbsoluteFilter
filename = os.path.join(settings.MEDIA_ROOT, 'css/url/test.css')
content = "p { background: url('../../images/image.gif') }"
output = "p { background: url('%simages/image.gif') }" % settings.MEDIA_URL
output = "p { background: url('%simages/image.gif?275088b9bcf0') }" % settings.MEDIA_URL
filter = CssAbsoluteFilter(content)
self.assertEqual(output, filter.input(filename=filename))
settings.MEDIA_URL = 'https://media.example.com/'
filename = os.path.join(settings.MEDIA_ROOT, 'css/url/test.css')
output = "p { background: url('%simages/image.gif') }" % settings.MEDIA_URL
output = "p { background: url('%simages/image.gif?275088b9bcf0') }" % settings.MEDIA_URL
self.assertEqual(output, filter.input(filename=filename))

def test_css_absolute_filter_relative_path(self):
from compressor.filters.css_default import CssAbsoluteFilter
filename = os.path.join(django_settings.TEST_DIR, 'whatever', '..', 'media', 'whatever/../css/url/test.css')
content = "p { background: url('../../images/image.gif') }"
output = "p { background: url('%simages/image.gif') }" % settings.MEDIA_URL
output = "p { background: url('%simages/image.gif?275088b9bcf0') }" % settings.MEDIA_URL
filter = CssAbsoluteFilter(content)
self.assertEqual(output, filter.input(filename=filename))
settings.MEDIA_URL = 'https://media.example.com/'
output = "p { background: url('%simages/image.gif') }" % settings.MEDIA_URL
output = "p { background: url('%simages/image.gif?275088b9bcf0') }" % settings.MEDIA_URL
self.assertEqual(output, filter.input(filename=filename))


def test_css_hunks(self):
out = [u"p { background: url('/media/images/test.png'); }\np { background: url('/media/images/test.png'); }\np { background: url('/media/images/test.png'); }\np { background: url('/media/images/test.png'); }\n",
u"p { background: url('/media/images/test.png'); }\np { background: url('/media/images/test.png'); }\np { background: url('/media/images/test.png'); }\np { background: url('/media/images/test.png'); }\n",
out = [u"p { background: url('/media/images/test.png?86ea855a37ed'); }\np { background: url('/media/images/test.png?86ea855a37ed'); }\np { background: url('/media/images/test.png?86ea855a37ed'); }\np { background: url('/media/images/test.png?86ea855a37ed'); }\n",
u"p { background: url('/media/images/test.png?86ea855a37ed'); }\np { background: url('/media/images/test.png?86ea855a37ed'); }\np { background: url('/media/images/test.png?86ea855a37ed'); }\np { background: url('/media/images/test.png?86ea855a37ed'); }\n",
]
self.assertEqual(out, self.cssNode.hunks)

Expand All @@ -172,7 +172,7 @@ def setUp(self):
self.cssNode = CssCompressor(self.css)

def test_data_uris(self):
out = [u'.add { background-image: url(""); }\n.python { background-image: url("/media/img/python.png"); }\n.datauri { background-image: url(" vr4MkhoXe0rZigAAAABJRU5ErkJggg=="); }\n']
out = [u'.add { background-image: url(""); }\n.python { background-image: url("/media/img/python.png?f59031cef54b"); }\n.datauri { background-image: url(" vr4MkhoXe0rZigAAAABJRU5ErkJggg=="); }\n']
self.assertEqual(out, self.cssNode.hunks)


Expand Down

0 comments on commit dbaf995

Please sign in to comment.