import os import re import socket from BeautifulSoup import BeautifulSoup try: import lxml except ImportError: lxml = None from django.core.cache.backends import dummy from django.core.files.storage import get_storage_class from django.template import Template, Context, TemplateSyntaxError from django.test import TestCase from compressor import base from compressor.cache import get_hashed_mtime from compressor.conf import settings from compressor.css import CssCompressor from compressor.js import JsCompressor from compressor.management.commands.compress import Command as CompressCommand class CompressorTestCase(TestCase): def setUp(self): settings.COMPRESS_ENABLED = True self.css = """ """ self.css_node = CssCompressor(self.css) self.js = """ """ self.js_node = JsCompressor(self.js) def test_css_split(self): out = [ ('file', os.path.join(settings.COMPRESS_ROOT, u'css/one.css'), u''), ('hunk', u'p { border:5px solid green;}', u''), ('file', os.path.join(settings.COMPRESS_ROOT, u'css/two.css'), u''), ] split = self.css_node.split_contents() split = [(x[0], x[1], self.css_node.parser.elem_str(x[2])) for x in split] self.assertEqual(out, split) def test_css_hunks(self): out = ['body { background:#990; }', u'p { border:5px solid green;}', 'body { color:#fff; }'] self.assertEqual(out, list(self.css_node.hunks)) def test_css_output(self): out = u'body { background:#990; }\np { border:5px solid green;}\nbody { color:#fff; }' self.assertEqual(out, self.css_node.combined) def test_css_mtimes(self): is_date = re.compile(r'^\d{10}[\.\d]+$') for date in self.css_node.mtimes: self.assert_(is_date.match(str(float(date))), "mtimes is returning something that doesn't look like a date: %s" % date) def test_css_return_if_off(self): settings.COMPRESS_ENABLED = False self.assertEqual(self.css, self.css_node.output()) def test_cachekey(self): host_name = socket.gethostname() is_cachekey = re.compile(r'django_compressor\.%s\.\w{12}' % host_name) self.assert_(is_cachekey.match(self.css_node.cachekey), "cachekey is returning something that doesn't look like r'django_compressor\.%s\.\w{12}'" % host_name) def test_css_hash(self): self.assertEqual('f7c661b7a124', self.css_node.hash) def test_css_return_if_on(self): output = u'' self.assertEqual(output, self.css_node.output().strip()) def test_js_split(self): out = [('file', os.path.join(settings.COMPRESS_ROOT, u'js/one.js'), ''), ('hunk', u'obj.value = "value";', '') ] split = self.js_node.split_contents() split = [(x[0], x[1], self.js_node.parser.elem_str(x[2])) for x in split] self.assertEqual(out, split) def test_js_hunks(self): out = ['obj = {};', u'obj.value = "value";'] self.assertEqual(out, list(self.js_node.hunks)) def test_js_concat(self): out = u'obj = {};\nobj.value = "value";' self.assertEqual(out, self.js_node.concat()) def test_js_output(self): out = u'obj={};obj.value="value";' self.assertEqual(out, self.js_node.combined) def test_js_return_if_off(self): try: enabled = settings.COMPRESS_ENABLED settings.COMPRESS_ENABLED = False self.assertEqual(self.js, self.js_node.output()) finally: settings.COMPRESS_ENABLED = enabled def test_js_return_if_on(self): output = u'' self.assertEqual(output, self.js_node.output()) def test_custom_output_dir(self): try: old_output_dir = settings.COMPRESS_OUTPUT_DIR settings.COMPRESS_OUTPUT_DIR = 'custom' output = u'' self.assertEqual(output, JsCompressor(self.js).output()) settings.COMPRESS_OUTPUT_DIR = '' output = u'' self.assertEqual(output, JsCompressor(self.js).output()) settings.COMPRESS_OUTPUT_DIR = '/custom/nested/' output = u'' self.assertEqual(output, JsCompressor(self.js).output()) finally: settings.COMPRESS_OUTPUT_DIR = old_output_dir if lxml: class LxmlCompressorTestCase(CompressorTestCase): def test_css_split(self): out = [ ('file', os.path.join(settings.COMPRESS_ROOT, u'css/one.css'), u''), ('hunk', u'p { border:5px solid green;}', u''), ('file', os.path.join(settings.COMPRESS_ROOT, u'css/two.css'), u''), ] split = self.css_node.split_contents() split = [(x[0], x[1], self.css_node.parser.elem_str(x[2])) for x in split] self.assertEqual(out, split) def setUp(self): self.old_parser = settings.COMPRESS_PARSER settings.COMPRESS_PARSER = 'compressor.parser.LxmlParser' super(LxmlCompressorTestCase, self).setUp() def tearDown(self): settings.COMPRESS_PARSER = self.old_parser class CssAbsolutizingTestCase(TestCase): def setUp(self): settings.COMPRESS_ENABLED = True settings.COMPRESS_URL = '/media/' self.css = """ """ self.css_node = CssCompressor(self.css) def test_css_absolute_filter(self): from compressor.filters.css_default import CssAbsoluteFilter filename = os.path.join(settings.COMPRESS_ROOT, 'css/url/test.css') content = "p { background: url('../../images/image.gif') }" output = "p { background: url('%simages/image.gif?%s') }" % (settings.COMPRESS_URL, get_hashed_mtime(filename)) filter = CssAbsoluteFilter(content) self.assertEqual(output, filter.input(filename=filename)) settings.COMPRESS_URL = 'http://media.example.com/' filename = os.path.join(settings.COMPRESS_ROOT, 'css/url/test.css') output = "p { background: url('%simages/image.gif?%s') }" % (settings.COMPRESS_URL, get_hashed_mtime(filename)) 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.COMPRESS_ROOT, 'css/url/test.css') content = "p { background: url('../../images/image.gif') }" output = "p { background: url('%simages/image.gif?%s') }" % (settings.COMPRESS_URL, get_hashed_mtime(filename)) filter = CssAbsoluteFilter(content) self.assertEqual(output, filter.input(filename=filename)) settings.COMPRESS_URL = 'https://media.example.com/' filename = os.path.join(settings.COMPRESS_ROOT, 'css/url/test.css') output = "p { background: url('%simages/image.gif?%s') }" % (settings.COMPRESS_URL, get_hashed_mtime(filename)) 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(settings.TEST_DIR, 'whatever', '..', 'media', 'whatever/../css/url/test.css') content = "p { background: url('../../images/image.gif') }" output = "p { background: url('%simages/image.gif?%s') }" % (settings.COMPRESS_URL, get_hashed_mtime(filename)) filter = CssAbsoluteFilter(content) self.assertEqual(output, filter.input(filename=filename)) settings.COMPRESS_URL = 'https://media.example.com/' output = "p { background: url('%simages/image.gif?%s') }" % (settings.COMPRESS_URL, get_hashed_mtime(filename)) self.assertEqual(output, filter.input(filename=filename)) def test_css_hunks(self): hash_dict = { 'hash1': get_hashed_mtime(os.path.join(settings.COMPRESS_ROOT, 'css/url/url1.css')), 'hash2': get_hashed_mtime(os.path.join(settings.COMPRESS_ROOT, 'css/url/2/url2.css')), } out = [u"p { background: url('/media/images/test.png?%(hash1)s'); }\np { background: url('/media/images/test.png?%(hash1)s'); }\np { background: url('/media/images/test.png?%(hash1)s'); }\np { background: url('/media/images/test.png?%(hash1)s'); }\n" % hash_dict, u"p { background: url('/media/images/test.png?%(hash2)s'); }\np { background: url('/media/images/test.png?%(hash2)s'); }\np { background: url('/media/images/test.png?%(hash2)s'); }\np { background: url('/media/images/test.png?%(hash2)s'); }\n" % hash_dict] self.assertEqual(out, list(self.css_node.hunks)) class CssDataUriTestCase(TestCase): def setUp(self): settings.COMPRESS_ENABLED = True settings.COMPRESS_CSS_FILTERS = [ 'compressor.filters.css_default.CssAbsoluteFilter', 'compressor.filters.datauri.CssDataUriFilter', ] settings.COMPRESS_URL = '/media/' self.css = """ """ self.css_node = CssCompressor(self.css) def test_data_uris(self): datauri_hash = get_hashed_mtime(os.path.join(settings.COMPRESS_ROOT, 'css/datauri.css')) out = [u'.add { background-image: url(""); }\n.python { background-image: url("/media/img/python.png?%s"); }\n.datauri { background-image: url(" vr4MkhoXe0rZigAAAABJRU5ErkJggg=="); }\n' % datauri_hash] self.assertEqual(out, list(self.css_node.hunks)) class CssMediaTestCase(TestCase): def setUp(self): self.css = """ """ self.css_node = CssCompressor(self.css) def test_css_output(self): links = BeautifulSoup(self.css_node.output()).findAll('link') media = [u'screen', u'print', u'all', None] self.assertEqual(len(links), 4) self.assertEqual(media, [l.get('media', None) for l in links]) def test_avoid_reordering_css(self): css = self.css + '' node = CssCompressor(css) media = [u'screen', u'print', u'all', None, u'print'] links = BeautifulSoup(node.output()).findAll('link') self.assertEqual(media, [l.get('media', None) for l in links]) class CssMinTestCase(TestCase): def test_cssmin_filter(self): from compressor.filters.cssmin import CSSMinFilter content = """p { background: rgb(51,102,153) url('../../images/image.gif'); } """ output = "p{background:#369 url('../../images/image.gif')}" self.assertEqual(output, CSSMinFilter(content).output()) def render(template_string, context_dict=None): """A shortcut for testing template output.""" if context_dict is None: context_dict = {} c = Context(context_dict) t = Template(template_string) return t.render(c).strip() class TemplatetagTestCase(TestCase): def setUp(self): settings.COMPRESS_ENABLED = True def test_empty_tag(self): template = u"""{% load compress %}{% compress js %}{% block js %} {% endblock %}{% endcompress %}""" context = { 'MEDIA_URL': settings.COMPRESS_URL } self.assertEqual(u'', render(template, context)) def test_css_tag(self): template = u"""{% load compress %}{% compress css %} {% endcompress %} """ context = { 'MEDIA_URL': settings.COMPRESS_URL } out = u'' self.assertEqual(out, render(template, context)) def test_nonascii_css_tag(self): template = u"""{% load compress %}{% compress css %} {% endcompress %} """ context = { 'MEDIA_URL': settings.COMPRESS_URL } out = '' self.assertEqual(out, render(template, context)) def test_js_tag(self): template = u"""{% load compress %}{% compress js %} {% endcompress %} """ context = { 'MEDIA_URL': settings.COMPRESS_URL } out = u'' self.assertEqual(out, render(template, context)) def test_nonascii_js_tag(self): template = u"""{% load compress %}{% compress js %} {% endcompress %} """ context = { 'MEDIA_URL': settings.COMPRESS_URL } out = u'' self.assertEqual(out, render(template, context)) def test_nonascii_latin1_js_tag(self): template = u"""{% load compress %}{% compress js %} {% endcompress %} """ context = { 'MEDIA_URL': settings.COMPRESS_URL } out = u'' self.assertEqual(out, render(template, context)) def test_compress_tag_with_illegal_arguments(self): template = u"""{% load compress %}{% compress pony %} {% endcompress %}""" self.assertRaises(TemplateSyntaxError, render, template, {}) class StorageTestCase(TestCase): def setUp(self): self._storage = base.default_storage base.default_storage = get_storage_class('compressor.storage.GzipCompressorFileStorage')() settings.COMPRESS_ENABLED = True def tearDown(self): base.default_storage = self._storage def test_css_tag_with_storage(self): template = u"""{% load compress %}{% compress css %} {% endcompress %} """ context = { 'MEDIA_URL': settings.COMPRESS_URL } out = u'' self.assertEqual(out, render(template, context)) class VerboseTestCase(CompressorTestCase): def setUp(self): super(VerboseTestCase, self).setUp() settings.COMPRESS_VERBOSE = True class CacheBackendTestCase(CompressorTestCase): def test_correct_backend(self): from compressor.cache import cache self.assertEqual(cache.__class__, dummy.CacheClass) class OfflineGenerationTestCase(TestCase): """Uses templates/test_compressor_offline.html""" def setUp(self): self._old_compress = settings.COMPRESS_ENABLED settings.COMPRESS_ENABLED = True def tearDown(self): settings.COMPRESS_ENABLED = self._old_compress def test_offline(self): count, result = CompressCommand().compress() self.assertEqual(2, count) self.assertEqual(result, [ u'\n', u'', ]) def test_offline_with_context(self): self._old_offline_context = settings.COMPRESS_OFFLINE_CONTEXT settings.COMPRESS_OFFLINE_CONTEXT = { 'color': 'blue', } count, result = CompressCommand().compress() self.assertEqual(2, count) self.assertEqual(result, [ u'\n', u'', ]) settings.COMPRESS_OFFLINE_CONTEXT = self._old_offline_context