mirror of
https://github.com/rendercv/rendercv.git
synced 2026-02-07 03:51:26 -05:00
* Rename `data` folder with schema * Start refactoring data models * Work on entry models * Keep working on entries * Keep working on data models * Push old data files * Keep working on data models * First draft of schema.cv * Keep working on schema * Keep working on schema * Improve schema.models * Keep working on rendercv.schema * Work on schema.design * Keep working on rendercv.schema * Complete variant_class_generator * Keep working rendercv.schema * Keep working on rendercv.schema * Final touches to rendercv.schema * Improve json schema descriptions in rendercv.schema * Start working on rendercv.schema tests * Keep implementing rendercv.schema tests * Add more tests for rendercv.schema * Improve rendercv.schema * Improve docstrings and comments in rendercv.schema * Implement better pydantic error handling in `rendercv.schema` * Improve variant class system * Fix rendercv.schema tests * Start working on rendercv.templater * Update template names * Switching to new rendercv typst template soon * Work on new templater * Rename renderer with renderer_old * Don't use utils in rendercv.schema * Complete connections * Update renderer folder structure * Work on new renderer * Work on new renderer * Date processing on new renderer * Improve date processing, support multiple emails, phones, and websites * Improve markdown to Typst * Complete entry template processing * Time span computation in new renderer * Better entry templates * Setup new templates * Improve rendercv.schema * Start adding tests for rendercv.renderer * New markdown parser! * Improve markdown to typst conversion * Finalize markdown parser * Add new test files for rendercv.renderer * Fix cv and connections * Add connections test * Improve connection tests * Improve entry templates * Add model processor tests * Improve templater * Rename old folders * Improve schema * Add file generation logic to renderer * Fix naming issues * Fix schema tests * Add path type tests * Add font family and typst dimension type tests * Rename old tests * Fix design tests * Start integration testing of renderer * Improve entry tempates * Handle nested highlights properly * Finalize Typst preamble template * Start working on new CLI * Remove old test files * Implement override dictionary in new schema * Start working on new CLI * Better prints on render command * New structure * New render printer * Add all the commands to new CLI * Work on new command in new cli * Improve new command * Add error handler to new cli * Work on create theme command * Complete create theme command * Remove old source files * Improve exceptions * Create new docs * Add writing tests guide * Fix cli printer and write tests * Test copy templates * Add app tests * Bring back accidentally removed files * Imporve cli and tests * Fix path issues * Improve * Improve * Add reference file comparison tests * Fix path resolver * Start working on test_pdf_png * Implement comparison of multiple files (png) * Start testing typst * Fix templating issues * Fix header and entry templates issues * Implement short second rows * Fix date issues * Fix nested bullets and add summary * Update testdata * Implement footer * Update testdata * Reimagined design and locale schema, first iteration * Reimagined design and locale second iteration * Update design and locale schemas * Adapt templater to the new design and locale * Fix tests * Update lib.typ and testdata for the new locale and design * Implement proper tests with all combinations of entries * Remove some docstrings * fix connections logic * Improve * Start working on examples * Update testdata * Fix long second row issue * fix templating issues * Fix lib.typ issues * Update testdata * Fix clean_trailing_parts * Update test cv * update test cv * Update theme defaults * update schema and fix moderncv * Fix moderncv issues * Update testdata * Update testdata and examples * Fix issues about photo * Fix typst photo path issues * improve entry templates from yaml * add new locale * Rename writing tests doc * Update writing tests * Improve tests * Add more cli tests * Increase test coverage * Rename variant pydantic model generator * Improve tests * Update testdata and improve tests * Format, fix pre-commit errors * Fix scripts and update entry figures * Improve tests * Write docstrings of schema * Write schema docstrings * Setup api reference * Start working on new docs * Work on docs * Improve progress panel of render command * Finalize new docs index * Complete CLI docs * Work on YAML input structure page * Finalize user guide * Start working on developer guide * Improve api reference * Improve developer guide * Improve developer guide * Improve developer gide * Improve developer guide * Improve developer guide * Update developer guide * Improve developer guide * Improve developer guide * Improve developer guide * Developer guide first draft * update developer guide * Update examples * Update testdata * Handle wrong installation (rendercv instead of rendercv[full]) * Remove unnecessary files * Write set up vs code page * Update README.md * Change docs description * Compress design options gif * minor updates * Polish all the json schema descriptions * Update testdata and examples * Remove some emdashed from docs * Add whatsapp support * Add TestEscapeTypstCharacters to tests * Implement custom connections * Add page break before sections feature * Revert page break before sections feature * Rebase to main * Fix social network tests, update schema
221 lines
7.2 KiB
Python
221 lines
7.2 KiB
Python
import pathlib
|
|
|
|
import pydantic
|
|
import pytest
|
|
|
|
from rendercv.schema.models.path import (
|
|
ExistingPathRelativeToInput,
|
|
PlannedPathRelativeToInput,
|
|
)
|
|
from rendercv.schema.models.validation_context import ValidationContext
|
|
|
|
|
|
@pytest.fixture
|
|
def existing_file(tmp_path):
|
|
existing = tmp_path / "existing.txt"
|
|
existing.touch()
|
|
return existing
|
|
|
|
|
|
@pytest.fixture
|
|
def context_with_input_file(tmp_path):
|
|
return {"context": ValidationContext(input_file_path=tmp_path / "input.yaml")}
|
|
|
|
|
|
existing_input_relative_path_adapter = pydantic.TypeAdapter(ExistingPathRelativeToInput)
|
|
planned_input_relative_path_adapter = pydantic.TypeAdapter(PlannedPathRelativeToInput)
|
|
|
|
|
|
class TestExistingInputRelativePath:
|
|
def test_absolute_path_to_existing_file(
|
|
self, existing_file, context_with_input_file
|
|
):
|
|
result = existing_input_relative_path_adapter.validate_python(
|
|
existing_file, context=context_with_input_file
|
|
)
|
|
|
|
assert result == existing_file
|
|
|
|
def test_absolute_path_to_nonexistent_file_raises_error(
|
|
self, tmp_path, context_with_input_file
|
|
):
|
|
nonexistent = tmp_path / "nonexistent.txt"
|
|
|
|
with pytest.raises(pydantic.ValidationError, match="does not exist"):
|
|
existing_input_relative_path_adapter.validate_python(
|
|
nonexistent, context=context_with_input_file
|
|
)
|
|
|
|
def test_relative_path_to_existing_file(self, tmp_path, context_with_input_file):
|
|
existing = tmp_path / "relative_existing.txt"
|
|
existing.touch()
|
|
relative_path = pathlib.Path("relative_existing.txt")
|
|
|
|
result = existing_input_relative_path_adapter.validate_python(
|
|
relative_path, context=context_with_input_file
|
|
)
|
|
|
|
# Should be resolved relative to input_file_path directory
|
|
assert result == tmp_path / "relative_existing.txt"
|
|
assert result.exists()
|
|
|
|
def test_relative_path_to_nonexistent_file_raises_error(
|
|
self, context_with_input_file
|
|
):
|
|
relative_path = pathlib.Path("nonexistent_relative.txt")
|
|
|
|
with pytest.raises(pydantic.ValidationError, match="does not exist"):
|
|
existing_input_relative_path_adapter.validate_python(
|
|
relative_path, context=context_with_input_file
|
|
)
|
|
|
|
@pytest.mark.parametrize(
|
|
"relative_path",
|
|
[
|
|
pathlib.Path("subdir/file.txt"),
|
|
pathlib.Path("../sibling/file.txt"),
|
|
pathlib.Path("./same_dir/file.txt"),
|
|
],
|
|
)
|
|
def test_various_relative_path_formats(
|
|
self, tmp_path, context_with_input_file, relative_path
|
|
):
|
|
expected_path = tmp_path / relative_path
|
|
expected_path.parent.mkdir(parents=True, exist_ok=True)
|
|
expected_path.touch()
|
|
|
|
result = existing_input_relative_path_adapter.validate_python(
|
|
relative_path, context=context_with_input_file
|
|
)
|
|
|
|
assert result == expected_path
|
|
assert result.exists()
|
|
|
|
def test_path_string_validation(self, existing_file, context_with_input_file):
|
|
result = existing_input_relative_path_adapter.validate_python(
|
|
str(existing_file), context=context_with_input_file
|
|
)
|
|
|
|
assert result == existing_file
|
|
|
|
|
|
class TestPlannedInputRelativePath:
|
|
def test_absolute_path_to_existing_file(
|
|
self, existing_file, context_with_input_file
|
|
):
|
|
result = planned_input_relative_path_adapter.validate_python(
|
|
existing_file, context=context_with_input_file
|
|
)
|
|
|
|
assert result == existing_file
|
|
|
|
def test_absolute_path_to_nonexistent_file_is_accepted(
|
|
self, tmp_path, context_with_input_file
|
|
):
|
|
nonexistent = tmp_path / "planned_file.txt"
|
|
result = planned_input_relative_path_adapter.validate_python(
|
|
nonexistent, context=context_with_input_file
|
|
)
|
|
|
|
assert result == nonexistent
|
|
assert not result.exists()
|
|
|
|
def test_relative_path_to_existing_file(self, tmp_path, context_with_input_file):
|
|
existing = tmp_path / "relative_existing.txt"
|
|
existing.touch()
|
|
|
|
relative_path = pathlib.Path("relative_existing.txt")
|
|
|
|
result = planned_input_relative_path_adapter.validate_python(
|
|
relative_path, context=context_with_input_file
|
|
)
|
|
|
|
assert result == tmp_path / "relative_existing.txt"
|
|
assert result.exists()
|
|
|
|
def test_relative_path_to_nonexistent_file_is_accepted(
|
|
self, tmp_path, context_with_input_file
|
|
):
|
|
relative_path = pathlib.Path("planned_output.pdf")
|
|
result = planned_input_relative_path_adapter.validate_python(
|
|
relative_path, context=context_with_input_file
|
|
)
|
|
|
|
assert result == tmp_path / "planned_output.pdf"
|
|
assert not result.exists()
|
|
|
|
@pytest.mark.parametrize(
|
|
"relative_path",
|
|
[
|
|
pathlib.Path("output/result.pdf"),
|
|
pathlib.Path("../build/output.html"),
|
|
pathlib.Path("./generated/doc.md"),
|
|
],
|
|
)
|
|
def test_various_relative_path_formats_for_planned_paths(
|
|
self, tmp_path, context_with_input_file, relative_path
|
|
):
|
|
result = planned_input_relative_path_adapter.validate_python(
|
|
relative_path, context=context_with_input_file
|
|
)
|
|
|
|
expected_path = tmp_path / relative_path
|
|
assert result == expected_path
|
|
|
|
def test_path_string_validation(self, tmp_path, context_with_input_file):
|
|
planned_path = "output/generated.pdf"
|
|
result = planned_input_relative_path_adapter.validate_python(
|
|
planned_path, context=context_with_input_file
|
|
)
|
|
|
|
assert result == tmp_path / planned_path
|
|
|
|
|
|
class TestPathResolutionBehavior:
|
|
@pytest.mark.parametrize(
|
|
"path_adapter",
|
|
[existing_input_relative_path_adapter, planned_input_relative_path_adapter],
|
|
)
|
|
def test_absolute_path_remains_unchanged(
|
|
self, existing_file, context_with_input_file, path_adapter
|
|
):
|
|
result = path_adapter.validate_python(
|
|
existing_file, context=context_with_input_file
|
|
)
|
|
|
|
assert result == existing_file
|
|
assert result.is_absolute()
|
|
|
|
@pytest.mark.parametrize(
|
|
"path_adapter",
|
|
[existing_input_relative_path_adapter, planned_input_relative_path_adapter],
|
|
)
|
|
def test_relative_path_gets_resolved(
|
|
self, tmp_path, context_with_input_file, path_adapter
|
|
):
|
|
existing = tmp_path / "test.txt"
|
|
existing.touch()
|
|
relative_path = pathlib.Path("test.txt")
|
|
|
|
result = path_adapter.validate_python(
|
|
relative_path, context=context_with_input_file
|
|
)
|
|
|
|
assert result.is_absolute()
|
|
assert result == tmp_path / "test.txt"
|
|
|
|
def test_only_existing_type_validates_file_existence(
|
|
self, tmp_path, context_with_input_file
|
|
):
|
|
nonexistent_path = tmp_path / "does_not_exist.txt"
|
|
|
|
with pytest.raises(pydantic.ValidationError, match="does not exist"):
|
|
existing_input_relative_path_adapter.validate_python(
|
|
nonexistent_path, context=context_with_input_file
|
|
)
|
|
|
|
result = planned_input_relative_path_adapter.validate_python(
|
|
nonexistent_path, context=context_with_input_file
|
|
)
|
|
assert result == nonexistent_path
|