Source code for semantic_release.commit_parser.token

from __future__ import annotations

from typing import TYPE_CHECKING, NamedTuple, NoReturn, TypeVar, Union

from semantic_release.commit_parser.util import force_str
from semantic_release.errors import CommitParseError

if TYPE_CHECKING:  # pragma: no cover
    from git.objects.commit import Commit

    from semantic_release.enums import LevelBump


[docs] class ParsedMessageResult(NamedTuple): """ A read-only named tuple object representing the result from parsing a commit message. Essentially this is a data structure which holds the parsed information from a commit message without the actual commit object itself. Very helpful for unit testing. Most of the fields will replicate the fields of a :py:class:`ParsedCommit <semantic_release.commit_parser.token.ParsedCommit>` """ bump: LevelBump type: str category: str scope: str descriptions: tuple[str, ...] breaking_descriptions: tuple[str, ...] = () release_notices: tuple[str, ...] = () linked_issues: tuple[str, ...] = () linked_merge_request: str = "" include_in_changelog: bool = True
[docs] class ParsedCommit(NamedTuple): """A read-only named tuple object representing the result of parsing a commit message.""" bump: LevelBump """A LevelBump enum value indicating what type of change this commit introduces.""" type: str """ The type of the commit as a string, per the commit message style. This is up to the parser to implement; for example, the EmojiCommitParser parser fills this field with the emoji representing the most significant change for the commit. """ scope: str """ The scope, as a string, parsed from the commit. Generally an optional field based on the commit message style which means it very likely can be an empty string. Commit styles which do not have a meaningful concept of "scope" usually fill this field with an empty string. """ descriptions: list[str] """ A list of paragraphs from the commit message. Paragraphs are generally delimited by a double-newline since git commit messages are sometimes manually wordwrapped with a single newline, but this is up to the parser to implement. """ breaking_descriptions: list[str] """ A list of paragraphs which are deemed to identify and describe breaking changes by the parser. An example would be a paragraph which begins with the text ``BREAKING CHANGE:`` in the commit message but the parser gennerally strips the prefix and includes the rest of the paragraph in this list. """ commit: Commit """The original commit object (a class defined by GitPython) that was parsed""" release_notices: tuple[str, ...] = () """ A tuple of release notices, which are additional information about the changes that affect the user. An example would be a paragraph which begins with the text ``NOTICE:`` in the commit message but the parser generally strips the prefix and includes the rest of the paragraph in this list. """ linked_issues: tuple[str, ...] = () """ A tuple of issue numbers as strings, if the commit is contains issue references. If there are no issue references, this should be an empty tuple. Although, we generally refer to them as "issue numbers", it generally should be a string to adhere to the prefixes used by the VCS (ex. ``#`` for GitHub, GitLab, etc.) or issue tracker (ex. JIRA uses ``AAA-###``). """ linked_merge_request: str = "" """ A pull request or merge request definition, if the commit is labeled with a pull/merge request number. This is a string value which includes any special character prefix used by the VCS (e.g. ``#`` for GitHub, ``!`` for GitLab). """ include_in_changelog: bool = True """ A boolean value indicating whether this commit should be included in the changelog. This enables parsers to flag commits which are not user-facing or are otherwise not relevant to the changelog to be filtered out by PSR's internal algorithms. """ @property def message(self) -> str: """ A string representation of the commit message. This is a pass through property for convience to access the ``message`` attribute of the ``commit`` object. If the message is of type ``bytes`` then it will be decoded to a ``UTF-8`` string. """ return force_str(self.commit.message).replace("\r", "") @property def hexsha(self) -> str: """ A hex representation of the hash value of the commit. This is a pass through property for convience to access the ``hexsha`` attribute of the ``commit``. """ return self.commit.hexsha @property def short_hash(self) -> str: """A short representation of the hash value (in hex) of the commit.""" return self.hexsha[:7] @property def linked_pull_request(self) -> str: """An alias to the linked_merge_request attribute.""" return self.linked_merge_request
[docs] def is_merge_commit(self) -> bool: return bool(len(self.commit.parents) > 1)
[docs] @staticmethod def from_parsed_message_result( commit: Commit, parsed_message_result: ParsedMessageResult ) -> ParsedCommit: """A convience method to create a ParsedCommit object from a ParsedMessageResult object and a Commit object.""" return ParsedCommit( bump=parsed_message_result.bump, # TODO: breaking v10, swap back to type rather than category type=parsed_message_result.category, scope=parsed_message_result.scope, descriptions=list(parsed_message_result.descriptions), breaking_descriptions=list(parsed_message_result.breaking_descriptions), commit=commit, release_notices=parsed_message_result.release_notices, linked_issues=parsed_message_result.linked_issues, linked_merge_request=parsed_message_result.linked_merge_request, include_in_changelog=parsed_message_result.include_in_changelog, )
[docs] class ParseError(NamedTuple): """A read-only named tuple object representing an error that occurred while parsing a commit message.""" commit: Commit """The original commit object (a class defined by GitPython) that was parsed""" error: str """A string with a description for why the commit parsing failed.""" @property def message(self) -> str: """ A string representation of the commit message. This is a pass through property for convience to access the ``message`` attribute of the ``commit`` object. If the message is of type ``bytes`` then it will be decoded to a ``UTF-8`` string. """ return force_str(self.commit.message).replace("\r", "") @property def hexsha(self) -> str: """ A hex representation of the hash value of the commit. This is a pass through property for convience to access the ``hexsha`` attribute of the ``commit``. """ return self.commit.hexsha @property def short_hash(self) -> str: """A short representation of the hash value (in hex) of the commit.""" return self.hexsha[:7]
[docs] def is_merge_commit(self) -> bool: return bool(len(self.commit.parents) > 1)
[docs] def raise_error(self) -> NoReturn: """A convience method to raise a CommitParseError with the error message.""" raise CommitParseError(self.error)
_T = TypeVar("_T", bound=ParsedCommit) _E = TypeVar("_E", bound=ParseError) # For extensions, this type can be used to build an alias # for example CustomParseResult = ParseResultType[CustomParsedCommit, ParseError] ParseResultType = Union[_T, _E] ParseResult = ParseResultType[ParsedCommit, ParseError]