Skip to content

Commit

Permalink
test: add tests
Browse files Browse the repository at this point in the history
  • Loading branch information
Agrendalath committed Nov 14, 2023
1 parent e121fbf commit 3b8071d
Show file tree
Hide file tree
Showing 7 changed files with 774 additions and 127 deletions.
2 changes: 0 additions & 2 deletions openedx_certificates/compat.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,6 @@
from django.contrib.auth.models import User
from opaque_keys.edx.keys import CourseKey

# TODO: Do we still need all these pylint disable comments? We switched to ruff.


def get_celery_app() -> Celery:
"""Get Celery app to reuse configuration and queues."""
Expand Down
2 changes: 2 additions & 0 deletions requirements/test.in
Original file line number Diff line number Diff line change
Expand Up @@ -7,3 +7,5 @@ pytest-cov # pytest extension for code coverage statistics
django-coverage-plugin # https://github.com/nedbat/django_coverage_plugin
pytest-django # pytest extension for better Django support
code-annotations # provides commands used by the pii_check make target.
dj-inmemorystorage # provides an in-memory storage backend for Django
factory-boy # provides a fixtures replacement for pytest
3 changes: 3 additions & 0 deletions test_settings.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,9 +64,12 @@ def root(path: Path) -> Path:
'context_processors': [
'django.contrib.auth.context_processors.auth', # this is required for admin
'django.contrib.messages.context_processors.messages', # this is required for admin
'django.template.context_processors.request', # this is required for admin
],
},
},
]

TESTING = True
USE_TZ = True
DEFAULT_AUTO_FIELD = 'django.db.models.AutoField'
29 changes: 29 additions & 0 deletions test_utils/factories.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
"""Factories for creating test data."""

from datetime import datetime

import factory
from django.contrib.auth.models import User
from factory.django import DjangoModelFactory
from pytz import UTC


class UserFactory(DjangoModelFactory):
"""A Factory for User objects."""

class Meta: # noqa: D106
model = User
django_get_or_create = ('email', 'username')

_DEFAULT_PASSWORD = 'test' # noqa: S105

username = factory.Sequence('robot{}'.format)
email = factory.Sequence('robot+test+{}@edx.org'.format)
password = factory.django.Password(_DEFAULT_PASSWORD)
first_name = factory.Sequence('Robot{}'.format)
last_name = 'Test'
is_staff = False
is_active = True
is_superuser = False
last_login = datetime(2012, 1, 1, tzinfo=UTC)
date_joined = datetime(2011, 1, 1, tzinfo=UTC)
229 changes: 205 additions & 24 deletions tests/test_generators.py
Original file line number Diff line number Diff line change
@@ -1,34 +1,215 @@
"""This module contains unit tests for the generate_pdf_certificate function."""
from __future__ import annotations

import tempfile
import unittest
from pathlib import Path
import io
from unittest.mock import Mock, call, patch
from uuid import uuid4

from pypdf import PdfReader
import pytest
from django.conf import settings
from django.core.files.base import ContentFile
from django.core.files.storage import DefaultStorage, FileSystemStorage
from django.test import override_settings
from inmemorystorage import InMemoryStorage
from opaque_keys.edx.keys import CourseKey
from pypdf import PdfWriter

from openedx_certificates.generators import generate_pdf_certificate
from openedx_certificates.generators import (
_get_user_name,
_register_font,
_save_certificate,
_write_text_on_template,
generate_pdf_certificate,
)


class TestGeneratePdfCertificate(unittest.TestCase):
"""Unit tests for the generate_pdf_certificate function."""
def test_get_user_name():
"""Test the _get_user_name function."""
user = Mock(first_name="First", last_name="Last")
user.profile.name = "Profile Name"

def test_generate_pdf_certificate(self):
"""Generate a PDF certificate and check that it contains the correct data."""
data = {
'username': 'Test user',
'course_name': 'Some course',
'template_path': 'openedx_certificates/static/certificate_templates/achievement.pdf',
}
# Test when profile name is available
assert _get_user_name(user) == "Profile Name"

