mirror of
https://github.com/drewcassidy/yaclog.git
synced 2024-09-01 14:58:58 +00:00
changelog module documentation
This commit is contained in:
parent
ae4a47d3f6
commit
1676b28f03
@ -17,7 +17,7 @@
|
|||||||
import datetime
|
import datetime
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
from typing import List, Tuple, Optional
|
from typing import List, Tuple, Optional, Dict
|
||||||
|
|
||||||
bullets = '+-*'
|
bullets = '+-*'
|
||||||
brackets = '[]'
|
brackets = '[]'
|
||||||
@ -61,16 +61,51 @@ def _join_markdown(segments: List[str]) -> str:
|
|||||||
|
|
||||||
|
|
||||||
class VersionEntry:
|
class VersionEntry:
|
||||||
def __init__(self):
|
"""Holds a single version entry in a :py:class:`Changelog`"""
|
||||||
self.sections = {'': []}
|
|
||||||
self.name: str = 'Unreleased'
|
def __init__(self, name: str = 'Unreleased',
|
||||||
self.date: Optional[datetime.date] = None
|
date: Optional[datetime.date] = None, tags: Optional[List[str]] = None,
|
||||||
self.tags: List[str] = []
|
link: Optional[str] = None, link_id: Optional[str] = None):
|
||||||
self.link: Optional[str] = None
|
"""
|
||||||
self.link_id: Optional[str] = None
|
Create a new version entry
|
||||||
|
|
||||||
|
:param str name: The version's name
|
||||||
|
:param date: When the version was released
|
||||||
|
:param tags: The version's tags
|
||||||
|
:param link: The version's URL
|
||||||
|
:param link_id: The version's link ID, uses the version name by default when writing
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.name: str = name
|
||||||
|
self.date: Optional[datetime.date] = date
|
||||||
|
self.tags: List[str] = tags if tags else []
|
||||||
|
self.link: Optional[str] = link
|
||||||
|
self.link_id: Optional[str] = link_id
|
||||||
self.line_no: int = -1
|
self.line_no: int = -1
|
||||||
|
self.sections: Dict[str, List[str]] = {'': []}
|
||||||
|
|
||||||
|
def add_entry(self, contents: str, section: str = '') -> None:
|
||||||
|
"""
|
||||||
|
Add a new entry to the version
|
||||||
|
|
||||||
|
:param contents: The contents string to add
|
||||||
|
:param section: Which section to add to.
|
||||||
|
"""
|
||||||
|
|
||||||
|
section = section.title()
|
||||||
|
if section not in self.sections.keys():
|
||||||
|
self.sections[section] = []
|
||||||
|
|
||||||
|
self.sections[section].append(contents)
|
||||||
|
|
||||||
def body(self, md: bool = True) -> str:
|
def body(self, md: bool = True) -> str:
|
||||||
|
"""
|
||||||
|
Get the version's body as a string
|
||||||
|
|
||||||
|
:param md: Whether or not to use markdown syntax in headings
|
||||||
|
:return: The formatted version body, without the version header
|
||||||
|
"""
|
||||||
|
|
||||||
segments = []
|
segments = []
|
||||||
|
|
||||||
for section, entries in self.sections.items():
|
for section, entries in self.sections.items():
|
||||||
@ -81,11 +116,18 @@ class VersionEntry:
|
|||||||
segments.append(f'{section.upper()}:')
|
segments.append(f'{section.upper()}:')
|
||||||
|
|
||||||
if len(entries) > 0:
|
if len(entries) > 0:
|
||||||
segments.append(_join_markdown(entries))
|
segments += entries
|
||||||
|
|
||||||
return _join_markdown(segments)
|
return _join_markdown(segments)
|
||||||
|
|
||||||
def header(self, md: bool = True) -> str:
|
def header(self, md: bool = True) -> str:
|
||||||
|
"""
|
||||||
|
Get the version's header as a string
|
||||||
|
|
||||||
|
:param md: Whether or not to use markdown syntax in headings
|
||||||
|
:return: The formatted version header
|
||||||
|
"""
|
||||||
|
|
||||||
segments = []
|
segments = []
|
||||||
|
|
||||||
if md:
|
if md:
|
||||||
@ -107,22 +149,49 @@ class VersionEntry:
|
|||||||
return ' '.join(segments)
|
return ' '.join(segments)
|
||||||
|
|
||||||
def text(self, md: bool = True) -> str:
|
def text(self, md: bool = True) -> str:
|
||||||
return self.header(md) + '\n\n' + self.body(md)
|
"""
|
||||||
|
Get the version's contents as a string
|
||||||
|
|
||||||
|
:param md: Whether or not to use markdown syntax in headings
|
||||||
|
:return: The formatted version header and body
|
||||||
|
"""
|
||||||
|
|
||||||
|
contents = self.header(md)
|
||||||
|
body = self.body(md)
|
||||||
|
if body:
|
||||||
|
contents += '\n\n' + body
|
||||||
|
return contents
|
||||||
|
|
||||||
def __str__(self) -> str:
|
def __str__(self) -> str:
|
||||||
return self.header(False)
|
return self.header(False)
|
||||||
|
|
||||||
|
|
||||||
class Changelog:
|
class Changelog:
|
||||||
def __init__(self, path=None):
|
def __init__(self, path=None, header: str = default_header):
|
||||||
self.path: os.PathLike = path
|
"""
|
||||||
self.header: str = ''
|
Create a new changelog object. Contents will be automatically read from disk if the file exists
|
||||||
|
|
||||||
|
:param path: The changelog's path on disk
|
||||||
|
:param header: The header at the top of the changelog to use if the file does not exist
|
||||||
|
"""
|
||||||
|
self.path = path
|
||||||
|
self.header: str = header
|
||||||
self.versions: List[VersionEntry] = []
|
self.versions: List[VersionEntry] = []
|
||||||
self.links = {}
|
self.links = {}
|
||||||
|
|
||||||
if not path or not os.path.exists(path):
|
if path and os.path.exists(path):
|
||||||
self.header = default_header
|
self.read()
|
||||||
return
|
|
||||||
|
def read(self, path=None) -> None:
|
||||||
|
"""
|
||||||
|
Read a markdown changelog file from disk
|
||||||
|
|
||||||
|
:param path: The changelog's path on disk. By default, :py:attr:`~Changelog.path` is used.
|
||||||
|
"""
|
||||||
|
|
||||||
|
if not path:
|
||||||
|
# use the object path if none was provided
|
||||||
|
path = self.path
|
||||||
|
|
||||||
# Read file
|
# Read file
|
||||||
with open(path, 'r') as fp:
|
with open(path, 'r') as fp:
|
||||||
@ -257,8 +326,15 @@ class Changelog:
|
|||||||
# strip whitespace from header
|
# strip whitespace from header
|
||||||
self.header = _join_markdown(header_segments)
|
self.header = _join_markdown(header_segments)
|
||||||
|
|
||||||
def write(self, path: os.PathLike = None):
|
def write(self, path: os.PathLike = None) -> None:
|
||||||
|
"""
|
||||||
|
Write a markdown changelog to a file.
|
||||||
|
|
||||||
|
:param path: The changelog's path on disk. By default, :py:attr:`~Changelog.path` is used.
|
||||||
|
"""
|
||||||
|
|
||||||
if path is None:
|
if path is None:
|
||||||
|
# use the object path if none was provided
|
||||||
path = self.path
|
path = self.path
|
||||||
|
|
||||||
segments = [self.header]
|
segments = [self.header]
|
||||||
|
Loading…
Reference in New Issue
Block a user