2010-02-06 01:06:16 +03:00
|
|
|
# -*- coding: utf-8 -*-
|
2010-04-01 00:55:24 +04:00
|
|
|
from django import http
|
2010-02-06 01:06:16 +03:00
|
|
|
from django.core.cache import cache
|
|
|
|
|
2010-04-01 00:55:24 +04:00
|
|
|
from mock import patch
|
2010-02-06 01:06:16 +03:00
|
|
|
from nose.tools import eq_, assert_raises
|
2010-05-05 20:56:10 +04:00
|
|
|
from pyquery import PyQuery as pq
|
2010-02-06 01:06:16 +03:00
|
|
|
|
|
|
|
import test_utils
|
|
|
|
|
2010-02-10 22:35:25 +03:00
|
|
|
import amo
|
2010-05-21 22:01:11 +04:00
|
|
|
import amo.tests
|
2010-02-06 01:06:16 +03:00
|
|
|
from amo.urlresolvers import reverse
|
2010-02-10 22:35:25 +03:00
|
|
|
from amo.helpers import urlparams
|
2010-04-01 00:55:24 +04:00
|
|
|
from addons.models import Addon, Category
|
|
|
|
from browse import views
|
2010-02-06 01:06:16 +03:00
|
|
|
from browse.views import locale_display_name
|
|
|
|
|
|
|
|
|
|
|
|
def test_locale_display_name():
|
2010-02-17 03:11:30 +03:00
|
|
|
|
2010-02-06 01:06:16 +03:00
|
|
|
def check(locale, english, native):
|
|
|
|
actual = locale_display_name(locale)
|
|
|
|
eq_(actual, (english, native))
|
|
|
|
|
|
|
|
check('el', 'Greek', u'Ελληνικά')
|
2010-05-05 21:21:50 +04:00
|
|
|
check('el-XX', 'Greek', u'Ελληνικά')
|
2010-02-06 01:06:16 +03:00
|
|
|
assert_raises(KeyError, check, 'fake-lang', '', '')
|
|
|
|
|
|
|
|
|
2010-05-21 22:01:11 +04:00
|
|
|
class TestLanguageTools(amo.tests.TestCase):
|
2010-02-06 01:06:16 +03:00
|
|
|
fixtures = ['browse/test_views']
|
|
|
|
|
|
|
|
def setUp(self):
|
2010-05-21 22:01:11 +04:00
|
|
|
super(TestLanguageTools, self).setUp()
|
2010-02-06 01:06:16 +03:00
|
|
|
cache.clear()
|
2010-05-19 05:24:20 +04:00
|
|
|
self.url = reverse('browse.language-tools')
|
2010-02-17 02:22:39 +03:00
|
|
|
response = self.client.get(self.url, follow=True)
|
2010-02-06 01:06:16 +03:00
|
|
|
self.locales = response.context['locales']
|
|
|
|
|
|
|
|
def test_sorting(self):
|
|
|
|
"""The locales should be sorted by English display name."""
|
2010-05-05 21:21:50 +04:00
|
|
|
displays = [locale.display for _, locale in self.locales]
|
2010-02-06 01:06:16 +03:00
|
|
|
eq_(displays, sorted(displays))
|
|
|
|
|
|
|
|
def test_native_missing_region(self):
|
|
|
|
"""
|
|
|
|
If we had to strip a locale's region to find a display name, we
|
|
|
|
append it to the native name for disambiguation.
|
|
|
|
"""
|
|
|
|
el = dict(self.locales)['el-XX']
|
|
|
|
assert el.native.endswith(' (el-xx)')
|
|
|
|
|
|
|
|
def test_missing_locale(self):
|
|
|
|
"""If we don't know about a locale, show the addon name and locale."""
|
|
|
|
wa = dict(self.locales)['wa']
|
|
|
|
eq_(wa.display, 'Walloon Language Pack (wa)')
|
|
|
|
eq_(wa.native, '')
|
|
|
|
|
|
|
|
def test_packs_and_dicts(self):
|
|
|
|
ca = dict(self.locales)['ca-valencia']
|
|
|
|
eq_(len(ca.dicts), 1)
|
|
|
|
eq_(len(ca.packs), 2)
|
2010-02-10 22:35:25 +03:00
|
|
|
|
2010-02-17 02:22:39 +03:00
|
|
|
def test_empty_target_locale(self):
|
|
|
|
"""Make sure nothing breaks with empty target locales."""
|
|
|
|
for addon in Addon.objects.all():
|
|
|
|
addon.target_locale = ''
|
|
|
|
addon.save()
|
|
|
|
response = self.client.get(self.url, follow=True)
|
|
|
|
eq_(response.status_code, 200)
|
|
|
|
eq_(response.context['locales'], [])
|
|
|
|
|
|
|
|
def test_null_target_locale(self):
|
|
|
|
"""Make sure nothing breaks with null target locales."""
|
|
|
|
for addon in Addon.objects.all():
|
|
|
|
addon.target_locale = None
|
|
|
|
addon.save()
|
|
|
|
response = self.client.get(self.url, follow=True)
|
|
|
|
eq_(response.status_code, 200)
|
|
|
|
eq_(response.context['locales'], [])
|
|
|
|
|
2010-02-10 22:35:25 +03:00
|
|
|
|
|
|
|
class TestThemes(test_utils.TestCase):
|
2010-05-06 00:17:12 +04:00
|
|
|
fixtures = ['base/fixtures']
|
2010-02-10 22:35:25 +03:00
|
|
|
|
|
|
|
def setUp(self):
|
2010-05-19 05:24:20 +04:00
|
|
|
super(TestThemes, self).setUp()
|
2010-02-10 22:35:25 +03:00
|
|
|
# Make all the add-ons themes.
|
|
|
|
for addon in Addon.objects.all():
|
|
|
|
addon.type_id = amo.ADDON_THEME
|
|
|
|
addon.save()
|
2010-05-18 03:35:17 +04:00
|
|
|
for category in Category.objects.all():
|
|
|
|
category.type_id = amo.ADDON_THEME
|
|
|
|
category.save()
|
2010-02-10 22:35:25 +03:00
|
|
|
|
|
|
|
self.base_url = reverse('browse.themes')
|
2010-04-14 08:22:29 +04:00
|
|
|
self.exp_url = urlparams(self.base_url, unreviewed=True)
|
2010-02-10 22:35:25 +03:00
|
|
|
|
|
|
|
def test_default_sort(self):
|
|
|
|
"""Default sort should be by downloads."""
|
|
|
|
response = self.client.get(self.base_url)
|
|
|
|
eq_(response.context['sorting'], 'downloads')
|
|
|
|
|
2010-04-14 08:22:29 +04:00
|
|
|
def test_unreviewed(self):
|
|
|
|
# Only 3 without unreviewed.
|
2010-02-10 22:35:25 +03:00
|
|
|
response = self.client.get(self.base_url)
|
2010-03-12 02:16:12 +03:00
|
|
|
eq_(len(response.context['themes'].object_list), 8)
|
2010-02-10 22:35:25 +03:00
|
|
|
|
|
|
|
response = self.client.get(self.exp_url)
|
2010-03-12 02:16:12 +03:00
|
|
|
eq_(len(response.context['themes'].object_list), 10)
|
2010-02-10 22:35:25 +03:00
|
|
|
|
|
|
|
def _get_sort(self, sort):
|
|
|
|
response = self.client.get(urlparams(self.exp_url, sort=sort))
|
|
|
|
eq_(response.context['sorting'], sort)
|
|
|
|
return [a.id for a in response.context['themes'].object_list]
|
|
|
|
|
|
|
|
def test_download_sort(self):
|
|
|
|
ids = self._get_sort('downloads')
|
2010-03-12 02:16:12 +03:00
|
|
|
eq_(ids, [55, 1843, 73, 3615, 5369, 7172, 6113, 10869, 6704, 40])
|
2010-02-10 22:35:25 +03:00
|
|
|
|
|
|
|
def test_name_sort(self):
|
|
|
|
ids = self._get_sort('name')
|
2010-03-12 02:16:12 +03:00
|
|
|
eq_(ids, [55, 3615, 1843, 6704, 10869, 7172, 40, 5369, 73, 6113])
|
2010-02-10 22:35:25 +03:00
|
|
|
|
2010-03-24 04:03:47 +03:00
|
|
|
def test_created_sort(self):
|
|
|
|
ids = self._get_sort('created')
|
|
|
|
eq_(ids, [10869, 7172, 6704, 6113, 5369, 3615, 55, 73, 1843, 40])
|
|
|
|
|
|
|
|
def test_updated_sort(self):
|
|
|
|
ids = self._get_sort('updated')
|
2010-03-12 02:16:12 +03:00
|
|
|
eq_(ids, [6113, 3615, 7172, 5369, 10869, 6704, 1843, 73, 40, 55])
|
2010-02-10 22:35:25 +03:00
|
|
|
|
|
|
|
def test_rating_sort(self):
|
|
|
|
ids = self._get_sort('rating')
|
2010-03-12 02:16:12 +03:00
|
|
|
eq_(ids, [6113, 7172, 1843, 6704, 10869, 40, 5369, 3615, 55, 73])
|
2010-04-01 00:55:24 +04:00
|
|
|
|
2010-05-18 03:35:17 +04:00
|
|
|
def test_category_count(self):
|
2010-05-19 05:24:20 +04:00
|
|
|
cat = Category.objects.all()[0]
|
2010-05-18 03:35:17 +04:00
|
|
|
response = self.client.get(reverse('browse.themes', args=[cat.slug]))
|
|
|
|
doc = pq(response.content)
|
|
|
|
actual_count = int(doc('hgroup h3').text().split()[0])
|
|
|
|
page = response.context['themes']
|
|
|
|
expected_count = page.paginator.count
|
|
|
|
eq_(actual_count, expected_count)
|
|
|
|
|
2010-04-01 00:55:24 +04:00
|
|
|
|
|
|
|
class TestCategoryPages(test_utils.TestCase):
|
2010-05-06 00:17:12 +04:00
|
|
|
fixtures = ['base/fixtures']
|
2010-04-01 00:55:24 +04:00
|
|
|
|
2010-04-19 21:06:59 +04:00
|
|
|
def test_browsing_urls(self):
|
|
|
|
"""Every browse page URL exists."""
|
|
|
|
for _, slug in amo.ADDON_SLUGS.items():
|
2010-05-19 05:24:20 +04:00
|
|
|
assert reverse('browse.%s' % slug)
|
2010-04-19 21:06:59 +04:00
|
|
|
|
2010-04-01 00:55:24 +04:00
|
|
|
def test_matching_opts(self):
|
|
|
|
"""Every filter on landing pages is available on listing pages."""
|
|
|
|
for key, _ in views.CategoryLandingFilter.opts:
|
|
|
|
if key != 'featured':
|
|
|
|
assert key in dict(views.AddonFilter.opts)
|
|
|
|
|
|
|
|
@patch('browse.views.category_landing')
|
|
|
|
def test_goto_category_landing(self, landing_mock):
|
|
|
|
"""We hit a landing page if there's a category and no sorting."""
|
|
|
|
landing_mock.return_value = http.HttpResponse()
|
|
|
|
|
|
|
|
self.client.get(reverse('browse.extensions'))
|
|
|
|
assert not landing_mock.called
|
|
|
|
|
|
|
|
slug = Category.objects.all()[0].slug
|
|
|
|
category_url = reverse('browse.extensions', args=[slug])
|
|
|
|
self.client.get('%s?sort=created' % category_url)
|
|
|
|
assert not landing_mock.called
|
|
|
|
|
|
|
|
self.client.get(category_url)
|
|
|
|
assert landing_mock.called
|
2010-04-09 05:42:02 +04:00
|
|
|
|
|
|
|
def test_creatured_addons(self):
|
|
|
|
"""Make sure the creatured add-ons are for the right category."""
|
|
|
|
# Featured in bookmarks.
|
|
|
|
url = reverse('browse.extensions', args=['bookmarks'])
|
|
|
|
response = self.client.get(url, follow=True)
|
|
|
|
creatured = response.context['filter'].all()['featured']
|
|
|
|
eq_(len(creatured), 1)
|
|
|
|
eq_(creatured[0].id, 3615)
|
|
|
|
|
|
|
|
# Not featured in search-tools.
|
|
|
|
url = reverse('browse.extensions', args=['search-tools'])
|
|
|
|
response = self.client.get(url, follow=True)
|
|
|
|
creatured = response.context['filter'].all()['featured']
|
|
|
|
eq_(len(creatured), 0)
|
2010-05-05 20:56:10 +04:00
|
|
|
|
|
|
|
def test_added_date(self):
|
|
|
|
url = reverse('browse.extensions') + '?sort=created'
|
|
|
|
doc = pq(self.client.get(url).content)
|
|
|
|
s = doc('.featured .item .updated').text()
|
|
|
|
assert s.strip().startswith('Added'), s
|
2010-05-19 05:24:20 +04:00
|
|
|
|
|
|
|
|
|
|
|
class TestLegacyRedirects(test_utils.TestCase):
|
|
|
|
fixtures = ['base/fixtures']
|
|
|
|
|
|
|
|
def test_types(self):
|
|
|
|
def redirects(from_, to):
|
2010-05-21 22:01:11 +04:00
|
|
|
r = self.client.get('/en-US/firefox' + from_)
|
|
|
|
self.assertRedirects(r, '/en-US/firefox' + to, status_code=301)
|
2010-05-19 05:24:20 +04:00
|
|
|
|
|
|
|
redirects('/browse/type:1', '/extensions/')
|
|
|
|
redirects('/browse/type:1/', '/extensions/')
|
|
|
|
redirects('/browse/type:1/cat:all', '/extensions/')
|
|
|
|
redirects('/browse/type:1/cat:all/', '/extensions/')
|
|
|
|
redirects('/browse/type:1/cat:72', '/extensions/alerts-updates/')
|
|
|
|
redirects('/browse/type:1/cat:72/', '/extensions/alerts-updates/')
|
|
|
|
|
|
|
|
redirects('/browse/type:2', '/themes/')
|
|
|
|
redirects('/browse/type:3', '/language-tools')
|
|
|
|
redirects('/browse/type:4', '/search-engines/')
|
|
|
|
# redirects('/browse/type:7', '/plugins/')
|