diff --git a/boards/tests.py b/boards/tests.py deleted file mode 100644 --- a/boards/tests.py +++ /dev/null @@ -1,349 +0,0 @@ -# coding=utf-8 -import time -import logging -import simplejson -from django.core.paginator import Paginator - -from django.test import TestCase -from django.test.client import Client -from django.core.urlresolvers import reverse, NoReverseMatch -from boards.abstracts.settingsmanager import get_settings_manager - -from boards.models import Post, Tag, Thread, KeyPair, GlobalId -from boards import urls -from boards import settings -from boards.views.api import api_get_threaddiff -from boards.utils import datetime_to_epoch -from boards.views.sync import generate_request_get -import neboard - -TEST_TAG = 'test_tag' - -PAGE_404 = 'boards/404.html' - -TEST_TEXT = 'test text' - -NEW_THREAD_PAGE = '/' -THREAD_PAGE_ONE = '/thread/1/' -THREAD_PAGE = '/thread/' -TAG_PAGE = '/tag/' -HTTP_CODE_REDIRECT = 302 -HTTP_CODE_OK = 200 -HTTP_CODE_NOT_FOUND = 404 - -logger = logging.getLogger(__name__) - - -class PostTests(TestCase): - - def _create_post(self): - tag = Tag.objects.create(name=TEST_TAG) - return Post.objects.create_post(title='title', text='text', - tags=[tag]) - - def test_post_add(self): - """Test adding post""" - - post = self._create_post() - - self.assertIsNotNone(post, 'No post was created.') - self.assertEqual(TEST_TAG, post.get_thread().tags.all()[0].name, - 'No tags were added to the post.') - - def test_delete_post(self): - """Test post deletion""" - - post = self._create_post() - post_id = post.id - - Post.objects.delete_post(post) - - self.assertFalse(Post.objects.filter(id=post_id).exists()) - - def test_delete_thread(self): - """Test thread deletion""" - - opening_post = self._create_post() - thread = opening_post.get_thread() - reply = Post.objects.create_post("", "", thread=thread) - - thread.delete() - - self.assertFalse(Post.objects.filter(id=reply.id).exists()) - - def test_post_to_thread(self): - """Test adding post to a thread""" - - op = self._create_post() - post = Post.objects.create_post("", "", thread=op.get_thread()) - - self.assertIsNotNone(post, 'Reply to thread wasn\'t created') - self.assertEqual(op.get_thread().last_edit_time, post.pub_time, - 'Post\'s create time doesn\'t match thread last edit' - ' time') - - def test_delete_posts_by_ip(self): - """Test deleting posts with the given ip""" - - post = self._create_post() - post_id = post.id - - Post.objects.delete_posts_by_ip('0.0.0.0') - - self.assertFalse(Post.objects.filter(id=post_id).exists()) - - def test_get_thread(self): - """Test getting all posts of a thread""" - - opening_post = self._create_post() - - for i in range(0, 2): - Post.objects.create_post('title', 'text', - thread=opening_post.get_thread()) - - thread = opening_post.get_thread() - - self.assertEqual(3, thread.replies.count()) - - def test_create_post_with_tag(self): - """Test adding tag to post""" - - tag = Tag.objects.create(name='test_tag') - post = Post.objects.create_post(title='title', text='text', tags=[tag]) - - thread = post.get_thread() - self.assertIsNotNone(post, 'Post not created') - self.assertTrue(tag in thread.tags.all(), 'Tag not added to thread') - self.assertTrue(thread in tag.threads.all(), 'Thread not added to tag') - - def test_thread_max_count(self): - """Test deletion of old posts when the max thread count is reached""" - - for i in range(settings.MAX_THREAD_COUNT + 1): - self._create_post() - - self.assertEqual(settings.MAX_THREAD_COUNT, - len(Thread.objects.filter(archived=False))) - - def test_pages(self): - """Test that the thread list is properly split into pages""" - - for i in range(settings.MAX_THREAD_COUNT): - self._create_post() - - all_threads = Thread.objects.filter(archived=False) - - paginator = Paginator(Thread.objects.filter(archived=False), - settings.THREADS_PER_PAGE) - posts_in_second_page = paginator.page(2).object_list - first_post = posts_in_second_page[0] - - self.assertEqual(all_threads[settings.THREADS_PER_PAGE].id, - first_post.id) - - -class PagesTest(TestCase): - - def test_404(self): - """Test receiving error 404 when opening a non-existent page""" - - tag_name = 'test_tag' - tag = Tag.objects.create(name=tag_name) - client = Client() - - Post.objects.create_post('title', TEST_TEXT, tags=[tag]) - - existing_post_id = Post.objects.all()[0].id - response_existing = client.get(THREAD_PAGE + str(existing_post_id) + - '/') - self.assertEqual(HTTP_CODE_OK, response_existing.status_code, - 'Cannot open existing thread') - - response_not_existing = client.get(THREAD_PAGE + str( - existing_post_id + 1) + '/') - self.assertEqual(PAGE_404, response_not_existing.templates[0].name, - 'Not existing thread is opened') - - response_existing = client.get(TAG_PAGE + tag_name + '/') - self.assertEqual(HTTP_CODE_OK, - response_existing.status_code, - 'Cannot open existing tag') - - response_not_existing = client.get(TAG_PAGE + 'not_tag' + '/') - self.assertEqual(PAGE_404, - response_not_existing.templates[0].name, - 'Not existing tag is opened') - - reply_id = Post.objects.create_post('', TEST_TEXT, - thread=Post.objects.all()[0] - .get_thread()) - response_not_existing = client.get(THREAD_PAGE + str( - reply_id) + '/') - self.assertEqual(PAGE_404, - response_not_existing.templates[0].name, - 'Reply is opened as a thread') - - -class FormTest(TestCase): - def test_post_validation(self): - client = Client() - - valid_tags = 'tag1 tag_2 тег_3' - invalid_tags = '$%_356 ---' - - response = client.post(NEW_THREAD_PAGE, {'title': 'test title', - 'text': TEST_TEXT, - 'tags': valid_tags}) - self.assertEqual(response.status_code, HTTP_CODE_REDIRECT, - msg='Posting new message failed: got code ' + - str(response.status_code)) - - self.assertEqual(1, Post.objects.count(), - msg='No posts were created') - - client.post(NEW_THREAD_PAGE, {'text': TEST_TEXT, - 'tags': invalid_tags}) - self.assertEqual(1, Post.objects.count(), msg='The validation passed ' - 'where it should fail') - - # Change posting delay so we don't have to wait for 30 seconds or more - old_posting_delay = neboard.settings.POSTING_DELAY - # Wait for the posting delay or we won't be able to post - neboard.settings.POSTING_DELAY = 1 - time.sleep(neboard.settings.POSTING_DELAY + 1) - response = client.post(THREAD_PAGE_ONE, {'text': TEST_TEXT, - 'tags': valid_tags}) - self.assertEqual(HTTP_CODE_REDIRECT, response.status_code, - msg='Posting new message failed: got code ' + - str(response.status_code)) - # Restore posting delay - neboard.settings.POSTING_DELAY = old_posting_delay - - self.assertEqual(2, Post.objects.count(), - msg='No posts were created') - - -class ViewTest(TestCase): - - def test_all_views(self): - """ - Try opening all views defined in ulrs.py that don't need additional - parameters - """ - - client = Client() - for url in urls.urlpatterns: - try: - view_name = url.name - logger.debug('Testing view %s' % view_name) - - try: - response = client.get(reverse(view_name)) - - self.assertEqual(HTTP_CODE_OK, response.status_code, - '%s view not opened' % view_name) - except NoReverseMatch: - # This view just needs additional arguments - pass - except Exception as e: - self.fail('Got exception %s at %s view' % (e, view_name)) - except AttributeError: - # This is normal, some views do not have names - pass - - -class AbstractTest(TestCase): - def test_settings_manager(self): - request = MockRequest() - settings_manager = get_settings_manager(request) - - settings_manager.set_setting('test_setting', 'test_value') - self.assertEqual('test_value', settings_manager.get_setting( - 'test_setting'), 'Setting update failed.') - - -class MockRequest: - def __init__(self): - self.session = dict() - self.GET = dict() - self.POST = dict() - - -class KeyTest(TestCase): - def test_create_key(self): - key = KeyPair.objects.generate_key('ecdsa') - - self.assertIsNotNone(key, 'The key was not created.') - - def test_validation(self): - key = KeyPair.objects.generate_key(key_type='ecdsa') - message = 'msg' - signature = key.sign(message) - valid = KeyPair.objects.verify(key.public_key, message, signature, - key_type='ecdsa') - - self.assertTrue(valid, 'Message verification failed.') - - def test_primary_constraint(self): - KeyPair.objects.generate_key(key_type='ecdsa', primary=True) - - try: - KeyPair.objects.generate_key(key_type='ecdsa', primary=True) - self.fail('Exception should be thrown indicating there can be only' - ' one primary key.') - except Exception: - pass - - def test_request_get(self): - model_id = GlobalId(key_type='test', key='test key', local_id='1') - model_id.save() - - self.assertTrue('' in - generate_request_get([model_id]), - 'Wrong XML generated for the GET request.') - - -class ApiTest(TestCase): - def test_thread_diff(self): - tag = Tag.objects.create(name=TEST_TAG) - opening_post = Post.objects.create_post(title='title', text='text', - tags=[tag]) - - last_edit_time = datetime_to_epoch(opening_post.last_edit_time) - - # Check the exact timestamp post was added - empty_response = api_get_threaddiff(MockRequest(), - str(opening_post.thread_new.id), - str(last_edit_time)) - diff = simplejson.loads(empty_response.content) - self.assertEqual(0, len(diff['added']), - 'There must be no added posts in the diff.') - self.assertEqual(0, len(diff['updated']), - 'There must be no updated posts in the diff.') - - reply = Post.objects.create_post(title='', - text='[post]%d[/post]\ntext' % opening_post.id, - thread=opening_post.thread_new) - - # Check the timestamp before post was added - response = api_get_threaddiff(MockRequest(), - str(opening_post.thread_new.id), - str(last_edit_time)) - diff = simplejson.loads(response.content) - self.assertEqual(1, len(diff['added']), - 'There must be 1 added posts in the diff.') - self.assertEqual(1, len(diff['updated']), - 'There must be 1 updated posts in the diff.') - - empty_response = api_get_threaddiff(MockRequest(), - str(opening_post.thread_new.id), - str(datetime_to_epoch( - reply.last_edit_time))) - diff = simplejson.loads(empty_response.content) - self.assertEqual(0, len(diff['added']), - 'There must be no added posts in the diff.') - self.assertEqual(0, len(diff['updated']), - 'There must be no updated posts in the diff.') - diff --git a/boards/tests/__init__.py b/boards/tests/__init__.py new file mode 100644 diff --git a/boards/tests/mocks.py b/boards/tests/mocks.py new file mode 100644 --- /dev/null +++ b/boards/tests/mocks.py @@ -0,0 +1,5 @@ +class MockRequest: + def __init__(self): + self.session = dict() + self.GET = dict() + self.POST = dict() \ No newline at end of file diff --git a/boards/tests/test_abstract.py b/boards/tests/test_abstract.py new file mode 100644 --- /dev/null +++ b/boards/tests/test_abstract.py @@ -0,0 +1,13 @@ +from django.test import TestCase +from boards.abstracts.settingsmanager import get_settings_manager +from boards.tests.mocks import MockRequest + + +class AbstractTest(TestCase): + def test_settings_manager(self): + request = MockRequest() + settings_manager = get_settings_manager(request) + + settings_manager.set_setting('test_setting', 'test_value') + self.assertEqual('test_value', settings_manager.get_setting( + 'test_setting'), 'Setting update failed.') \ No newline at end of file diff --git a/boards/tests/test_api.py b/boards/tests/test_api.py new file mode 100644 --- /dev/null +++ b/boards/tests/test_api.py @@ -0,0 +1,50 @@ +import simplejson + +from django.test import TestCase + +from boards.models import Tag, Post +from boards.tests.mocks import MockRequest +from boards.utils import datetime_to_epoch +from boards.views.api import api_get_threaddiff + + +class ApiTest(TestCase): + def test_thread_diff(self): + tag = Tag.objects.create(name='test_tag') + opening_post = Post.objects.create_post(title='title', text='text', + tags=[tag]) + + last_edit_time = datetime_to_epoch(opening_post.last_edit_time) + + # Check the exact timestamp post was added + empty_response = api_get_threaddiff(MockRequest(), + str(opening_post.thread_new.id), + str(last_edit_time)) + diff = simplejson.loads(empty_response.content) + self.assertEqual(0, len(diff['added']), + 'There must be no added posts in the diff.') + self.assertEqual(0, len(diff['updated']), + 'There must be no updated posts in the diff.') + + reply = Post.objects.create_post(title='', + text='[post]%d[/post]\ntext' % opening_post.id, + thread=opening_post.thread_new) + + # Check the timestamp before post was added + response = api_get_threaddiff(MockRequest(), + str(opening_post.thread_new.id), + str(last_edit_time)) + diff = simplejson.loads(response.content) + self.assertEqual(1, len(diff['added']), + 'There must be 1 added posts in the diff.') + self.assertEqual(1, len(diff['updated']), + 'There must be 1 updated posts in the diff.') + + empty_response = api_get_threaddiff(MockRequest(), + str(opening_post.thread_new.id), + str(datetime_to_epoch(reply.last_edit_time))) + diff = simplejson.loads(empty_response.content) + self.assertEqual(0, len(diff['added']), + 'There must be no added posts in the diff.') + self.assertEqual(0, len(diff['updated']), + 'There must be no updated posts in the diff.') \ No newline at end of file diff --git a/boards/tests/test_forms.py b/boards/tests/test_forms.py new file mode 100644 --- /dev/null +++ b/boards/tests/test_forms.py @@ -0,0 +1,55 @@ +from django.test import TestCase, Client +import time +from boards import settings +from boards.models import Post +import neboard + + +TEST_TAG = 'test_tag' + +PAGE_404 = 'boards/404.html' + +TEST_TEXT = 'test text' + +NEW_THREAD_PAGE = '/' +THREAD_PAGE_ONE = '/thread/1/' +HTTP_CODE_REDIRECT = 302 + + +class FormTest(TestCase): + def test_post_validation(self): + client = Client() + + valid_tags = 'tag1 tag_2 тег_3' + invalid_tags = '$%_356 ---' + + response = client.post(NEW_THREAD_PAGE, {'title': 'test title', + 'text': TEST_TEXT, + 'tags': valid_tags}) + self.assertEqual(response.status_code, HTTP_CODE_REDIRECT, + msg='Posting new message failed: got code ' + + str(response.status_code)) + + self.assertEqual(1, Post.objects.count(), + msg='No posts were created') + + client.post(NEW_THREAD_PAGE, {'text': TEST_TEXT, + 'tags': invalid_tags}) + self.assertEqual(1, Post.objects.count(), msg='The validation passed ' + 'where it should fail') + + # Change posting delay so we don't have to wait for 30 seconds or more + old_posting_delay = neboard.settings.POSTING_DELAY + # Wait fot the posting delay or we won't be able to post + neboard.settings.POSTING_DELAY = 1 + time.sleep(neboard.settings.POSTING_DELAY + 1) + response = client.post(THREAD_PAGE_ONE, {'text': TEST_TEXT, + 'tags': valid_tags}) + self.assertEqual(HTTP_CODE_REDIRECT, response.status_code, + msg='Posting new message failed: got code ' + + str(response.status_code)) + # Restore posting delay + settings.POSTING_DELAY = old_posting_delay + + self.assertEqual(2, Post.objects.count(), + msg='No posts were created') diff --git a/boards/tests/test_keys.py b/boards/tests/test_keys.py new file mode 100644 --- /dev/null +++ b/boards/tests/test_keys.py @@ -0,0 +1,39 @@ +from django.test import TestCase +from boards.models import KeyPair, GlobalId +from boards.views.sync import generate_request_get + + +class KeyTest(TestCase): + def test_create_key(self): + key = KeyPair.objects.generate_key('ecdsa') + + self.assertIsNotNone(key, 'The key was not created.') + + def test_validation(self): + key = KeyPair.objects.generate_key(key_type='ecdsa') + message = 'msg' + signature = key.sign(message) + valid = KeyPair.objects.verify(key.public_key, message, signature, + key_type='ecdsa') + + self.assertTrue(valid, 'Message verification failed.') + + def test_primary_constraint(self): + KeyPair.objects.generate_key(key_type='ecdsa', primary=True) + + try: + KeyPair.objects.generate_key(key_type='ecdsa', primary=True) + self.fail('Exception should be thrown indicating there can be only' + ' one primary key.') + except Exception: + pass + + def test_request_get(self): + model_id = GlobalId(key_type='test', key='test key', local_id='1') + model_id.save() + + self.assertTrue('' in + generate_request_get([model_id]), + 'Wrong XML generated for the GET request.') diff --git a/boards/tests/test_pages.py b/boards/tests/test_pages.py new file mode 100644 --- /dev/null +++ b/boards/tests/test_pages.py @@ -0,0 +1,56 @@ +from django.test import TestCase, Client +from boards.models import Tag, Post + +TEST_TEXT = 'test' + +NEW_THREAD_PAGE = '/' +THREAD_PAGE_ONE = '/thread/1/' +THREAD_PAGE = '/thread/' +TAG_PAGE = '/tag/' +HTTP_CODE_REDIRECT = 302 +HTTP_CODE_OK = 200 +HTTP_CODE_NOT_FOUND = 404 + +PAGE_404 = 'boards/404.html' + + +class PagesTest(TestCase): + + def test_404(self): + """Test receiving error 404 when opening a non-existent page""" + + tag_name = 'test_tag' + tag = Tag.objects.create(name=tag_name) + client = Client() + + Post.objects.create_post('title', TEST_TEXT, tags=[tag]) + + existing_post_id = Post.objects.all()[0].id + response_existing = client.get(THREAD_PAGE + str(existing_post_id) + + '/') + self.assertEqual(HTTP_CODE_OK, response_existing.status_code, + 'Cannot open existing thread') + + response_not_existing = client.get(THREAD_PAGE + str( + existing_post_id + 1) + '/') + self.assertEqual(PAGE_404, response_not_existing.templates[0].name, + 'Not existing thread is opened') + + response_existing = client.get(TAG_PAGE + tag_name + '/') + self.assertEqual(HTTP_CODE_OK, + response_existing.status_code, + 'Cannot open existing tag') + + response_not_existing = client.get(TAG_PAGE + 'not_tag' + '/') + self.assertEqual(PAGE_404, + response_not_existing.templates[0].name, + 'Not existing tag is opened') + + reply_id = Post.objects.create_post('', TEST_TEXT, + thread=Post.objects.all()[0] + .get_thread()) + response_not_existing = client.get(THREAD_PAGE + str( + reply_id) + '/') + self.assertEqual(PAGE_404, + response_not_existing.templates[0].name, + 'Reply is opened as a thread') diff --git a/boards/tests/test_post.py b/boards/tests/test_post.py new file mode 100644 --- /dev/null +++ b/boards/tests/test_post.py @@ -0,0 +1,112 @@ +from django.core.paginator import Paginator +from django.test import TestCase +from boards import settings +from boards.models import Tag, Post, Thread + + +class PostTests(TestCase): + + def _create_post(self): + tag = Tag.objects.create(name='test_tag') + return Post.objects.create_post(title='title', text='text', + tags=[tag]) + + def test_post_add(self): + """Test adding post""" + + post = self._create_post() + + self.assertIsNotNone(post, 'No post was created.') + self.assertEqual('test_tag', post.get_thread().tags.all()[0].name, + 'No tags were added to the post.') + + def test_delete_post(self): + """Test post deletion""" + + post = self._create_post() + post_id = post.id + + Post.objects.delete_post(post) + + self.assertFalse(Post.objects.filter(id=post_id).exists()) + + def test_delete_thread(self): + """Test thread deletion""" + + opening_post = self._create_post() + thread = opening_post.get_thread() + reply = Post.objects.create_post("", "", thread=thread) + + thread.delete() + + self.assertFalse(Post.objects.filter(id=reply.id).exists()) + + def test_post_to_thread(self): + """Test adding post to a thread""" + + op = self._create_post() + post = Post.objects.create_post("", "", thread=op.get_thread()) + + self.assertIsNotNone(post, 'Reply to thread wasn\'t created') + self.assertEqual(op.get_thread().last_edit_time, post.pub_time, + 'Post\'s create time doesn\'t match thread last edit' + ' time') + + def test_delete_posts_by_ip(self): + """Test deleting posts with the given ip""" + + post = self._create_post() + post_id = post.id + + Post.objects.delete_posts_by_ip('0.0.0.0') + + self.assertFalse(Post.objects.filter(id=post_id).exists()) + + def test_get_thread(self): + """Test getting all posts of a thread""" + + opening_post = self._create_post() + + for i in range(2): + Post.objects.create_post('title', 'text', + thread=opening_post.get_thread()) + + thread = opening_post.get_thread() + + self.assertEqual(3, thread.replies.count()) + + def test_create_post_with_tag(self): + """Test adding tag to post""" + + tag = Tag.objects.create(name='test_tag') + post = Post.objects.create_post(title='title', text='text', tags=[tag]) + + thread = post.get_thread() + self.assertIsNotNone(post, 'Post not created') + self.assertTrue(tag in thread.tags.all(), 'Tag not added to thread') + self.assertTrue(thread in tag.threads.all(), 'Thread not added to tag') + + def test_thread_max_count(self): + """Test deletion of old posts when the max thread count is reached""" + + for i in range(settings.MAX_THREAD_COUNT + 1): + self._create_post() + + self.assertEqual(settings.MAX_THREAD_COUNT, + len(Thread.objects.filter(archived=False))) + + def test_pages(self): + """Test that the thread list is properly split into pages""" + + for i in range(settings.MAX_THREAD_COUNT): + self._create_post() + + all_threads = Thread.objects.filter(archived=False) + + paginator = Paginator(Thread.objects.filter(archived=False), + settings.THREADS_PER_PAGE) + posts_in_second_page = paginator.page(2).object_list + first_post = posts_in_second_page[0] + + self.assertEqual(all_threads[settings.THREADS_PER_PAGE].id, + first_post.id) \ No newline at end of file diff --git a/boards/tests/test_views.py b/boards/tests/test_views.py new file mode 100644 --- /dev/null +++ b/boards/tests/test_views.py @@ -0,0 +1,38 @@ +import logging +from django.core.urlresolvers import reverse, NoReverseMatch +from django.test import TestCase, Client +from boards import urls + + +logger = logging.getLogger(__name__) + +HTTP_CODE_OK = 200 + + +class ViewTest(TestCase): + + def test_all_views(self): + """ + Try opening all views defined in ulrs.py that don't need additional + parameters + """ + + client = Client() + for url in urls.urlpatterns: + try: + view_name = url.name + logger.debug('Testing view %s' % view_name) + + try: + response = client.get(reverse(view_name)) + + self.assertEqual(HTTP_CODE_OK, response.status_code, + '%s view not opened' % view_name) + except NoReverseMatch: + # This view just needs additional arguments + pass + except Exception as e: + self.fail('Got exception %s at %s view' % (e, view_name)) + except AttributeError: + # This is normal, some views do not have names + pass