2024-01-18 22:38:19 +00:00
|
|
|
import pytest
|
2024-01-28 20:14:25 +00:00
|
|
|
import time_machine
|
2024-01-18 17:24:30 +00:00
|
|
|
|
2024-01-18 22:38:19 +00:00
|
|
|
from rendercv import data_models as dm
|
2024-01-18 17:24:30 +00:00
|
|
|
|
2024-01-28 20:14:25 +00:00
|
|
|
import pydantic
|
2024-01-18 17:24:30 +00:00
|
|
|
|
2024-01-28 20:14:25 +00:00
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def publication_entry():
|
2024-01-18 22:38:19 +00:00
|
|
|
return {
|
2024-01-28 20:14:25 +00:00
|
|
|
"title": "My Title",
|
|
|
|
"authors": ["John Doe", "Jane Doe"],
|
|
|
|
"doi": "10.1109/TASC.2023.3340648",
|
|
|
|
"date": "2023-12-08",
|
2024-01-18 22:38:19 +00:00
|
|
|
}
|
|
|
|
|
2024-01-26 18:48:08 +00:00
|
|
|
|
2024-01-28 20:14:25 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def experience_entry():
|
|
|
|
return {
|
|
|
|
"company": "CERN",
|
|
|
|
"position": "Researcher",
|
|
|
|
}
|
2024-01-26 18:48:08 +00:00
|
|
|
|
|
|
|
|
2024-01-28 20:14:25 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def education_entry():
|
|
|
|
return {
|
|
|
|
"institution": "Boğaziçi University",
|
|
|
|
"area": "Mechanical Engineering",
|
2024-01-26 18:48:08 +00:00
|
|
|
}
|
|
|
|
|
2024-01-28 20:14:25 +00:00
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def normal_entry():
|
|
|
|
return {
|
|
|
|
"name": "My Entry",
|
2024-01-18 17:24:30 +00:00
|
|
|
}
|
2024-01-28 20:14:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def one_line_entry():
|
|
|
|
return {
|
|
|
|
"name": "My One Line Entry",
|
|
|
|
"details": "My Details",
|
2024-01-18 22:38:19 +00:00
|
|
|
}
|
2024-01-28 20:14:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def text_entry():
|
|
|
|
return "My Text Entry"
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"start_date, end_date, date, expected_date_string, expected_time_span",
|
|
|
|
[
|
|
|
|
("2020-01-01", "2021-01-01", None, "Jan. 2020 to Jan. 2021", "1 year 1 month"),
|
2024-01-30 18:50:57 +00:00
|
|
|
("2020-01", "2021-01", None, "Jan. 2020 to Jan. 2021", "1 year 1 month"),
|
|
|
|
("2020-01", "2021-01-01", None, "Jan. 2020 to Jan. 2021", "1 year 1 month"),
|
|
|
|
("2020-01-01", "2021-01", None, "Jan. 2020 to Jan. 2021", "1 year 1 month"),
|
2024-01-28 20:14:25 +00:00
|
|
|
("2020-01-01", None, None, "Jan. 2020 to present", "4 years 1 month"),
|
|
|
|
("2020-02-01", "present", None, "Feb. 2020 to present", "3 years 11 months"),
|
2024-02-02 17:02:18 +00:00
|
|
|
("2020-01-01", "2021-01-01", "2023-02-01", "Feb. 2023", ""),
|
2024-01-28 20:14:25 +00:00
|
|
|
("2020", "2021", None, "2020 to 2021", "1 year"),
|
|
|
|
("2020", None, None, "2020 to present", "4 years"),
|
|
|
|
("2020-10-10", "2022", None, "Oct. 2020 to 2022", "2 years"),
|
|
|
|
("2022", "2023-10-10", None, "2022 to Oct. 2023", "1 year"),
|
2024-02-02 17:02:18 +00:00
|
|
|
("2020-01-01", "present", "My Custom Date", "My Custom Date", ""),
|
|
|
|
("2020-01-01", None, "My Custom Date", "My Custom Date", ""),
|
|
|
|
(None, None, "My Custom Date", "My Custom Date", ""),
|
|
|
|
(None, "2020-01-01", "My Custom Date", "My Custom Date", ""),
|
|
|
|
(None, None, "2020-01-01", "Jan. 2020", ""),
|
|
|
|
(None, None, None, "", ""),
|
2024-01-28 20:14:25 +00:00
|
|
|
],
|
|
|
|
)
|
|
|
|
@time_machine.travel("2024-01-01")
|
|
|
|
def test_dates(start_date, end_date, date, expected_date_string, expected_time_span):
|
|
|
|
entry_base = dm.EntryBase(start_date=start_date, end_date=end_date, date=date)
|
|
|
|
|
|
|
|
assert entry_base.date_string == expected_date_string
|
2024-01-30 18:50:57 +00:00
|
|
|
assert entry_base.time_span_string == expected_time_span
|
2024-01-28 20:14:25 +00:00
|
|
|
|
2024-02-02 17:02:18 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"date, expected_date_string",
|
|
|
|
[
|
|
|
|
("2020-01-01", "Jan. 2020"),
|
|
|
|
("2020-01", "Jan. 2020"),
|
|
|
|
("2020", "2020"),
|
|
|
|
]
|
|
|
|
)
|
|
|
|
def test_publication_dates(publication_entry, date, expected_date_string):
|
|
|
|
publication_entry["date"] = date
|
|
|
|
publication_entry = dm.PublicationEntry(**publication_entry)
|
|
|
|
assert publication_entry.date_string == expected_date_string
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"date",
|
|
|
|
[
|
|
|
|
"aaa",
|
|
|
|
None,
|
|
|
|
"2025"
|
|
|
|
]
|
|
|
|
)
|
|
|
|
def test_invalid_publication_dates(publication_entry, date):
|
|
|
|
with pytest.raises(pydantic.ValidationError):
|
|
|
|
publication_entry["date"] = date
|
|
|
|
dm.PublicationEntry(**publication_entry)
|
|
|
|
|
2024-01-28 20:14:25 +00:00
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"start_date, end_date, date",
|
|
|
|
[
|
|
|
|
("aaa", "2021-01-01", None),
|
|
|
|
("2020-01-01", "aaa", None),
|
|
|
|
(None, "2020-01-01", None),
|
|
|
|
("2023-01-01", "2021-01-01", None),
|
|
|
|
("2999-01-01", None, None),
|
|
|
|
("2020-01-01", "2999-01-01", None),
|
|
|
|
("2022", "2021", None),
|
|
|
|
("2021", "2060", None),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_invalid_dates(start_date, end_date, date):
|
|
|
|
with pytest.raises(pydantic.ValidationError):
|
|
|
|
dm.EntryBase(start_date=start_date, end_date=end_date, date=date)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"url, url_text, expected_url_text",
|
|
|
|
[
|
|
|
|
("https://linkedin.com", None, "view on LinkedIn"),
|
|
|
|
("https://github.com", None, "view on GitHub"),
|
|
|
|
("https://instagram.com", None, "view on Instagram"),
|
|
|
|
("https://youtube.com", None, "view on YouTube"),
|
|
|
|
("https://twitter.com", "My URL Text", "My URL Text"),
|
|
|
|
("https://google.com", None, "view on my website"),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_url_text(url, url_text, expected_url_text):
|
|
|
|
entry_base = dm.EntryBase(url=url, url_text=url_text)
|
|
|
|
assert entry_base.url_text == expected_url_text
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"doi, expected_doi_url",
|
|
|
|
[
|
|
|
|
("10.1109/TASC.2023.3340648", "https://doi.org/10.1109/TASC.2023.3340648"),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_doi_url(publication_entry, doi, expected_doi_url):
|
|
|
|
publication_entry["doi"] = doi
|
|
|
|
publication_entry = dm.PublicationEntry(**publication_entry)
|
|
|
|
assert publication_entry.doi_url == expected_doi_url
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"doi",
|
|
|
|
["aaa10.1109/TASC.2023.3340648", "aaa"],
|
|
|
|
)
|
|
|
|
def test_invalid_doi(publication_entry, doi):
|
|
|
|
with pytest.raises(pydantic.ValidationError):
|
|
|
|
publication_entry["doi"] = doi
|
|
|
|
dm.PublicationEntry(**publication_entry)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"network, username, expected_url",
|
|
|
|
[
|
|
|
|
("LinkedIn", "myusername", "https://linkedin.com/in/myusername"),
|
|
|
|
("GitHub", "myusername", "https://github.com/myusername"),
|
|
|
|
("Instagram", "myusername", "https://instagram.com/myusername"),
|
|
|
|
("Orcid", "myusername", "https://orcid.org/myusername"),
|
2024-01-28 20:37:00 +00:00
|
|
|
("Twitter", "myusername", "https://twitter.com/myusername"),
|
|
|
|
("Mastodon", "@myusername", "https://mastodon.social/@myusername"),
|
2024-01-28 20:14:25 +00:00
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_social_network_url(network, username, expected_url):
|
|
|
|
social_network = dm.SocialNetwork(network=network, username=username)
|
|
|
|
assert str(social_network.url) == expected_url
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"title, default_entry",
|
|
|
|
[
|
|
|
|
("Education", "education_entry"),
|
|
|
|
("Experience", "experience_entry"),
|
|
|
|
("Work Experience", "experience_entry"),
|
|
|
|
("Research Experience", "experience_entry"),
|
|
|
|
("Publications", "publication_entry"),
|
|
|
|
("Papers", "publication_entry"),
|
|
|
|
("Projects", "normal_entry"),
|
|
|
|
("Academic Projects", "normal_entry"),
|
|
|
|
("University Projects", "normal_entry"),
|
|
|
|
("Personal Projects", "normal_entry"),
|
|
|
|
("Certificates", "normal_entry"),
|
|
|
|
("Extracurricular Activities", "experience_entry"),
|
|
|
|
("Test Scores", "one_line_entry"),
|
|
|
|
("Skills", "one_line_entry"),
|
|
|
|
("programming_skills", "normal_entry"),
|
|
|
|
("other_skills", "one_line_entry"),
|
|
|
|
("Awards", "one_line_entry"),
|
|
|
|
("Interests", "one_line_entry"),
|
|
|
|
("Summary", "text_entry"),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_sections_with_default_types(
|
|
|
|
education_entry,
|
|
|
|
experience_entry,
|
|
|
|
publication_entry,
|
|
|
|
normal_entry,
|
|
|
|
one_line_entry,
|
|
|
|
text_entry,
|
|
|
|
title,
|
|
|
|
default_entry,
|
|
|
|
):
|
|
|
|
input = {
|
|
|
|
"name": "John Doe",
|
|
|
|
"sections": {
|
|
|
|
title: [
|
|
|
|
eval(default_entry),
|
|
|
|
eval(default_entry),
|
|
|
|
],
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
cv = dm.CurriculumVitae(**input)
|
|
|
|
assert len(cv.sections) == 1
|
|
|
|
assert len(cv.sections[0].entries) == 2
|
|
|
|
|
|
|
|
# test with other entry types:
|
|
|
|
entries = [
|
|
|
|
(education_entry, "EducationEntry"),
|
|
|
|
(experience_entry, "ExperienceEntry"),
|
|
|
|
(publication_entry, "PublicationEntry"),
|
|
|
|
(normal_entry, "NormalEntry"),
|
|
|
|
(one_line_entry, "OneLineEntry"),
|
|
|
|
(text_entry, "TextEntry"),
|
|
|
|
]
|
|
|
|
for entry, entry_type in entries:
|
|
|
|
input["sections"][title] = {
|
|
|
|
"entry_type": entry_type,
|
|
|
|
"entries": [entry, entry],
|
|
|
|
}
|
|
|
|
cv = dm.CurriculumVitae(**input)
|
|
|
|
assert len(cv.sections) == 1
|
|
|
|
assert len(cv.sections[0].entries) == 2
|
|
|
|
|
|
|
|
|
|
|
|
def test_sections_without_default_types(
|
|
|
|
education_entry,
|
|
|
|
experience_entry,
|
|
|
|
publication_entry,
|
|
|
|
normal_entry,
|
|
|
|
one_line_entry,
|
|
|
|
text_entry,
|
|
|
|
):
|
|
|
|
input = {"name": "John Doe", "sections": dict()}
|
|
|
|
entries = [
|
|
|
|
(education_entry, "EducationEntry"),
|
|
|
|
(experience_entry, "ExperienceEntry"),
|
|
|
|
(publication_entry, "PublicationEntry"),
|
|
|
|
(normal_entry, "NormalEntry"),
|
|
|
|
(one_line_entry, "OneLineEntry"),
|
|
|
|
(text_entry, "TextEntry"),
|
|
|
|
]
|
|
|
|
for i, (entry, entry_type) in enumerate(entries):
|
|
|
|
input["sections"][f"My Section {i}"] = {
|
|
|
|
"entry_type": entry_type,
|
|
|
|
"entries": [entry, entry],
|
|
|
|
}
|
|
|
|
|
|
|
|
cv = dm.CurriculumVitae(**input)
|
|
|
|
assert len(cv.sections) == len(entries)
|
|
|
|
for i, entry in enumerate(entries):
|
|
|
|
assert len(cv.sections[i].entries) == 2
|
2024-01-31 17:26:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_section_with_invalid_entry_type():
|
|
|
|
input = {"name": "John Doe", "sections": dict()}
|
|
|
|
input["sections"]["My Section"] = {
|
|
|
|
"entry_type": "InvalidEntryType",
|
|
|
|
"entries": [],
|
|
|
|
}
|
|
|
|
with pytest.raises(pydantic.ValidationError):
|
|
|
|
dm.CurriculumVitae(**input)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"section_title",
|
|
|
|
[
|
|
|
|
"Education",
|
|
|
|
"Experience",
|
|
|
|
"Work Experience",
|
|
|
|
"Research Experience",
|
|
|
|
"Publications",
|
|
|
|
"Papers",
|
|
|
|
"Projects",
|
|
|
|
"Academic Projects",
|
|
|
|
"University Projects",
|
|
|
|
"Personal Projects",
|
|
|
|
"Certificates",
|
|
|
|
"Extracurricular Activities",
|
|
|
|
"Test Scores",
|
|
|
|
"Skills",
|
|
|
|
"Programming Skills",
|
|
|
|
"Other Skills",
|
|
|
|
"Awards",
|
|
|
|
"Interests",
|
|
|
|
"Summary",
|
|
|
|
"My Custom Section",
|
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_sections_with_invalid_entries(section_title):
|
|
|
|
input = {"name": "John Doe", "sections": dict()}
|
|
|
|
input["sections"][section_title] = [
|
|
|
|
{
|
|
|
|
"this": "is",
|
|
|
|
"an": "invalid",
|
|
|
|
"entry": 10,
|
|
|
|
}
|
|
|
|
]
|
|
|
|
with pytest.raises(pydantic.ValidationError):
|
|
|
|
dm.CurriculumVitae(**input)
|