mirror of
https://github.com/tembo-pages/tembo-core.git
synced 2025-12-22 05:35:43 +00:00
479 lines
17 KiB
Python
479 lines
17 KiB
Python
"""Submodule containing the factories & page objects to create Tembo pages."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import pathlib
|
|
import re
|
|
from abc import ABCMeta, abstractmethod
|
|
from dataclasses import dataclass
|
|
from typing import Collection, Optional
|
|
|
|
import jinja2
|
|
import pendulum
|
|
from jinja2.exceptions import TemplateNotFound
|
|
|
|
import tembo.utils
|
|
from tembo import exceptions
|
|
from tembo.journal.abstract import Page, PageCreator
|
|
|
|
|
|
@dataclass
|
|
class PageCreatorOptions:
|
|
"""
|
|
Options [dataclass][dataclasses.dataclass] to create a Page.
|
|
|
|
This is passed to an implemented instance of [PageCreator][tembo.journal.pages.PageCreator]
|
|
|
|
Attributes:
|
|
base_path (str): The base path.
|
|
page_path (str): The path of the page relative to the base path.
|
|
filename (str): The filename of the page.
|
|
extension (str): The extension of the page.
|
|
name (str): The name of the scope.
|
|
user_input (Collection[str] | None, optional): User input tokens.
|
|
example (str | None, optional): User example command.
|
|
template_path (str | None, optional): The path which contains the templates. This should
|
|
be the full path and not relative to the base path.
|
|
template_filename (str | None, optional): The template filename with extension relative
|
|
to the template path.
|
|
"""
|
|
|
|
base_path: str
|
|
page_path: str
|
|
filename: str
|
|
extension: str
|
|
name: str
|
|
user_input: Optional[Collection[str]] = None
|
|
example: Optional[str] = None
|
|
template_path: Optional[str] = None
|
|
template_filename: Optional[str] = None
|
|
|
|
|
|
class PageCreator:
|
|
"""
|
|
A PageCreator factory base class.
|
|
|
|
This factory should implement methods to create [Page][tembo.journal.pages.Page] objects.
|
|
|
|
!!! abstract
|
|
This factory is an abstract base class and should be implemented for each
|
|
[Page][tembo.journal.pages.Page] type.
|
|
|
|
The methods
|
|
|
|
- `_check_base_path_exists()`
|
|
- `_convert_base_path_to_path()`
|
|
- `_load_template()`
|
|
|
|
are not abstract and are shared between all [Page][tembo.journal.pages.Page] types.
|
|
"""
|
|
|
|
@abstractmethod
|
|
def __init__(self, options: PageCreatorOptions) -> None:
|
|
"""
|
|
When implemented this should initialise the `PageCreator` factory.
|
|
|
|
Args:
|
|
options (PageCreatorOptions): An instance of
|
|
[PageCreatorOptions][tembo.journal.pages.PageCreatorOptions]
|
|
|
|
!!! abstract
|
|
This method is abstract and should be implemented for each
|
|
[Page][tembo.journal.pages.Page] type.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
@property
|
|
@abstractmethod
|
|
def options(self) -> PageCreatorOptions:
|
|
"""
|
|
When implemented this should return the `PageCreatorOptions` on the class.
|
|
|
|
Returns:
|
|
PageCreatorOptions: the instance of
|
|
[PageCreatorOptions][tembo.journal.pages.PageCreatorOptions] set on the class.
|
|
|
|
!!! abstract
|
|
This method is abstract and should be implemented for each
|
|
[Page][tembo.journal.pages.Page] type.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
@abstractmethod
|
|
def create_page(self) -> Page:
|
|
"""
|
|
When implemented this should create a `Page` object.
|
|
|
|
Returns:
|
|
Page: an implemented instance of [Page][tembo.journal.pages.Page] such as
|
|
[ScopedPage][tembo.journal.pages.ScopedPage].
|
|
|
|
!!! abstract
|
|
This method is abstract and should be implemented for each
|
|
[Page][tembo.journal.pages.Page] type.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
def _check_base_path_exists(self) -> None:
|
|
"""
|
|
Check that the base path exists.
|
|
|
|
Raises:
|
|
exceptions.BasePathDoesNotExistError: raised if the base path does not exist.
|
|
"""
|
|
if not pathlib.Path(self.options.base_path).expanduser().exists():
|
|
raise exceptions.BasePathDoesNotExistError(
|
|
f"Tembo base path of {self.options.base_path} does not exist."
|
|
)
|
|
|
|
def _convert_base_path_to_path(self) -> pathlib.Path:
|
|
"""
|
|
Convert the `base_path` from a `str` to a `pathlib.Path` object.
|
|
|
|
Returns:
|
|
pathlib.Path: the `base_path` as a `pathlib.Path` object.
|
|
"""
|
|
path_to_file = (
|
|
pathlib.Path(self.options.base_path).expanduser()
|
|
/ pathlib.Path(self.options.page_path.replace(" ", "_")).expanduser()
|
|
/ self.options.filename.replace(" ", "_")
|
|
)
|
|
# check for existing `.` in the extension
|
|
extension = (
|
|
self.options.extension[1:]
|
|
if self.options.extension[0] == "."
|
|
else self.options.extension
|
|
)
|
|
# return path with a file
|
|
return path_to_file.with_suffix(f".{extension}")
|
|
|
|
def _load_template(self) -> str:
|
|
"""
|
|
Load the template file.
|
|
|
|
Raises:
|
|
exceptions.TemplateFileNotFoundError: raised if the template file is specified but
|
|
not found.
|
|
|
|
Returns:
|
|
str: the contents of the template file.
|
|
"""
|
|
if self.options.template_filename is None:
|
|
return ""
|
|
if self.options.template_path is not None:
|
|
converted_template_path = pathlib.Path(self.options.template_path).expanduser()
|
|
else:
|
|
converted_template_path = (
|
|
pathlib.Path(self.options.base_path).expanduser() / ".templates"
|
|
)
|
|
|
|
file_loader = jinja2.FileSystemLoader(converted_template_path)
|
|
env = jinja2.Environment(loader=file_loader, autoescape=True)
|
|
|
|
try:
|
|
loaded_template = env.get_template(self.options.template_filename)
|
|
except TemplateNotFound as template_not_found:
|
|
_template_file = f"{converted_template_path}/{template_not_found.args[0]}"
|
|
raise exceptions.TemplateFileNotFoundError(
|
|
f"Template file {_template_file} does not exist."
|
|
) from template_not_found
|
|
return loaded_template.render()
|
|
|
|
|
|
class ScopedPageCreator(PageCreator):
|
|
"""
|
|
Factory to create a scoped page.
|
|
|
|
Attributes:
|
|
base_path (str): base path of tembo.
|
|
page_path (str): path of the page relative to the base path.
|
|
filename (str): filename relative to the page path.
|
|
extension (str): extension of file.
|
|
"""
|
|
|
|
def __init__(self, options: PageCreatorOptions) -> None:
|
|
"""
|
|
Initialise a `ScopedPageCreator` factory.
|
|
|
|
Args:
|
|
options (PageCreatorOptions): An instance of
|
|
[PageCreatorOptions][tembo.journal.pages.PageCreatorOptions].
|
|
"""
|
|
self._all_input_tokens: list[str] = []
|
|
self._options = options
|
|
|
|
@property
|
|
def options(self) -> PageCreatorOptions:
|
|
"""
|
|
Return the `PageCreatorOptions` instance set on the factory.
|
|
|
|
Returns:
|
|
PageCreatorOptions:
|
|
An instance of [PageCreatorOptions][tembo.journal.pages.PageCreatorOptions].
|
|
"""
|
|
return self._options
|
|
|
|
def create_page(self) -> Page:
|
|
"""
|
|
Create a [ScopedPage][tembo.journal.pages.ScopedPage] object.
|
|
|
|
This method will
|
|
|
|
- Check the `base_path` exists
|
|
- Verify the input tokens match the number defined in the `config.yml`
|
|
- Substitue the input tokens in the filepath
|
|
- Load the template contents and substitue the input tokens
|
|
|
|
Raises:
|
|
exceptions.MismatchedTokenError: Raises
|
|
[MismatchedTokenError][tembo.exceptions.MismatchedTokenError] if the number of
|
|
input tokens does not match the number of unique input tokens defined.
|
|
exceptions.BasePathDoesNotExistError: Raises
|
|
[BasePathDoesNotExistError][tembo.exceptions.BasePathDoesNotExistError] if the
|
|
base path does not exist.
|
|
exceptions.TemplateFileNotFoundError: Raises
|
|
[TemplateFileNotFoundError][tembo.exceptions.TemplateFileNotFoundError] if the
|
|
template file is specified but not found.
|
|
|
|
|
|
Returns:
|
|
Page: A [ScopedPage][tembo.journal.pages.ScopedPage] object using the
|
|
`PageCreatorOptions`.
|
|
"""
|
|
try:
|
|
self._check_base_path_exists()
|
|
except exceptions.BasePathDoesNotExistError as base_path_does_not_exist_error:
|
|
raise base_path_does_not_exist_error
|
|
self._all_input_tokens = self._get_input_tokens()
|
|
try:
|
|
self._verify_input_tokens()
|
|
except exceptions.MismatchedTokenError as mismatched_token_error:
|
|
raise mismatched_token_error
|
|
|
|
path = self._convert_base_path_to_path()
|
|
path = pathlib.Path(self._substitute_tokens(str(path)))
|
|
|
|
try:
|
|
template_contents = self._load_template()
|
|
except exceptions.TemplateFileNotFoundError as template_not_found_error:
|
|
raise template_not_found_error
|
|
if self.options.template_filename is not None:
|
|
template_contents = self._substitute_tokens(template_contents)
|
|
|
|
return ScopedPage(path, template_contents)
|
|
|
|
def _get_input_tokens(self) -> list[str]:
|
|
"""Get the input tokens from the path & user template."""
|
|
path = str(
|
|
pathlib.Path(
|
|
self.options.base_path,
|
|
self.options.page_path,
|
|
self.options.filename,
|
|
)
|
|
.expanduser()
|
|
.with_suffix(f".{self.options.extension}")
|
|
)
|
|
template_contents = self._load_template()
|
|
# get the input tokens from both the path and the template
|
|
all_input_tokens = []
|
|
for tokenified_string in (path, template_contents):
|
|
all_input_tokens.extend(re.findall(r"(\{input\d*\})", tokenified_string))
|
|
return sorted(list(set(all_input_tokens)))
|
|
|
|
def _verify_input_tokens(self) -> None:
|
|
"""
|
|
Verify the input tokens.
|
|
|
|
The number of input tokens should match the number of unique input tokens defined in the
|
|
path and the user's template.
|
|
|
|
Raises:
|
|
exceptions.MismatchedTokenError: Raises
|
|
[MismatchedTokenError][tembo.exceptions.MismatchedTokenError] if the number of
|
|
input tokens does not match the number of unique input tokens defined.
|
|
"""
|
|
if len(self._all_input_tokens) > 0 and self.options.user_input is None:
|
|
raise exceptions.MismatchedTokenError(expected=len(self._all_input_tokens), given=0)
|
|
if self.options.user_input is None:
|
|
return
|
|
if len(self._all_input_tokens) != len(self.options.user_input):
|
|
raise exceptions.MismatchedTokenError(
|
|
expected=len(self._all_input_tokens),
|
|
given=len(self.options.user_input),
|
|
)
|
|
|
|
def _substitute_tokens(self, tokenified_string: str) -> str:
|
|
"""For a tokened string, substitute input, name and date tokens."""
|
|
tokenified_string = self.__substitute_input_tokens(tokenified_string)
|
|
tokenified_string = self.__substitute_name_tokens(tokenified_string)
|
|
tokenified_string = self.__substitute_date_tokens(tokenified_string)
|
|
return tokenified_string
|
|
|
|
def __substitute_input_tokens(self, tokenified_string: str) -> str:
|
|
"""
|
|
Substitue the input tokens in a `str` with the user input.
|
|
|
|
Args:
|
|
tokenified_string (str): a string with input tokens.
|
|
|
|
Returns:
|
|
str: the string with the input tokens replaced by the user input.
|
|
|
|
Examples:
|
|
A `user_input` of `("monthly_meeting",)` with a `tokenified_string` of
|
|
`/meetings/{input0}/` results in a string of `/meetings/monthly_meeting/`
|
|
"""
|
|
if self.options.user_input is not None:
|
|
for input_value, extracted_token in zip(
|
|
self.options.user_input, self._all_input_tokens
|
|
):
|
|
tokenified_string = tokenified_string.replace(
|
|
extracted_token, input_value.replace(" ", "_")
|
|
)
|
|
return tokenified_string
|
|
|
|
def __substitute_name_tokens(self, tokenified_string: str) -> str:
|
|
"""Find any `{name}` tokens and substitute for the name value in a `str`."""
|
|
name_extraction = re.findall(r"(\{name\})", tokenified_string)
|
|
for extracted_input in name_extraction:
|
|
tokenified_string = tokenified_string.replace(extracted_input, self.options.name)
|
|
return tokenified_string
|
|
|
|
@staticmethod
|
|
def __substitute_date_tokens(tokenified_string: str) -> str:
|
|
"""Find any {d:%d-%M-%Y} tokens in a `str`."""
|
|
# extract the full token string
|
|
date_extraction_token = re.findall(r"(\{d\:[^}]*\})", tokenified_string)
|
|
for extracted_token in date_extraction_token:
|
|
# extract the inner %d-%M-%Y only
|
|
strftime_value = re.match(r"\{d\:([^\}]*)\}", extracted_token)
|
|
if strftime_value is not None:
|
|
strftime_value = strftime_value.group(1)
|
|
if isinstance(strftime_value, str):
|
|
tokenified_string = tokenified_string.replace(
|
|
extracted_token, pendulum.now().strftime(strftime_value)
|
|
)
|
|
return tokenified_string
|
|
|
|
|
|
class Page(metaclass=ABCMeta):
|
|
"""
|
|
Abstract Page class.
|
|
|
|
This interface is used to define a `Page` object.
|
|
|
|
A `Page` represents a note/page that will be saved to disk.
|
|
|
|
!!! abstract
|
|
This object is an abstract base class and should be implemented for each `Page` type.
|
|
"""
|
|
|
|
@abstractmethod
|
|
def __init__(self, path: pathlib.Path, page_content: str) -> None:
|
|
"""
|
|
When implemented this should initalise a Page object.
|
|
|
|
Args:
|
|
path (pathlib.Path): the full path of the page including the filename as a
|
|
[Path][pathlib.Path].
|
|
page_content (str): the contents of the page.
|
|
|
|
!!! abstract
|
|
This method is abstract and should be implemented for each `Page` type.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
@property
|
|
@abstractmethod
|
|
def path(self) -> pathlib.Path:
|
|
"""
|
|
When implemented this should return the full path of the page including the filename.
|
|
|
|
Returns:
|
|
pathlib.Path: the path as a [Path][pathlib.Path] object.
|
|
|
|
!!! abstract
|
|
This property is abstract and should be implemented for each `Page` type.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
@abstractmethod
|
|
def save_to_disk(self) -> tembo.utils.Success:
|
|
"""
|
|
When implemented this should save the page to disk.
|
|
|
|
Returns:
|
|
tembo.utils.Success: A Tembo [Success][tembo.utils.__init__.Success] object.
|
|
|
|
!!! abstract
|
|
This method is abstract and should be implemented for each `Page` type.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
|
|
class ScopedPage(Page):
|
|
"""
|
|
A page that uses substitute tokens.
|
|
|
|
Attributes:
|
|
page_content (str): the content of the page from the template.
|
|
"""
|
|
|
|
def __init__(self, path: pathlib.Path, page_content: str) -> None:
|
|
"""
|
|
Initalise a scoped page object.
|
|
|
|
Args:
|
|
path (pathlib.Path): a [Path][pathlib.Path] object of the page's filepath including
|
|
the filename.
|
|
page_content (str): the content of the page from the template.
|
|
"""
|
|
self._path = path
|
|
self.page_content = page_content
|
|
|
|
def __str__(self) -> str:
|
|
"""
|
|
Return a `str` representation of a `ScopedPage`.
|
|
|
|
Examples:
|
|
```
|
|
>>> str(ScopedPage(Path("/home/bob/tembo/meetings/my_meeting_0.md"), ""))
|
|
ScopedPage("/home/bob/tembo/meetings/my_meeting_0.md")
|
|
```
|
|
|
|
Returns:
|
|
str: The `ScopedPage` as a `str`.
|
|
"""
|
|
return f'ScopedPage("{self.path}")'
|
|
|
|
@property
|
|
def path(self) -> pathlib.Path:
|
|
"""
|
|
Return the full path of the page.
|
|
|
|
Returns:
|
|
pathlib.path: The full path of the page as a [Path][pathlib.Path] object.
|
|
"""
|
|
return self._path
|
|
|
|
def save_to_disk(self) -> tembo.utils.Success:
|
|
"""
|
|
Save the scoped page to disk and write the `page_content`.
|
|
|
|
Raises:
|
|
exceptions.ScopedPageAlreadyExists: Raises
|
|
[ScopedPageAlreadyExists][tembo.exceptions.ScopedPageAlreadyExists] if the page already exists.
|
|
|
|
Returns:
|
|
tembo.utils.Success: A [Success][tembo.utils.__init__.Success] with the path of the
|
|
ScopedPage as the message.
|
|
"""
|
|
# create the parent directories
|
|
scoped_page_file = pathlib.Path(self.path)
|
|
scoped_page_file.parents[0].mkdir(parents=True, exist_ok=True)
|
|
if scoped_page_file.exists():
|
|
raise exceptions.ScopedPageAlreadyExists(f"{self.path} already exists")
|
|
with scoped_page_file.open("w", encoding="utf-8") as scoped_page:
|
|
scoped_page.write(self.page_content)
|
|
return tembo.utils.Success(str(self.path))
|