rendercv/tests/test_renderer.py

416 lines
14 KiB
Python
Raw Normal View History

2024-02-06 20:18:46 +00:00
import math
2024-02-07 18:18:04 +00:00
import filecmp
import shutil
2024-02-11 18:36:56 +00:00
import os
2024-02-09 19:14:46 +00:00
import pathlib
2024-02-06 20:18:46 +00:00
import pytest
import jinja2
2024-02-07 18:18:04 +00:00
import time_machine
2024-02-11 22:08:32 +00:00
import pypdf
2024-02-06 20:18:46 +00:00
from rendercv import renderer as r
2024-02-07 18:18:04 +00:00
from rendercv import data_models as dm
2024-02-06 20:18:46 +00:00
2024-02-09 19:14:46 +00:00
def test_latex_file_class(tmp_path, rendercv_data_model, jinja2_environment):
latex_file = r.LaTeXFile(rendercv_data_model, jinja2_environment)
latex_file.get_latex_code()
latex_file.generate_latex_file(tmp_path / "test.tex")
2024-02-14 18:50:00 +00:00
def test_markdown_file_class(tmp_path, rendercv_data_model, jinja2_environment):
latex_file = r.MarkdownFile(rendercv_data_model, jinja2_environment)
latex_file.get_markdown_code()
latex_file.generate_markdown_file(tmp_path / "test.tex")
@pytest.mark.parametrize(
"string, expected_string",
[
("My Text", "My Text"),
("My # Text", "My \\# Text"),
("My % Text", "My \\% Text"),
("My & Text", "My \\& Text"),
("My ~ Text", "My \\textasciitilde{} Text"),
("##%%&&~~", "\\#\\#\\%\\%\\&\\&\\textasciitilde{}\\textasciitilde{}"),
(
"[link](you shouldn't escape whatever is in here & % # ~)",
"[link](you shouldn't escape whatever is in here & % # ~)",
),
],
)
def test_escape_latex_characters(string, expected_string):
assert r.escape_latex_characters(string) == expected_string
@pytest.mark.parametrize(
"markdown_string, expected_latex_string",
[
("My Text", "My Text"),
("**My** Text", "\\textbf{My} Text"),
("*My* Text", "\\textit{My} Text"),
("***My*** Text", "\\textit{\\textbf{My}} Text"),
("[My](https://myurl.com) Text", "\\href{https://myurl.com}{My} Text"),
("`My` Text", "\\texttt{My} Text"),
(
"[**My** *Text* ***Is*** `Here`](https://myurl.com)",
(
"\\href{https://myurl.com}{\\textbf{My} \\textit{Text}"
" \\textit{\\textbf{Is}} \\texttt{Here}}"
),
),
],
)
def test_markdown_to_latex(markdown_string, expected_latex_string):
assert r.markdown_to_latex(markdown_string) == expected_latex_string
2024-02-14 19:15:36 +00:00
def test_transform_markdown_data_model_to_latex_data_model(rendercv_data_model):
latex_data_model = r.transform_markdown_data_model_to_latex_data_model(
rendercv_data_model
)
assert isinstance(latex_data_model, dm.RenderCVDataModel)
assert latex_data_model.cv.name == rendercv_data_model.cv.name
assert latex_data_model.design == rendercv_data_model.design
2024-02-06 20:18:46 +00:00
@pytest.mark.parametrize(
"value, something, match_str, expected",
[
("Hello World", "textbf", None, "\\textbf{Hello World}"),
("Hello World", "textbf", "World", "Hello \\textbf{World}"),
("Hello World", "textbf", "Universe", "Hello World"),
("", "textbf", "Universe", ""),
("Hello World", "textbf", "", "Hello World"),
],
)
def test_make_matched_part_something(value, something, match_str, expected):
result = r.make_matched_part_something(value, something, match_str)
assert result == expected
@pytest.mark.parametrize(
"value, match_str, expected",
[
("Hello World", None, "\\textbf{Hello World}"),
("Hello World", "World", "Hello \\textbf{World}"),
("Hello World", "Universe", "Hello World"),
("", "Universe", ""),
("Hello World", "", "Hello World"),
],
)
def test_make_matched_part_bold(value, match_str, expected):
result = r.make_matched_part_bold(value, match_str)
assert result == expected
@pytest.mark.parametrize(
"value, match_str, expected",
[
("Hello World", None, "\\underline{Hello World}"),
("Hello World", "World", "Hello \\underline{World}"),
("Hello World", "Universe", "Hello World"),
("", "Universe", ""),
("Hello World", "", "Hello World"),
],
)
def test_make_matched_part_underlined(value, match_str, expected):
result = r.make_matched_part_underlined(value, match_str)
assert result == expected
@pytest.mark.parametrize(
"value, match_str, expected",
[
("Hello World", None, "\\textit{Hello World}"),
("Hello World", "World", "Hello \\textit{World}"),
("Hello World", "Universe", "Hello World"),
("", "Universe", ""),
("Hello World", "", "Hello World"),
],
)
def test_make_matched_part_italic(value, match_str, expected):
result = r.make_matched_part_italic(value, match_str)
assert result == expected
@pytest.mark.parametrize(
"value, match_str, expected",
[
("Hello World", None, "\\mbox{Hello World}"),
("Hello World", "World", "Hello \\mbox{World}"),
("Hello World", "Universe", "Hello World"),
("", "Universe", ""),
("Hello World", "", "Hello World"),
],
)
def test_make_matched_part_non_line_breakable(value, match_str, expected):
result = r.make_matched_part_non_line_breakable(value, match_str)
assert result == expected
@pytest.mark.parametrize(
"name, expected",
[
("John Doe", "J. Doe"),
("John Jacob Jingleheimer Schmidt", "J. J. J. Schmidt"),
("SingleName", "SingleName"),
("", ""),
],
)
def test_abbreviate_name(name, expected):
result = r.abbreviate_name(name)
assert result == expected
@pytest.mark.parametrize(
"length, divider, expected",
[
("10pt", 2, "5.0pt"),
("15cm", 3, "5.0cm"),
("20mm", 4, "5.0mm"),
("25ex", 5, "5.0ex"),
("30em", 6, "5.0em"),
("10pt", 3, "3.33pt"),
("10pt", 4, "2.5pt"),
("0pt", 1, "0.0pt"),
],
)
def test_divide_length_by(length, divider, expected):
result = r.divide_length_by(length, divider)
assert math.isclose(
float(result[:-2]), float(expected[:-2]), rel_tol=1e-2
), f"Expected {expected}, but got {result}"
@pytest.mark.parametrize(
"length, divider",
[("10pt", 0), ("10pt", -1), ("invalid", 4)],
)
def test_invalid_divide_length_by(length, divider):
with pytest.raises(ValueError):
r.divide_length_by(length, divider)
2024-02-09 19:14:46 +00:00
def test_get_an_item_with_a_specific_attribute_value():
entry_objects = [
dm.OneLineEntry(
name="Test1",
details="Test2",
),
dm.OneLineEntry(
name="Test3",
details="Test4",
),
]
result = r.get_an_item_with_a_specific_attribute_value(
entry_objects, "name", "Test3"
)
assert result == entry_objects[1]
result = r.get_an_item_with_a_specific_attribute_value(
entry_objects, "name", "DoesntExist"
)
assert result is None
with pytest.raises(AttributeError):
r.get_an_item_with_a_specific_attribute_value(entry_objects, "invalid", "Test5")
2024-02-06 20:18:46 +00:00
def test_setup_jinja2_environment():
env = r.setup_jinja2_environment()
# Check if the returned object is a jinja2.Environment instance
assert isinstance(env, jinja2.Environment)
# Check if the custom delimiters are correctly set
assert env.block_start_string == "((*"
assert env.block_end_string == "*))"
assert env.variable_start_string == "<<"
assert env.variable_end_string == ">>"
assert env.comment_start_string == "((#"
assert env.comment_end_string == "#))"
# Check if the custom filters are correctly set
assert "make_it_bold" in env.filters
assert "make_it_underlined" in env.filters
assert "make_it_italic" in env.filters
assert "make_it_nolinebreak" in env.filters
assert "make_it_something" in env.filters
assert "divide_length_by" in env.filters
assert "abbreviate_name" in env.filters
2024-02-07 18:18:04 +00:00
assert "get_an_item_with_a_specific_attribute_value" in env.filters
2024-02-06 20:18:46 +00:00
2024-02-07 18:18:04 +00:00
2024-02-11 15:12:34 +00:00
themes = ["classic", "moderncv", "mcdowell"]
2024-02-14 19:22:26 +00:00
update_reference_files = False
2024-02-07 18:18:04 +00:00
@pytest.mark.parametrize(
"theme_name",
themes,
)
@time_machine.travel("2024-01-01")
def test_generate_latex_file(tmp_path, reference_files_directory_path, theme_name):
2024-02-11 22:08:32 +00:00
reference_latex_files_directory_path = (
reference_files_directory_path / "latex_files"
)
2024-02-07 18:18:04 +00:00
file_name = f"{theme_name}_theme_CV.tex"
2024-02-09 19:14:46 +00:00
output_file_path = tmp_path / "make_sure_it_generates_the_directory" / file_name
2024-02-11 22:08:32 +00:00
reference_file_path = reference_latex_files_directory_path / file_name
2024-02-07 18:18:04 +00:00
data_model = dm.RenderCVDataModel(
cv=dm.CurriculumVitae(name=f"{theme_name} theme"),
2024-02-11 15:12:34 +00:00
design={"theme": theme_name},
2024-02-07 18:18:04 +00:00
)
2024-02-09 19:14:46 +00:00
r.generate_latex_file(data_model, tmp_path / "make_sure_it_generates_the_directory")
2024-02-11 18:36:56 +00:00
# Update the reference files if update_reference_files is True
if update_reference_files:
2024-02-11 22:08:32 +00:00
r.generate_latex_file(data_model, reference_latex_files_directory_path)
2024-02-07 18:18:04 +00:00
assert filecmp.cmp(output_file_path, reference_file_path)
2024-02-14 18:50:00 +00:00
@pytest.mark.parametrize(
"theme_name",
themes,
)
@time_machine.travel("2024-01-01")
def test_generate_markdown_file(tmp_path, reference_files_directory_path, theme_name):
reference_latex_files_directory_path = (
reference_files_directory_path / "markdown_and_html_files"
)
file_name = f"{theme_name}_theme_CV.md"
output_file_path = tmp_path / "make_sure_it_generates_the_directory" / file_name
reference_file_path = reference_latex_files_directory_path / file_name
data_model = dm.RenderCVDataModel(
cv=dm.CurriculumVitae(name=f"{theme_name} theme"),
design={"theme": theme_name},
)
r.generate_markdown_file(
data_model, tmp_path / "make_sure_it_generates_the_directory"
)
# Update the reference files if update_reference_files is True
if update_reference_files:
r.generate_markdown_file(data_model, reference_latex_files_directory_path)
assert filecmp.cmp(output_file_path, reference_file_path)
2024-02-07 18:18:04 +00:00
@pytest.mark.parametrize(
"theme_name",
themes,
)
def test_copy_theme_files_to_output_directory(
tmp_path, reference_files_directory_path, theme_name
):
reference_directory_path = (
reference_files_directory_path / f"{theme_name}_theme_auxiliary_files"
)
r.copy_theme_files_to_output_directory(theme_name, tmp_path)
2024-02-11 18:36:56 +00:00
# Update the reference files if update_reference_files is True
if update_reference_files:
reference_directory_path.mkdir(parents=True, exist_ok=True)
2024-02-11 22:08:32 +00:00
r.copy_theme_files_to_output_directory(theme_name, reference_directory_path)
2024-02-07 18:18:04 +00:00
assert filecmp.dircmp(tmp_path, reference_directory_path).diff_files == []
2024-02-11 18:36:56 +00:00
def test_copy_theme_files_to_output_directory_custom_theme(
tmp_path, reference_files_directory_path
):
theme_name = "dummytheme"
reference_directory = (
reference_files_directory_path / f"{theme_name}_theme_auxiliary_files"
)
# change current working directory to the refefence_files_directory_path:
os.chdir(reference_files_directory_path)
r.copy_theme_files_to_output_directory(theme_name, tmp_path)
2024-02-14 18:50:00 +00:00
# Update the reference files if update_reference_files is True
if update_reference_files:
r.copy_theme_files_to_output_directory(theme_name, reference_directory)
assert filecmp.dircmp(tmp_path, reference_directory).left_only == []
assert filecmp.dircmp(tmp_path, reference_directory).right_only == []
2024-02-11 18:36:56 +00:00
2024-02-07 18:18:04 +00:00
@pytest.mark.parametrize(
"theme_name",
themes,
)
@time_machine.travel("2024-01-01")
def test_generate_latex_file_and_copy_theme_files(
tmp_path, reference_files_directory_path, theme_name
):
reference_directory = reference_files_directory_path / f"{theme_name}_theme_full"
data_model = dm.RenderCVDataModel(
cv=dm.CurriculumVitae(name=f"{theme_name} theme"),
2024-02-11 15:12:34 +00:00
design={"theme": theme_name},
2024-02-07 18:18:04 +00:00
)
r.generate_latex_file_and_copy_theme_files(data_model, tmp_path)
2024-02-11 18:36:56 +00:00
# Update the reference files if update_reference_files is True
if update_reference_files:
r.generate_latex_file_and_copy_theme_files(data_model, reference_directory)
2024-02-07 18:18:04 +00:00
assert filecmp.dircmp(tmp_path, reference_directory).diff_files == []
@pytest.mark.parametrize(
"theme_name",
themes,
)
2024-02-11 22:08:32 +00:00
@time_machine.travel("2024-01-01")
2024-02-07 18:18:04 +00:00
def test_latex_to_pdf(tmp_path, reference_files_directory_path, theme_name):
reference_directory = reference_files_directory_path / f"{theme_name}_theme_full"
reference_pdf_file_path = reference_directory / f"{theme_name}_theme_CV.pdf"
shutil.copytree(reference_directory, tmp_path, dirs_exist_ok=True)
output_pdf_file_path = r.latex_to_pdf(tmp_path / f"{theme_name}_theme_CV.tex")
2024-02-11 18:36:56 +00:00
# Update the reference files if update_reference_files is True
if update_reference_files:
2024-02-11 22:08:32 +00:00
reference_pdf_file_path = r.latex_to_pdf(
2024-02-11 18:36:56 +00:00
reference_directory / f"{theme_name}_theme_CV.tex"
)
2024-02-07 18:18:04 +00:00
2024-02-11 22:08:32 +00:00
text1 = pypdf.PdfReader(output_pdf_file_path).pages[0].extract_text()
text2 = pypdf.PdfReader(reference_pdf_file_path).pages[0].extract_text()
assert text1 == text2
2024-02-09 19:14:46 +00:00
2024-02-14 19:15:36 +00:00
def test_latex_to_pdf_invalid_latex_file():
with pytest.raises(FileNotFoundError):
file_path = pathlib.Path("file_doesnt_exist.tex")
r.latex_to_pdf(file_path)
2024-02-14 18:50:00 +00:00
@pytest.mark.parametrize(
"theme_name",
themes,
)
@time_machine.travel("2024-01-01")
def test_markdown_to_html(tmp_path, reference_files_directory_path, theme_name):
reference_directory = reference_files_directory_path / "markdown_and_html_files"
2024-02-14 19:22:26 +00:00
reference_html_file_path = (
reference_directory / f"{theme_name}_theme_CV_PASTETOGRAMMARLY.html"
)
2024-02-14 18:50:00 +00:00
shutil.copytree(reference_directory, tmp_path, dirs_exist_ok=True)
output_html_file_path = r.markdown_to_html(tmp_path / f"{theme_name}_theme_CV.md")
# Update the reference files if update_reference_files is True
if update_reference_files:
reference_html_file_path = r.markdown_to_html(
reference_directory / f"{theme_name}_theme_CV.md"
)
text1 = output_html_file_path.read_text()
text2 = reference_html_file_path.read_text()
assert text1 == text2