mirror of https://github.com/eyhc1/rendercv.git
611 lines
17 KiB
Python
611 lines
17 KiB
Python
import os
|
|
import shutil
|
|
import subprocess
|
|
import sys
|
|
from datetime import date as Date
|
|
|
|
import pydantic
|
|
import ruamel.yaml
|
|
import pytest
|
|
import typer.testing
|
|
|
|
import rendercv.cli as cli
|
|
import rendercv.data_models as dm
|
|
from rendercv import __version__
|
|
|
|
|
|
def run_render_command(input_file_path, working_path, extra_arguments=[]):
|
|
# copy input file to the temporary directory to create the output directory there:
|
|
if not input_file_path == working_path / input_file_path.name:
|
|
shutil.copy(input_file_path, working_path)
|
|
|
|
# change the current working directory to the temporary directory:
|
|
os.chdir(working_path)
|
|
|
|
result = runner.invoke(cli.app, ["render", "John_Doe_CV.yaml"] + extra_arguments)
|
|
|
|
return result
|
|
|
|
|
|
def test_welcome():
|
|
cli.welcome()
|
|
|
|
|
|
def test_warning():
|
|
cli.warning("This is a warning message.")
|
|
|
|
|
|
def test_error():
|
|
with pytest.raises(typer.Exit):
|
|
cli.error("This is an error message.")
|
|
|
|
|
|
def test_error_without_text():
|
|
with pytest.raises(typer.Exit):
|
|
cli.error()
|
|
|
|
|
|
def test_error_without_text_with_exception():
|
|
with pytest.raises(typer.Exit):
|
|
cli.error(exception=ValueError("This is an error message."))
|
|
|
|
|
|
def test_information():
|
|
cli.information("This is an information message.")
|
|
|
|
|
|
def test_get_error_message_and_location_and_value_from_a_custom_error():
|
|
error_string = "('error message', 'location', 'value')"
|
|
result = cli.get_error_message_and_location_and_value_from_a_custom_error(
|
|
error_string
|
|
)
|
|
assert result == ("error message", "location", "value")
|
|
|
|
error_string = """("er'ror message", 'location', 'value')"""
|
|
result = cli.get_error_message_and_location_and_value_from_a_custom_error(
|
|
error_string
|
|
)
|
|
assert result == ("er'ror message", "location", "value")
|
|
|
|
error_string = "error message"
|
|
result = cli.get_error_message_and_location_and_value_from_a_custom_error(
|
|
error_string
|
|
)
|
|
assert result == (None, None, None)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"data_model_class, invalid_model",
|
|
[
|
|
(
|
|
dm.EducationEntry,
|
|
{
|
|
"area": "Mechanical Engineering",
|
|
"extra": "Extra",
|
|
},
|
|
),
|
|
(
|
|
dm.ExperienceEntry,
|
|
{
|
|
"company": "CERN",
|
|
},
|
|
),
|
|
(
|
|
dm.ExperienceEntry,
|
|
{
|
|
"position": "Researcher",
|
|
},
|
|
),
|
|
(
|
|
dm.ExperienceEntry,
|
|
{
|
|
"position": Date(2020, 10, 1),
|
|
},
|
|
),
|
|
(
|
|
dm.ExperienceEntry,
|
|
{
|
|
"company": "CERN",
|
|
"position": "Researcher",
|
|
"stat_date": "2023-12-08",
|
|
"end_date": "INVALID END DATE",
|
|
},
|
|
),
|
|
(
|
|
dm.ExperienceEntry,
|
|
{
|
|
"company": "CERN",
|
|
"position": "Researcher",
|
|
"highlights": "This is not a list.",
|
|
},
|
|
),
|
|
(
|
|
dm.PublicationEntry,
|
|
{
|
|
"doi": "10.1109/TASC.2023.3340648",
|
|
},
|
|
),
|
|
(
|
|
dm.ExperienceEntry,
|
|
{
|
|
"authors": ["John Doe", "Jane Doe"],
|
|
},
|
|
),
|
|
(
|
|
dm.OneLineEntry,
|
|
{
|
|
"name": "My One Line Entry",
|
|
},
|
|
),
|
|
(
|
|
dm.CurriculumVitae,
|
|
{
|
|
"name": "John Doe",
|
|
"sections": {
|
|
"education": [
|
|
{
|
|
"institution": "Boğaziçi University",
|
|
"area": "Mechanical Engineering",
|
|
"degree": "BS",
|
|
"date": "2028-12-08",
|
|
},
|
|
{
|
|
"degree": "BS",
|
|
},
|
|
]
|
|
},
|
|
},
|
|
),
|
|
(
|
|
dm.RenderCVDataModel,
|
|
{
|
|
"cv": {
|
|
"name": "John Doe",
|
|
},
|
|
"design": {"theme": "UPPERCASE"},
|
|
},
|
|
),
|
|
],
|
|
)
|
|
def test_handle_validation_error(data_model_class, invalid_model):
|
|
try:
|
|
data_model_class(**invalid_model)
|
|
except pydantic.ValidationError as e:
|
|
with pytest.raises(typer.Exit):
|
|
cli.handle_validation_error(e)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"exception",
|
|
[
|
|
ruamel.yaml.YAMLError("message"),
|
|
RuntimeError("message"),
|
|
FileNotFoundError("message"),
|
|
ValueError("message"),
|
|
UnicodeDecodeError("utf-8", b"", 1, 2, "message"),
|
|
],
|
|
)
|
|
def test_handle_exceptions(exception):
|
|
@cli.handle_exceptions
|
|
def function_that_raises_exception():
|
|
raise exception
|
|
|
|
with pytest.raises(typer.Exit):
|
|
function_that_raises_exception()
|
|
|
|
|
|
def test_live_progress_reporter_class():
|
|
with cli.LiveProgressReporter(number_of_steps=3) as progress:
|
|
progress.start_a_step("Test step 1")
|
|
progress.finish_the_current_step()
|
|
|
|
progress.start_a_step("Test step 2")
|
|
progress.finish_the_current_step()
|
|
|
|
progress.start_a_step("Test step 3")
|
|
progress.finish_the_current_step()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"folder_name",
|
|
["markdown"] + dm.available_themes,
|
|
)
|
|
def test_copy_templates(tmp_path, folder_name):
|
|
copied_path = cli.copy_templates(
|
|
folder_name=folder_name,
|
|
copy_to=tmp_path,
|
|
)
|
|
assert copied_path.exists()
|
|
|
|
|
|
def test_copy_templates_with_new_folder_name(tmp_path):
|
|
copied_path = cli.copy_templates(
|
|
folder_name="markdown",
|
|
copy_to=tmp_path,
|
|
new_folder_name="new_folder",
|
|
)
|
|
assert copied_path.exists()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"folder_name",
|
|
["markdown"] + dm.available_themes,
|
|
)
|
|
def test_copy_templates_destinations_exist(tmp_path, folder_name):
|
|
(tmp_path / folder_name).mkdir()
|
|
|
|
copied_path = cli.copy_templates(
|
|
folder_name=folder_name,
|
|
copy_to=tmp_path,
|
|
)
|
|
|
|
assert copied_path is None
|
|
|
|
|
|
runner = typer.testing.CliRunner()
|
|
|
|
|
|
def test_render_command(tmp_path, input_file_path):
|
|
result = run_render_command(
|
|
input_file_path,
|
|
tmp_path,
|
|
)
|
|
|
|
output_folder_path = tmp_path / "rendercv_output"
|
|
pdf_file_path = output_folder_path / "John_Doe_CV.pdf"
|
|
latex_file_path = output_folder_path / "John_Doe_CV.tex"
|
|
markdown_file_path = output_folder_path / "John_Doe_CV.md"
|
|
html_file_path = output_folder_path / "John_Doe_CV_PASTETOGRAMMARLY.html"
|
|
png_file_path = output_folder_path / "John_Doe_CV_1.png"
|
|
|
|
assert output_folder_path.exists()
|
|
assert pdf_file_path.exists()
|
|
assert latex_file_path.exists()
|
|
assert markdown_file_path.exists()
|
|
assert html_file_path.exists()
|
|
assert png_file_path.exists()
|
|
assert "Your CV is rendered!" in result.stdout
|
|
|
|
|
|
def test_render_command_with_relative_input_file_path(tmp_path, input_file_path):
|
|
new_folder = tmp_path / "another_folder"
|
|
new_folder.mkdir()
|
|
new_input_file_path = new_folder / input_file_path.name
|
|
|
|
shutil.copy(input_file_path, new_input_file_path)
|
|
|
|
os.chdir(tmp_path)
|
|
result = runner.invoke(
|
|
cli.app, ["render", str(new_input_file_path.relative_to(tmp_path))]
|
|
)
|
|
|
|
output_folder_path = tmp_path / "rendercv_output"
|
|
pdf_file_path = output_folder_path / "John_Doe_CV.pdf"
|
|
latex_file_path = output_folder_path / "John_Doe_CV.tex"
|
|
markdown_file_path = output_folder_path / "John_Doe_CV.md"
|
|
html_file_path = output_folder_path / "John_Doe_CV_PASTETOGRAMMARLY.html"
|
|
png_file_path = output_folder_path / "John_Doe_CV_1.png"
|
|
|
|
assert output_folder_path.exists()
|
|
assert pdf_file_path.exists()
|
|
assert latex_file_path.exists()
|
|
assert markdown_file_path.exists()
|
|
assert html_file_path.exists()
|
|
assert png_file_path.exists()
|
|
assert "Your CV is rendered!" in result.stdout
|
|
|
|
|
|
def test_render_command_with_different_output_path(input_file_path, tmp_path):
|
|
result = run_render_command(
|
|
input_file_path,
|
|
tmp_path,
|
|
[
|
|
"--output-folder-name",
|
|
"test",
|
|
],
|
|
)
|
|
|
|
output_folder_path = tmp_path / "test"
|
|
|
|
assert result.exit_code == 0
|
|
assert output_folder_path.exists()
|
|
assert "Your CV is rendered!" in result.stdout
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("option", "file_name"),
|
|
[
|
|
("--pdf-path", "test.pdf"),
|
|
("--latex-path", "test.tex"),
|
|
("--markdown-path", "test.md"),
|
|
("--html-path", "test.html"),
|
|
("--png-path", "test.png"),
|
|
],
|
|
)
|
|
def test_render_command_with_different_output_path_for_each_file(
|
|
option, file_name, tmp_path, input_file_path
|
|
):
|
|
run_render_command(
|
|
input_file_path,
|
|
tmp_path,
|
|
[
|
|
option,
|
|
file_name,
|
|
],
|
|
)
|
|
|
|
file_path = tmp_path / file_name
|
|
|
|
assert file_path.exists()
|
|
|
|
|
|
def test_render_command_with_custom_png_path_multiple_pages(tmp_path):
|
|
# create a new input file (for a CV with multiple pages) in the temporary directory:
|
|
os.chdir(tmp_path)
|
|
runner.invoke(cli.app, ["new", "John Doe"])
|
|
input_file_path = tmp_path / "John_Doe_CV.yaml"
|
|
|
|
run_render_command(
|
|
input_file_path,
|
|
tmp_path,
|
|
[
|
|
"--png-path",
|
|
"test.png",
|
|
],
|
|
)
|
|
|
|
png_page1_file_path = tmp_path / "test_1.png"
|
|
png_page2_file_path = tmp_path / "test_2.png"
|
|
|
|
assert png_page1_file_path.exists()
|
|
assert png_page2_file_path.exists()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("option", "file_name"),
|
|
[
|
|
("--dont-generate-markdown", "John_Doe_CV.md"),
|
|
("--dont-generate-html", "John_Doe_CV_PASTETOGRAMMARLY.html"),
|
|
("--dont-generate-png", "John_Doe_CV_1.png"),
|
|
],
|
|
)
|
|
def test_render_command_with_dont_generate_files(
|
|
tmp_path, input_file_path, option, file_name
|
|
):
|
|
run_render_command(
|
|
input_file_path,
|
|
tmp_path,
|
|
[
|
|
option,
|
|
],
|
|
)
|
|
|
|
file_path = tmp_path / "rendercv_output" / file_name
|
|
|
|
assert not file_path.exists()
|
|
|
|
|
|
def test_render_command_with_local_latex_command(tmp_path, input_file_path):
|
|
run_render_command(
|
|
input_file_path,
|
|
tmp_path,
|
|
[
|
|
"--use-local-latex-command",
|
|
"pdflatex",
|
|
],
|
|
)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"invalid_arguments",
|
|
[
|
|
["--keywithoutvalue"],
|
|
["--key", "value", "--keywithoutvalue"],
|
|
["keywithoutdashes", "value"],
|
|
["--cv.phone", "invalidphonenumber"],
|
|
["--cv.sections.arbitrary.10", "value"],
|
|
],
|
|
)
|
|
def test_render_command_with_invalid_arguments(
|
|
tmp_path, input_file_path, invalid_arguments
|
|
):
|
|
result = run_render_command(
|
|
input_file_path,
|
|
tmp_path,
|
|
invalid_arguments,
|
|
)
|
|
|
|
assert (
|
|
"There is a problem with the extra arguments!" in result.stdout
|
|
or "should start with double dashes!" in result.stdout
|
|
or "does not exist in the data model!" in result.stdout
|
|
or "There are some errors in the data model!" in result.stdout
|
|
)
|
|
|
|
|
|
def test_new_command(tmp_path):
|
|
# change the current working directory to the temporary directory:
|
|
os.chdir(tmp_path)
|
|
runner.invoke(cli.app, ["new", "Jahn Doe"])
|
|
|
|
markdown_source_files_path = tmp_path / "markdown"
|
|
theme_source_files_path = tmp_path / "classic"
|
|
input_file_path = tmp_path / "Jahn_Doe_CV.yaml"
|
|
|
|
assert markdown_source_files_path.exists()
|
|
assert theme_source_files_path.exists()
|
|
assert input_file_path.exists()
|
|
|
|
|
|
def test_new_command_with_invalid_theme(tmp_path):
|
|
# change the current working directory to the temporary directory:
|
|
os.chdir(tmp_path)
|
|
|
|
result = runner.invoke(cli.app, ["new", "Jahn Doe", "--theme", "invalid_theme"])
|
|
|
|
assert "The theme should be one of the following" in result.stdout
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("option", "folder_name"),
|
|
[
|
|
("--dont-create-theme-source-files", "classic"),
|
|
("--dont-create-markdown-source-files", "markdown"),
|
|
],
|
|
)
|
|
def test_new_command_with_dont_create_files(tmp_path, option, folder_name):
|
|
# change the current working directory to the temporary directory:
|
|
os.chdir(tmp_path)
|
|
runner.invoke(cli.app, ["new", "Jahn Doe", option])
|
|
|
|
source_files_path = tmp_path / folder_name
|
|
|
|
assert not source_files_path.exists()
|
|
|
|
|
|
def test_new_command_with_only_input_file(tmp_path):
|
|
# change the current working directory to the temporary directory:
|
|
os.chdir(tmp_path)
|
|
runner.invoke(
|
|
cli.app,
|
|
[
|
|
"new",
|
|
"Jahn Doe",
|
|
"--dont-create-markdown-source-files",
|
|
"--dont-create-theme-source-files",
|
|
],
|
|
)
|
|
|
|
markdown_source_files_path = tmp_path / "markdown"
|
|
theme_source_files_path = tmp_path / "classic"
|
|
input_file_path = tmp_path / "Jahn_Doe_CV.yaml"
|
|
|
|
assert not markdown_source_files_path.exists()
|
|
assert not theme_source_files_path.exists()
|
|
assert input_file_path.exists()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"file_or_folder_name",
|
|
[
|
|
"Jahn_Doe_CV.yaml",
|
|
"markdown",
|
|
"classic",
|
|
],
|
|
)
|
|
def test_new_command_with_existing_files(tmp_path, file_or_folder_name):
|
|
# change the current working directory to the temporary directory:
|
|
os.chdir(tmp_path)
|
|
|
|
if file_or_folder_name == "Jahn_Doe_CV.yaml":
|
|
(tmp_path / file_or_folder_name).touch()
|
|
else:
|
|
(tmp_path / file_or_folder_name).mkdir()
|
|
|
|
result = runner.invoke(cli.app, ["new", "Jahn Doe"])
|
|
|
|
assert "already exists!" in result.stdout
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"based_on",
|
|
dm.available_themes,
|
|
)
|
|
def test_create_theme_command(tmp_path, input_file_path, based_on):
|
|
# change the current working directory to the temporary directory:
|
|
os.chdir(tmp_path)
|
|
|
|
runner.invoke(cli.app, ["create-theme", "newtheme", "--based-on", based_on])
|
|
|
|
new_theme_source_files_path = tmp_path / "newtheme"
|
|
|
|
assert new_theme_source_files_path.exists()
|
|
|
|
# test if the new theme is actually working:
|
|
input_file_path = shutil.copy(input_file_path, tmp_path)
|
|
|
|
result = runner.invoke(
|
|
cli.app, ["render", str(input_file_path), "--design", "{'theme':'newtheme'}"]
|
|
)
|
|
|
|
output_folder_path = tmp_path / "rendercv_output"
|
|
pdf_file_path = output_folder_path / "John_Doe_CV.pdf"
|
|
latex_file_path = output_folder_path / "John_Doe_CV.tex"
|
|
markdown_file_path = output_folder_path / "John_Doe_CV.md"
|
|
html_file_path = output_folder_path / "John_Doe_CV_PASTETOGRAMMARLY.html"
|
|
png_file_path = output_folder_path / "John_Doe_CV_1.png"
|
|
|
|
assert output_folder_path.exists()
|
|
assert pdf_file_path.exists()
|
|
assert latex_file_path.exists()
|
|
assert markdown_file_path.exists()
|
|
assert html_file_path.exists()
|
|
assert png_file_path.exists()
|
|
assert "Your CV is rendered!" in result.stdout
|
|
|
|
|
|
def test_create_theme_command_invalid_based_on_theme(tmp_path):
|
|
result = runner.invoke(
|
|
cli.app, ["create-theme", "newtheme", "--based-on", "invalid_theme"]
|
|
)
|
|
|
|
assert "is not in the list of available themes" in result.stdout
|
|
|
|
|
|
def test_create_theme_command_theme_already_exists(tmp_path):
|
|
# change the current working directory to the temporary directory:
|
|
os.chdir(tmp_path)
|
|
|
|
(tmp_path / "newtheme").mkdir()
|
|
|
|
result = runner.invoke(cli.app, ["create-theme", "newtheme"])
|
|
|
|
assert "already exists!" in result.stdout
|
|
|
|
|
|
def test_main_file():
|
|
subprocess.run([sys.executable, "-m", "rendercv", "--help"], check=True)
|
|
|
|
|
|
def test_get_latest_version_number_from_pypi():
|
|
version = cli.get_latest_version_number_from_pypi()
|
|
assert isinstance(version, str)
|
|
|
|
|
|
def test_if_welcome_prints_new_version_available(monkeypatch):
|
|
monkeypatch.setattr(cli, "get_latest_version_number_from_pypi", lambda: "99999")
|
|
import io
|
|
import contextlib
|
|
|
|
with contextlib.redirect_stdout(io.StringIO()) as f:
|
|
cli.welcome()
|
|
output = f.getvalue()
|
|
|
|
assert "A new version of RenderCV is available!" in output
|
|
|
|
|
|
def test_rendercv_version_when_there_is_a_new_version(monkeypatch):
|
|
monkeypatch.setattr(cli, "get_latest_version_number_from_pypi", lambda: "99999")
|
|
|
|
result = runner.invoke(cli.app, ["--version"])
|
|
|
|
assert "A new version of RenderCV is available!" in result.stdout
|
|
|
|
|
|
def test_rendercv_version_when_there_is_not_a_new_version(monkeypatch):
|
|
monkeypatch.setattr(cli, "get_latest_version_number_from_pypi", lambda: __version__)
|
|
|
|
result = runner.invoke(cli.app, ["--version"])
|
|
|
|
assert __version__ in result.stdout
|
|
|
|
|
|
def test_warn_if_new_version_is_available(monkeypatch):
|
|
monkeypatch.setattr(cli, "get_latest_version_number_from_pypi", lambda: __version__)
|
|
|
|
assert not cli.warn_if_new_version_is_available()
|
|
|
|
monkeypatch.setattr(cli, "get_latest_version_number_from_pypi", lambda: "999")
|
|
|
|
assert cli.warn_if_new_version_is_available()
|