# Generate the PDF certificate.
with tempfile.NamedTemporaryFile(suffix='.pdf') as certificate_file:
data['output_path'] = certificate_file.name
generate_pdf_certificate(data)
# Test when profile name is not available
user.profile.name = None
assert _get_user_name(user) == "First Last"

assert Path(data['output_path']).exists()

pdf_reader = PdfReader(certificate_file)
page = pdf_reader.pages[0]
text = page.extract_text()
assert data['username'] in text
assert data['course_name'] in text
@patch("openedx_certificates.generators.ExternalCertificateAsset.get_asset_by_slug")
def test_register_font_without_custom_font(mock_get_asset_by_slug: Mock):
"""Test the _register_font falls back to the default font when no custom font is specified."""
options = {}
assert _register_font(options) == "Helvetica"
mock_get_asset_by_slug.assert_not_called()


@patch("openedx_certificates.generators.ExternalCertificateAsset.get_asset_by_slug")
@patch('openedx_certificates.generators.TTFont')
@patch("openedx_certificates.generators.pdfmetrics.registerFont")
def test_register_font_with_custom_font(mock_register_font: Mock, mock_font_class: Mock, mock_get_asset_by_slug: Mock):
"""Test the _register_font registers the custom font when specified."""
custom_font = "MyFont"
options = {"font": custom_font}

mock_get_asset_by_slug.return_value = "font_path"

assert _register_font(options) == custom_font
mock_get_asset_by_slug.assert_called_once_with(custom_font)
mock_font_class.assert_called_once_with(custom_font, mock_get_asset_by_slug.return_value)
mock_register_font.assert_called_once_with(mock_font_class.return_value)


@pytest.mark.parametrize(
("username", "course_name", "options"),
[
('John Doe', 'Programming 101', {}), # No options - use default coordinates.
('John Doe', 'Programming 101', {'name_y': 250, 'course_name_y': 200}), # Custom coordinates.
],
)
@patch('openedx_certificates.generators.canvas.Canvas', return_value=Mock(stringWidth=Mock(return_value=10)))
def test_write_text_on_template(mock_canvas_class: Mock, username: str, course_name: str, options: dict[str, int]):
"""Test the _write_text_on_template function."""
template_height = 300
template_width = 200
font = 'Helvetica'
string_width = mock_canvas_class.return_value.stringWidth.return_value

# Reset the mock to discard calls list from previous tests
mock_canvas_class.reset_mock()

template_mock = Mock()
template_mock.mediabox = [0, 0, template_width, template_height]

# Call the function with test parameters and mocks
_write_text_on_template(template_mock, font, username, course_name, options)

# Verifying that Canvas was the correct pagesize.
# Use `call_args_list` to ignore the first argument, which is an instance of io.BytesIO.
assert mock_canvas_class.call_args_list[0][1]['pagesize'] == (template_width, template_height)

# Mock Canvas object retrieved from Canvas constructor call
canvas_object = mock_canvas_class.return_value

# Expected coordinates for drawString method, based on fixed stringWidth
expected_name_x = (template_width - string_width) / 2
expected_name_y = options.get('name_y', 290)
expected_course_name_x = (template_width - string_width) / 2
expected_course_name_y = options.get('course_name_y', 220)

# Check the calls to setFont and drawString methods on Canvas object
assert canvas_object.setFont.call_args_list[0] == call(font, 32)
assert canvas_object.drawString.call_args_list[0] == call(expected_name_x, expected_name_y, username)

assert canvas_object.setFont.call_args_list[1] == call(font, 28)
assert canvas_object.drawString.call_args_list[1] == call(
expected_course_name_x,
expected_course_name_y,
course_name,
)


