mirror of https://github.com/eyhc1/rendercv.git
improve PastDate type (#4)
This commit is contained in:
parent
3070092a2b
commit
2698e8074b
|
@ -44,7 +44,7 @@ def user_friendly_errors(func: Callable) -> Callable:
|
||||||
# Translate Pydantic's error messages to make them more user-friendly
|
# Translate Pydantic's error messages to make them more user-friendly
|
||||||
custom_error_messages_by_type = {
|
custom_error_messages_by_type = {
|
||||||
"url_scheme": "This is not a valid URL 😿",
|
"url_scheme": "This is not a valid URL 😿",
|
||||||
"string_type": "This is not a valid string 🤭",
|
# "string_type": "This is not a valid string 🤭",
|
||||||
"missing": "This field is required, but it is missing 😆",
|
"missing": "This field is required, but it is missing 😆",
|
||||||
"literal_error": "Only the following values are allowed: {expected} 😒",
|
"literal_error": "Only the following values are allowed: {expected} 😒",
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,6 @@ from pydantic import (
|
||||||
model_validator,
|
model_validator,
|
||||||
computed_field,
|
computed_field,
|
||||||
EmailStr,
|
EmailStr,
|
||||||
PastDate,
|
|
||||||
)
|
)
|
||||||
from pydantic.json_schema import GenerateJsonSchema
|
from pydantic.json_schema import GenerateJsonSchema
|
||||||
from pydantic.functional_validators import AfterValidator
|
from pydantic.functional_validators import AfterValidator
|
||||||
|
@ -155,7 +154,43 @@ def escape_latex_characters(sentence: str) -> str:
|
||||||
return sentence
|
return sentence
|
||||||
|
|
||||||
|
|
||||||
def compute_time_span_string(start_date: Date, end_date: Date) -> str:
|
def parse_date_string(date_string: str) -> Date | int:
|
||||||
|
"""Parse a date string in YYYY-MM-DD, YYYY-MM, or YYYY format and return a
|
||||||
|
datetime.date object.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
date_string (str): The date string to parse.
|
||||||
|
Returns:
|
||||||
|
datetime.date: The parsed date.
|
||||||
|
"""
|
||||||
|
if re.match(r"\d{4}-\d{2}-\d{2}", date_string):
|
||||||
|
# Then it is in YYYY-MM-DD format
|
||||||
|
date = Date.fromisoformat(date_string)
|
||||||
|
elif re.match(r"\d{4}-\d{2}", date_string):
|
||||||
|
# Then it is in YYYY-MM format
|
||||||
|
# Assign a random day since days are not rendered in the CV
|
||||||
|
date = Date.fromisoformat(f"{date_string}-01")
|
||||||
|
elif re.match(r"\d{4}", date_string):
|
||||||
|
# Then it is in YYYY format
|
||||||
|
# Then keep it as an integer
|
||||||
|
date = int(date_string)
|
||||||
|
else:
|
||||||
|
raise ValueError(
|
||||||
|
f'The date string "{date_string}" is not in YYYY-MM-DD, YYYY-MM, or YYYY'
|
||||||
|
" format 🥶"
|
||||||
|
)
|
||||||
|
|
||||||
|
if isinstance(date, Date):
|
||||||
|
# Then it means the date is a Date object, so check if it is a past date:
|
||||||
|
if date > Date.today():
|
||||||
|
raise ValueError(
|
||||||
|
f'The date "{date_string}" is in the future. Please check the dates 🤯'
|
||||||
|
)
|
||||||
|
|
||||||
|
return date
|
||||||
|
|
||||||
|
|
||||||
|
def compute_time_span_string(start_date: Date | int, end_date: Date | int) -> str:
|
||||||
"""Compute the time span between two dates and return a string that represents it.
|
"""Compute the time span between two dates and return a string that represents it.
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
@ -168,26 +203,35 @@ def compute_time_span_string(start_date: Date, end_date: Date) -> str:
|
||||||
`#!python "2 years 5 months"`
|
`#!python "2 years 5 months"`
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
start_date (Date): The start date.
|
start_date (Date | int): The start date.
|
||||||
end_date (Date): The end date.
|
end_date (Date | int): The end date.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
str: The time span string.
|
str: The time span string.
|
||||||
"""
|
"""
|
||||||
# check if the types of start_date and end_date are correct:
|
# check if the types of start_date and end_date are correct:
|
||||||
if not isinstance(start_date, Date):
|
if not isinstance(start_date, (Date, int)):
|
||||||
raise TypeError("start_date is not a Date object!")
|
raise TypeError("start_date is not a Date object or an integer!")
|
||||||
if not isinstance(end_date, Date):
|
if not isinstance(end_date, (Date, int)):
|
||||||
raise TypeError("end_date is not a Date object!")
|
raise TypeError("end_date is not a Date object or an integer!")
|
||||||
|
|
||||||
# # check if start_date is before end_date:
|
|
||||||
if start_date > end_date:
|
|
||||||
raise ValueError(
|
|
||||||
"The start date is after the end date. Please check the dates!"
|
|
||||||
)
|
|
||||||
|
|
||||||
# calculate the number of days between start_date and end_date:
|
# calculate the number of days between start_date and end_date:
|
||||||
timespan_in_days = (end_date - start_date).days
|
if isinstance(start_date, Date) and isinstance(end_date, Date):
|
||||||
|
timespan_in_days = (end_date - start_date).days
|
||||||
|
elif isinstance(start_date, int) and isinstance(end_date, int):
|
||||||
|
timespan_in_days = (end_date - start_date) * 365
|
||||||
|
elif isinstance(start_date, int) and isinstance(end_date, Date):
|
||||||
|
timespan_in_days = (end_date - Date(start_date, 1, 1)).days
|
||||||
|
else:
|
||||||
|
raise TypeError(
|
||||||
|
f"start_date's type is {type(start_date)} and end_date's type is"
|
||||||
|
f" {type(end_date)}. This is not supported."
|
||||||
|
)
|
||||||
|
|
||||||
|
if timespan_in_days < 0:
|
||||||
|
raise ValueError(
|
||||||
|
f'"start_date" can not be after "end_date". Please check the dates 👻'
|
||||||
|
)
|
||||||
|
|
||||||
# calculate the number of years between start_date and end_date:
|
# calculate the number of years between start_date and end_date:
|
||||||
how_many_years = timespan_in_days // 365
|
how_many_years = timespan_in_days // 365
|
||||||
|
@ -234,8 +278,12 @@ def format_date(date: Date) -> str:
|
||||||
Returns:
|
Returns:
|
||||||
str: The formatted date.
|
str: The formatted date.
|
||||||
"""
|
"""
|
||||||
if not isinstance(date, Date):
|
if not isinstance(date, (Date, int)):
|
||||||
raise TypeError("date is not a Date object!")
|
raise TypeError("date is not a Date object or an integer!")
|
||||||
|
|
||||||
|
if isinstance(date, int):
|
||||||
|
# Then it means the user only provided the year, so just return the year
|
||||||
|
return str(date)
|
||||||
|
|
||||||
# Month abbreviations,
|
# Month abbreviations,
|
||||||
# taken from: https://web.library.yale.edu/cataloging/months
|
# taken from: https://web.library.yale.edu/cataloging/months
|
||||||
|
@ -336,6 +384,11 @@ LaTeXDimension = Annotated[
|
||||||
]
|
]
|
||||||
LaTeXString = Annotated[str, AfterValidator(escape_latex_characters)]
|
LaTeXString = Annotated[str, AfterValidator(escape_latex_characters)]
|
||||||
SpellCheckedString = Annotated[LaTeXString, AfterValidator(check_spelling)]
|
SpellCheckedString = Annotated[LaTeXString, AfterValidator(check_spelling)]
|
||||||
|
PastDate = Annotated[
|
||||||
|
str,
|
||||||
|
Field(pattern=r"\d{4}-?(\d{2})?-?(\d{2})?"),
|
||||||
|
AfterValidator(parse_date_string),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
class ClassicThemePageMargins(BaseModel):
|
class ClassicThemePageMargins(BaseModel):
|
||||||
|
@ -612,7 +665,7 @@ class Event(BaseModel):
|
||||||
description="The start date of the event in YYYY-MM-DD format.",
|
description="The start date of the event in YYYY-MM-DD format.",
|
||||||
examples=["2020-09-24"],
|
examples=["2020-09-24"],
|
||||||
)
|
)
|
||||||
end_date: Optional[PastDate | Literal["present"]] = Field(
|
end_date: Optional[Literal["present"] | PastDate] = Field(
|
||||||
default=None,
|
default=None,
|
||||||
title="End Date",
|
title="End Date",
|
||||||
description=(
|
description=(
|
||||||
|
@ -621,7 +674,7 @@ class Event(BaseModel):
|
||||||
),
|
),
|
||||||
examples=["2020-09-24", "present"],
|
examples=["2020-09-24", "present"],
|
||||||
)
|
)
|
||||||
date: Optional[LaTeXString | PastDate] = Field(
|
date: Optional[PastDate | LaTeXString] = Field(
|
||||||
default=None,
|
default=None,
|
||||||
title="Date",
|
title="Date",
|
||||||
description=(
|
description=(
|
||||||
|
@ -653,13 +706,13 @@ class Event(BaseModel):
|
||||||
|
|
||||||
@field_validator("date")
|
@field_validator("date")
|
||||||
@classmethod
|
@classmethod
|
||||||
def check_date(cls, date: LaTeXString | PastDate) -> LaTeXString | PastDate:
|
def check_date(cls, date: PastDate | LaTeXString) -> PastDate | LaTeXString:
|
||||||
"""Check if the date is a string or a Date object and return accordingly."""
|
"""Check if the date is a string or a Date object and return accordingly."""
|
||||||
if isinstance(date, str):
|
if isinstance(date, str):
|
||||||
try:
|
try:
|
||||||
# If this runs, it means the date is an ISO format string, and it can be
|
# If this runs, it means the date is an ISO format string, and it can be
|
||||||
# parsed
|
# parsed
|
||||||
date = Date.fromisoformat(date)
|
date = parse_date_string(date)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
# Then it means it is a custom string like "Fall 2023"
|
# Then it means it is a custom string like "Fall 2023"
|
||||||
date = date
|
date = date
|
||||||
|
@ -718,7 +771,19 @@ class Event(BaseModel):
|
||||||
else:
|
else:
|
||||||
end_date = model.end_date
|
end_date = model.end_date
|
||||||
|
|
||||||
if model.start_date > end_date:
|
if isinstance(model.start_date, int):
|
||||||
|
# Then it means user only provided the year, so convert it to a Date
|
||||||
|
# object with the first day of the year
|
||||||
|
start_date = Date(model.start_date, 1, 1)
|
||||||
|
elif isinstance(model.start_date, Date):
|
||||||
|
# Then it means user provided either YYYY-MM-DD or YYYY-MM
|
||||||
|
start_date = model.start_date
|
||||||
|
else:
|
||||||
|
raise RuntimeError(
|
||||||
|
"start_date is neither an integer nor a Date object 🤯"
|
||||||
|
)
|
||||||
|
|
||||||
|
if start_date > end_date:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
'"start_date" can not be after "end_date". Please check the dates 👻'
|
'"start_date" can not be after "end_date". Please check the dates 👻'
|
||||||
)
|
)
|
||||||
|
|
48
schema.json
48
schema.json
|
@ -640,7 +640,7 @@
|
||||||
"title": "Start Date",
|
"title": "Start Date",
|
||||||
"allOf": [
|
"allOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -648,11 +648,11 @@
|
||||||
"end_date": {
|
"end_date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"const": "present"
|
||||||
"type": "string"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"const": "present"
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "null"
|
"type": "null"
|
||||||
|
@ -669,10 +669,10 @@
|
||||||
"date": {
|
"date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"format": "date",
|
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -810,7 +810,7 @@
|
||||||
"title": "Start Date",
|
"title": "Start Date",
|
||||||
"allOf": [
|
"allOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -818,11 +818,11 @@
|
||||||
"end_date": {
|
"end_date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"const": "present"
|
||||||
"type": "string"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"const": "present"
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "null"
|
"type": "null"
|
||||||
|
@ -839,10 +839,10 @@
|
||||||
"date": {
|
"date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"format": "date",
|
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -929,7 +929,7 @@
|
||||||
"title": "Start Date",
|
"title": "Start Date",
|
||||||
"allOf": [
|
"allOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -937,11 +937,11 @@
|
||||||
"end_date": {
|
"end_date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"const": "present"
|
||||||
"type": "string"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"const": "present"
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "null"
|
"type": "null"
|
||||||
|
@ -958,10 +958,10 @@
|
||||||
"date": {
|
"date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"format": "date",
|
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -1042,7 +1042,7 @@
|
||||||
"title": "Start Date",
|
"title": "Start Date",
|
||||||
"allOf": [
|
"allOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -1050,11 +1050,11 @@
|
||||||
"end_date": {
|
"end_date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"const": "present"
|
||||||
"type": "string"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"const": "present"
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "null"
|
"type": "null"
|
||||||
|
@ -1071,10 +1071,10 @@
|
||||||
"date": {
|
"date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"format": "date",
|
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -1161,7 +1161,7 @@
|
||||||
"title": "Start Date",
|
"title": "Start Date",
|
||||||
"allOf": [
|
"allOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -1169,11 +1169,11 @@
|
||||||
"end_date": {
|
"end_date": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
"format": "date",
|
"const": "present"
|
||||||
"type": "string"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"const": "present"
|
"pattern": "\\d{4}-?(\\d{2})?-?(\\d{2})?",
|
||||||
|
"type": "string"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "null"
|
"type": "null"
|
||||||
|
|
Loading…
Reference in New Issue