You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
244 lines
11 KiB
Python
244 lines
11 KiB
Python
import os
|
|
from django.test import TestCase, Client, RequestFactory
|
|
from django.urls import reverse, NoReverseMatch
|
|
from django.conf import settings
|
|
from django.templatetags.static import static
|
|
from django.utils import timezone
|
|
from django.utils.translation import gettext_lazy as _
|
|
from django.core.files.uploadedfile import SimpleUploadedFile
|
|
from django.template import Template, Context, TemplateSyntaxError, VariableDoesNotExist
|
|
from unittest import mock
|
|
from unittest.mock import Mock
|
|
from importlib import reload
|
|
|
|
from members.models import Member, Group, DIVERSE
|
|
from startpage import urls
|
|
from startpage.views import redirect, handler500
|
|
from startpage.templatetags.markdown_extras import RenderAsTemplateNode, render_as_template
|
|
|
|
from .models import Post, Section, Image, Link, MemberOnPost
|
|
|
|
|
|
class BasicTestCase(TestCase):
|
|
def setUp(self):
|
|
orga = Section.objects.create(title='Organisation', urlname='orga', website_text='Section is a about everything.')
|
|
recent = Section.objects.create(title='Recent', urlname=settings.RECENT_SECTION, website_text='Recently recent.')
|
|
reports = Section.objects.create(title='Reports', urlname=settings.REPORTS_SECTION, website_text='Reporty reports.')
|
|
Post.objects.create(title='Climbing is fun', urlname='climbing-is-fun', website_text='Climbing is fun!',
|
|
section=recent)
|
|
Post.objects.create(title='Last trip', urlname='last-trip', website_text='A fun trip.',
|
|
section=reports)
|
|
file = SimpleUploadedFile("post_image.jpg", b"file_content", content_type="image/jpeg")
|
|
staff_post = Post.objects.create(title='Staff', urlname='staff', website_text='This is our staff: Peter.',
|
|
section=orga)
|
|
self.image_with_file = Image.objects.create(post=staff_post, f=file)
|
|
file = SimpleUploadedFile("member_image.jpg", b"file_content", content_type="image/jpeg")
|
|
m = Member.objects.create(prename='crazy', lastname='cool', birth_date=timezone.now().date(),
|
|
email=settings.TEST_MAIL, gender=DIVERSE,
|
|
image=file)
|
|
crazy_group = Group.objects.create(name='CrazyClimbers', show_website=True)
|
|
m.group.add(crazy_group)
|
|
m.save()
|
|
Group.objects.create(name='SuperClimbers', show_website=False)
|
|
crazy_post = Post.objects.create(title='The crazy climbers', urlname='crazy', website_text='foobar',
|
|
section=orga)
|
|
crazy_post.groups.add(crazy_group)
|
|
crazy_post.save()
|
|
|
|
self.post_no_section = Post.objects.create(title='No Section', urlname='no-section', section=None)
|
|
self.image_no_file = Image.objects.create(post=staff_post)
|
|
self.test_link = Link.objects.create(title='Test Link', url='https://example.com')
|
|
|
|
|
|
class ModelsTestCase(BasicTestCase):
|
|
def test_str(self):
|
|
orga = Section.objects.get(urlname='orga')
|
|
self.assertEqual(str(orga), orga.title, 'String representation does not match title.')
|
|
post = Post.objects.get(urlname='staff', section=orga)
|
|
self.assertEqual(post.absolute_section(), orga.title, 'Displayed section of post does not match section title.')
|
|
self.assertEqual(str(post), post.title, 'String representation does not match title.')
|
|
|
|
def test_absolute_urlnames(self):
|
|
orga = Section.objects.get(urlname='orga')
|
|
recent = Section.objects.get(urlname=settings.RECENT_SECTION)
|
|
reports = Section.objects.get(urlname=settings.REPORTS_SECTION)
|
|
self.assertEqual(orga.absolute_urlname(), '/de/orga')
|
|
|
|
post1 = Post.objects.get(urlname='staff', section=orga)
|
|
self.assertEqual(post1.absolute_urlname(), '/de/orga/staff')
|
|
self.assertEqual(post1.absolute_urlname(), reverse('startpage:post', args=(orga.urlname, 'staff')))
|
|
post2 = Post.objects.get(urlname='climbing-is-fun', section=recent)
|
|
self.assertEqual(post2.absolute_urlname(),
|
|
'/de/{name}/climbing-is-fun'.format(name=settings.RECENT_SECTION))
|
|
self.assertEqual(post2.absolute_urlname(), reverse('startpage:post', args=(recent.urlname, 'climbing-is-fun')))
|
|
post3 = Post.objects.get(urlname='last-trip', section=reports)
|
|
self.assertEqual(post3.absolute_urlname(),
|
|
'/de/{name}/last-trip'.format(name=settings.REPORTS_SECTION))
|
|
self.assertEqual(post3.absolute_urlname(), reverse('startpage:post', args=(reports.urlname, 'last-trip')))
|
|
|
|
def test_post_absolute_section_none(self):
|
|
"""Test Post.absolute_section when section is None"""
|
|
self.assertEqual(self.post_no_section.absolute_section(), 'Aktuelles')
|
|
|
|
def test_post_absolute_urlname_no_section(self):
|
|
"""Test Post.absolute_urlname when section is None"""
|
|
expected_url = reverse('startpage:post', args=('aktuelles', 'no-section'))
|
|
self.assertEqual(self.post_no_section.absolute_urlname(), expected_url)
|
|
|
|
def test_image_str_without_file(self):
|
|
"""Test Image.__str__ when no file is associated"""
|
|
self.assertEqual(str(self.image_no_file), str(_('Empty')))
|
|
|
|
def test_image_str_with_file(self):
|
|
"""Test Image.__str__ when file is associated"""
|
|
# The str should return basename of the file
|
|
expected = os.path.basename(self.image_with_file.f.name)
|
|
self.assertEqual(str(self.image_with_file), expected)
|
|
|
|
def test_link_str(self):
|
|
"""Test Link.__str__ method"""
|
|
self.assertEqual(str(self.test_link), 'Test Link')
|
|
|
|
def test_section_absolute_urlname_no_reverse_match(self):
|
|
"""Test Section.absolute_urlname when NoReverseMatch occurs"""
|
|
section = Section.objects.get(urlname='orga')
|
|
with mock.patch('startpage.models.reverse', side_effect=NoReverseMatch):
|
|
self.assertEqual(section.absolute_urlname(), str(_('deactivated')))
|
|
|
|
def test_post_absolute_urlname_no_reverse_match(self):
|
|
"""Test Post.absolute_urlname when NoReverseMatch occurs"""
|
|
post = Post.objects.get(urlname='staff')
|
|
with mock.patch('startpage.models.reverse', side_effect=NoReverseMatch):
|
|
self.assertEqual(post.absolute_urlname(), str(_('deactivated')))
|
|
|
|
|
|
class ViewTestCase(BasicTestCase):
|
|
def test_index(self):
|
|
c = Client()
|
|
url = reverse('startpage:index')
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 200, 'Response code is not 200 for index.')
|
|
|
|
def test_posts_no_category(self):
|
|
c = Client()
|
|
url = reverse('startpage:post', args=(settings.RECENT_SECTION, 'climbing-is-fun'))
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 200, 'Response code is not 200 for climbing post.')
|
|
|
|
def test_posts_orga(self):
|
|
c = Client()
|
|
url = reverse('startpage:post', args=('orga', 'staff'))
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 200, 'Response code is not 200 for staff post.')
|
|
|
|
def test_section(self):
|
|
c = Client()
|
|
url = reverse('startpage:section', args=('orga',))
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 200, 'Response code is not 200 for section page.')
|
|
|
|
def test_section_recent(self):
|
|
c = Client()
|
|
url = reverse('startpage:' + settings.RECENT_SECTION)
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 200, 'Response code is not 200 for section page.')
|
|
|
|
def test_section_reports(self):
|
|
c = Client()
|
|
url = reverse('startpage:' + settings.REPORTS_SECTION)
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 200, 'Response code is not 200 for section page.')
|
|
|
|
def test_404(self):
|
|
c = Client()
|
|
response = c.get('/de/asdfasdfasdf')
|
|
self.assertEqual(response.status_code, 404, 'Response code is not 404 when page not found.')
|
|
|
|
def test_impressum(self):
|
|
c = Client()
|
|
response = c.get('/de/impressum')
|
|
self.assertEqual(response.status_code, 200, 'Response code is not 200 for impressum.')
|
|
|
|
def test_gruppen(self):
|
|
c = Client()
|
|
url = reverse('startpage:gruppe_detail', args=('CrazyClimbers',))
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 200, 'Response code is not 200 for group.')
|
|
|
|
def test_gruppen_404(self):
|
|
c = Client()
|
|
url = reverse('startpage:gruppe_detail', args=('SuperClimbers',))
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 404, 'Response code is not 404 for group.')
|
|
url = reverse('startpage:gruppe_detail', args=('SuperClimbersNotExisting',))
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 404, 'Response code is not 404 for group.')
|
|
|
|
def test_post_with_groups(self):
|
|
c = Client()
|
|
url = reverse('startpage:post', args=('orga', 'crazy'))
|
|
response = c.get(url)
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
def test_post_image(self):
|
|
c = Client()
|
|
url = self.image_with_file.f.url
|
|
response = c.get('/de' + url)
|
|
self.assertEqual(response.status_code, 200, 'Images on posts should be visible without login.')
|
|
|
|
def test_urlpatterns_with_redirect_url(self):
|
|
"""Test URL patterns when STARTPAGE_REDIRECT_URL is not empty"""
|
|
|
|
# Mock settings to have a non-empty STARTPAGE_REDIRECT_URL
|
|
with mock.patch.object(settings, 'STARTPAGE_REDIRECT_URL', 'https://example.com'):
|
|
# Reload the urls module to trigger the conditional urlpatterns creation
|
|
reload(urls)
|
|
|
|
# Check that urlpatterns contains the redirect view
|
|
url_names = [pattern.name for pattern in urls.urlpatterns if hasattr(pattern, 'name')]
|
|
self.assertIn('index', url_names)
|
|
self.assertEqual(len(urls.urlpatterns), 2) # Should have index and impressum only
|
|
|
|
def test_redirect_view(self):
|
|
"""Test redirect view functionality"""
|
|
request = RequestFactory().get('/')
|
|
with mock.patch.object(settings, 'STARTPAGE_REDIRECT_URL', 'https://example.com'):
|
|
response = redirect(request)
|
|
self.assertEqual(response.status_code, 302)
|
|
self.assertEqual(response.url, 'https://example.com')
|
|
|
|
def test_handler500(self):
|
|
"""Test custom 500 error handler"""
|
|
request = RequestFactory().get('/')
|
|
response = handler500(request)
|
|
self.assertEqual(response.status_code, 500)
|
|
|
|
|
|
class MarkdownExtrasTestCase(TestCase):
|
|
def test_render_as_template_node_variable_does_not_exist(self):
|
|
node = RenderAsTemplateNode('nonexistent_var', 'result')
|
|
context = Context({})
|
|
result = node.render(context)
|
|
self.assertEqual(result, '')
|
|
|
|
def test_render_as_template_no_arguments(self):
|
|
token = Mock()
|
|
token.contents = 'render_as_template'
|
|
parser = Mock()
|
|
with self.assertRaises(TemplateSyntaxError):
|
|
render_as_template(parser, token)
|
|
|
|
def test_render_as_template_invalid_syntax(self):
|
|
token = Mock()
|
|
token.contents = 'render_as_template "content"'
|
|
parser = Mock()
|
|
with self.assertRaises(TemplateSyntaxError):
|
|
render_as_template(parser, token)
|
|
|
|
def test_render_as_template_unquoted_argument(self):
|
|
token = Mock()
|
|
token.contents = 'render_as_template content as result'
|
|
parser = Mock()
|
|
with self.assertRaises(TemplateSyntaxError):
|
|
render_as_template(parser, token)
|