@override_settings(LMS_ROOT_URL="http://example.com", MEDIA_URL="media/")
@pytest.mark.parametrize(
"storage",
[
(InMemoryStorage()), # Test a real storage, without mocking.
(Mock(spec=FileSystemStorage, exists=Mock(return_value=False))), # Test calls in a mocked storage.
# Test calls in a mocked storage when the file already exists.
(Mock(spec=FileSystemStorage, exists=Mock(return_value=True))),
],
)
@patch('openedx_certificates.generators.ContentFile', autospec=True)
def test_save_certificate(mock_contentfile: Mock, storage: DefaultStorage | Mock):
"""Test the _save_certificate function."""
# Mock the certificate.
certificate = Mock(spec=PdfWriter)
certificate_uuid = uuid4()
output_path = f'external_certificates/{certificate_uuid}.pdf'
pdf_bytes = io.BytesIO()
certificate.write.return_value = pdf_bytes
content_file = ContentFile(pdf_bytes.getvalue())
mock_contentfile.return_value = content_file

# Run the function.
with patch('openedx_certificates.generators.default_storage', storage):
url = _save_certificate(certificate, certificate_uuid)

# Check the calls in a mocked storage.
if isinstance(storage, Mock):
storage.exists.assert_called_once_with(output_path)
storage.save.assert_called_once_with(output_path, content_file)
storage.url.assert_not_called()
if storage.exists.return_value:
storage.delete.assert_called_once_with(output_path)
else:
storage.delete.assert_not_called()

if isinstance(storage, Mock):
assert url == f'{settings.LMS_ROOT_URL}/media/{output_path}'
else:
assert url == f'/{output_path}'


@pytest.mark.parametrize(
("course_name", "options", "expected_template_slug"),
[
('Test Course', {'template': 'template_slug'}, 'template_slug'),
('Test Course;Test Course', {'template': 'template_slug'}, 'template_slug'),
(
'Test Course;Test Course',
{'template': 'template_slug', 'template_two-lines': 'template_two_lines_slug'},
'template_two_lines_slug',
),
],
)
@patch(
'openedx_certificates.generators.ExternalCertificateAsset.get_asset_by_slug',
return_value=Mock(
open=Mock(
return_value=Mock(
__enter__=Mock(return_value=Mock(read=Mock(return_value=b'pdf_data'))),
__exit__=Mock(return_value=None),
),
),
),
)
@patch('openedx_certificates.generators._get_user_name')
@patch('openedx_certificates.generators.get_course_name')
@patch('openedx_certificates.generators._register_font')
@patch('openedx_certificates.generators.PdfReader')
@patch('openedx_certificates.generators.PdfWriter')
@patch(
'openedx_certificates.generators._write_text_on_template',
return_value=Mock(getpdfdata=Mock(return_value=b'pdf_data')),
)
@patch('openedx_certificates.generators._save_certificate', return_value='certificate_url')
def test_generate_pdf_certificate( # noqa: PLR0913
mock_save_certificate: Mock,
mock_write_text_on_template: Mock,
mock_pdf_writer: Mock,
mock_pdf_reader: Mock,
mock_register_font: Mock,
mock_get_course_name: Mock,
mock_get_user_name: Mock,
mock_get_asset_by_slug: Mock,
course_name: str,
options: dict[str, str],
expected_template_slug: str,
):
"""Test the generate_pdf_certificate function."""
course_id = CourseKey.from_string('course-v1:edX+DemoX+Demo_Course')
user = Mock()
mock_get_course_name.return_value = course_name

result = generate_pdf_certificate(course_id, user, Mock(), options)

assert result == 'certificate_url'
mock_get_asset_by_slug.assert_called_with(expected_template_slug)
mock_get_user_name.assert_called_once_with(user)
mock_get_course_name.assert_called_once_with(course_id)
mock_register_font.assert_called_once_with(options)
mock_pdf_reader.assert_called()
mock_pdf_writer.assert_called()
mock_write_text_on_template.assert_called_once()
mock_save_certificate.assert_called_once()
Loading

0 comments on commit 3b8071d

Please sign in to comment.