-
Notifications
You must be signed in to change notification settings - Fork 159
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #348 from yozachar/workshop
feat: adds basic `cron` validator
- Loading branch information
Showing
8 changed files
with
167 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
# cron | ||
|
||
::: validators.cron.cron |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
cron | ||
---- | ||
|
||
.. module:: validators.cron | ||
.. autofunction:: cron |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,78 @@ | ||
"""Cron.""" | ||
|
||
# local | ||
from .utils import validator | ||
|
||
|
||
def _validate_cron_component(component: str, min_val: int, max_val: int): | ||
if component == "*": | ||
return True | ||
|
||
if component.isdecimal(): | ||
return min_val <= int(component) <= max_val | ||
|
||
if "/" in component: | ||
parts = component.split("/") | ||
if len(parts) != 2 or not parts[1].isdecimal() or int(parts[1]) < 1: | ||
return False | ||
if parts[0] == "*": | ||
return True | ||
return parts[0].isdecimal() and min_val <= int(parts[0]) <= max_val | ||
|
||
if "-" in component: | ||
parts = component.split("-") | ||
if len(parts) != 2 or not parts[0].isdecimal() or not parts[1].isdecimal(): | ||
return False | ||
start, end = int(parts[0]), int(parts[1]) | ||
return min_val <= start <= max_val and min_val <= end <= max_val and start <= end | ||
|
||
if "," in component: | ||
for item in component.split(","): | ||
if not _validate_cron_component(item, min_val, max_val): | ||
return False | ||
return True | ||
# return all( | ||
# _validate_cron_component(item, min_val, max_val) for item in component.split(",") | ||
# ) # throws type error. why? | ||
|
||
return False | ||
|
||
|
||
@validator | ||
def cron(value: str, /): | ||
"""Return whether or not given value is a valid cron string. | ||
Examples: | ||
>>> cron('*/5 * * * *') | ||
# Output: True | ||
>>> cron('30-20 * * * *') | ||
# Output: ValidationError(func=cron, ...) | ||
Args: | ||
value: | ||
Cron string to validate. | ||
Returns: | ||
(Literal[True]): If `value` is a valid cron string. | ||
(ValidationError): If `value` is an invalid cron string. | ||
""" | ||
if not value: | ||
return False | ||
|
||
try: | ||
minutes, hours, days, months, weekdays = value.strip().split() | ||
except ValueError as err: | ||
raise ValueError("Badly formatted cron string") from err | ||
|
||
if not _validate_cron_component(minutes, 0, 59): | ||
return False | ||
if not _validate_cron_component(hours, 0, 23): | ||
return False | ||
if not _validate_cron_component(days, 1, 31): | ||
return False | ||
if not _validate_cron_component(months, 1, 12): | ||
return False | ||
if not _validate_cron_component(weekdays, 0, 6): | ||
return False | ||
|
||
return True |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,54 @@ | ||
"""Test Cron.""" | ||
|
||
# external | ||
import pytest | ||
|
||
# local | ||
from validators import ValidationError, cron | ||
|
||
|
||
@pytest.mark.parametrize( | ||
"value", | ||
[ | ||
"* * * * *", | ||
"*/5 * * * *", | ||
"0 0 * * *", | ||
"30 3 * * 1-5", | ||
"15 5 * * 1,3,5", | ||
"0 12 1 */2 *", | ||
"0 */3 * * *", | ||
"0 0 1 1 *", | ||
"0 12 * 1-6 1-5", | ||
"0 3-6 * * *", | ||
"*/15 0,6,12,18 * * *", | ||
"0 12 * * 0", | ||
"*/61 * * * *", | ||
# "5-10/2 * * * *", # this is valid, but not supported yet | ||
], | ||
) | ||
def test_returns_true_on_valid_cron(value: str): | ||
"""Test returns true on valid cron string.""" | ||
assert cron(value) | ||
|
||
|
||
@pytest.mark.parametrize( | ||
"value", | ||
[ | ||
"* * * * * *", | ||
"* * * *", | ||
"*/5 25 * * *", | ||
"*/5 * *-1 * *", | ||
"32-30 * * * *", | ||
"0 12 32 * *", | ||
"0 12 * * 8", | ||
"0 */0 * * *", | ||
"30-20 * * * *", | ||
"10-* * * * *", | ||
"*/15 0,6,12,24 * * *", | ||
"& * * & * *", | ||
"* - * * - *", | ||
], | ||
) | ||
def test_returns_failed_validation_on_invalid_cron(value: str): | ||
"""Test returns failed validation on invalid cron string.""" | ||
assert isinstance(cron(value), ValidationError) |