from pathlib import Path from typing import Optional, List from .formatting_config import FormattingConfig from .enums import TeXType, TeXFlavour from .formatterif import FormatterIF from .generic_text import GenericText import re from .constants import * from ..logger import logger class PyTeXFormatter(FormatterIF): def __init__( self, input_file: Optional[Path] = None, config: Optional[FormattingConfig] = None, tex_type: Optional[TeXType] = None, tex_flavour: Optional[TeXFlavour] = None, locate_file_config: bool = True, allow_infile_config: bool = True ): super().__init__( input_file=input_file, config=config ) self._config: Optional[FormattingConfig] = self._config # for type-hinting self._tex_type: Optional[TeXType] = tex_type self._tex_flavour: Optional[TeXFlavour] = tex_flavour self._allow_infile_config = allow_infile_config self._header: Optional[GenericText] = None if locate_file_config: file_config = self.parse_file_config() if allow_infile_config: infile_config = self.parse_infile_config() self._config = \ file_config.merge_with( infile_config, strict=True ).merge_with(self._config, strict=False) else: self._config = file_config.merge_with(self._config) else: if allow_infile_config: infile_config = self.parse_infile_config() self._config = infile_config.merge_with(self._config) def parse_file_config(self) -> Optional[FormattingConfig]: config_file = self.input_file.with_name(self.input_file.name + PYTEX_CONFIG_FILE_EXTENSION) if config_file.exists(): with open(config_file, 'r') as file: config = file.readlines() try: return FormattingConfig.from_yaml(config) except: raise NotImplementedError # Invalid yaml file format else: return None def parse_infile_config(self) -> Optional[FormattingConfig]: with open(self._input_file, "r") as file: line = file.readline() if re.match(self.config.escape_character + INFILE_CONFIG_BEGIN_CONFIG, line): if not line.strip().lstrip('%').strip() == self.config.escape_character + INFILE_CONFIG_BEGIN_CONFIG: logger.warning( "File {file}: Start of infile config invalid." ) config = [] while True: line = file.readline() if re.match(self.config.escape_character + INFILE_CONFIG_END_CONFIG, line): if not line.strip().lstrip( '%').strip() == self.config.escape_character + INFILE_CONFIG_END_CONFIG: logger.warning( "File {file}: End of infile config invalid." ) break if line == '': raise NotImplementedError # No matching end block config.append(line.lstrip('%').rstrip()) try: return FormattingConfig.from_yaml(config) except: raise NotImplementedError # Invalid yaml file format else: return None @property def config(self) -> FormattingConfig: if self._config is None: raise NotImplementedError return self._config @property def header(self) -> GenericText: if self._header is None: if not ( self.config.include_extra_header or self.config.include_build_time or self.config.include_pytex_version or self.config.include_pytex_info_text or self.config.include_repo_version or self.config.include_repo_info_text ): self._header = GenericText([]) else: self._header = GenericText([]) # TODO: handle license if self.config.include_extra_header: self._header += self.config.extra_header + [''] if self.config.include_repo_info_text: self._header += self.config.repo_info_text if self.config.include_pytex_info_text: self._header += self.config.pytex_info_text + [''] ## TODO handle rest return self._header def format_header(self, **kwargs) -> None: self._header = self.header.format( **kwargs ) # TODO: add standard keywords here