Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: enhanced error message #28

Merged
merged 1 commit into from
Apr 4, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/workflows/ci.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -68,4 +68,4 @@ jobs:
uses: liskin/gh-problem-matcher-wrap@v3
with:
linters: isort
run: isort --line-length=88 --check src/
run: isort --line-length=88 --check --profile black src/
1 change: 1 addition & 0 deletions github_actions/event.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
This module relies on the presence of specific environment variables
set by GitHub Actions.
"""

import json
import os
from typing import Any, Dict
Expand Down
1 change: 1 addition & 0 deletions github_actions/run.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
This script contains actions to be taken based on GitHub events,
specifically for push and pull_request events.
"""

import os
import subprocess
import sys
Expand Down
4 changes: 4 additions & 0 deletions pytest.ini
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
[pytest]
pythonpath = src
python_files = test_*.py
addopts = -vvv
6 changes: 3 additions & 3 deletions src/commitlint/__init__.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
"""Main module for commitlint"""
"""Main module for commitlint."""

from .commitlint import check_commit_message
from .linter import lint_commit_message

__all__ = ["check_commit_message"]
__all__ = ["lint_commit_message"]
62 changes: 43 additions & 19 deletions src/commitlint/cli.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,10 +19,11 @@
from typing import List

from .__version__ import __version__
from .commitlint import check_commit_message, remove_comments
from .exceptions import CommitlintException
from .git_helpers import get_commit_message_of_hash, get_commit_messages_of_hash_range
from .messages import VALIDATION_SUCCESSFUL
from .linter import lint_commit_message
from .linter.utils import remove_comments
from .messages import VALIDATION_FAILED, VALIDATION_SUCCESSFUL


def get_args() -> argparse.Namespace:
Expand Down Expand Up @@ -57,28 +58,45 @@ def get_args() -> argparse.Namespace:
# --to-hash is optional
parser.add_argument("--to-hash", type=str, help="To commit hash", default="HEAD")

# feature options
parser.add_argument(
"--skip-detail",
action="store_true",
help="Skip the detailed error message check",
)

# parsing args
args = parser.parse_args()

return args


def _show_errors(commit_message: str, errors: List[str]) -> None:
def _show_errors(
commit_message: str,
errors: List[str],
skip_detail: bool = False,
) -> None:
"""
Display a formatted error message for a list of errors.

Args:
commit_message (str): The commit message to display.
errors (List[str]): A list of error messages to be displayed.
skip_detail (bool): Whether to skip the detailed error message.

"""
error_count = len(errors)
commit_message = remove_comments(commit_message)

sys.stderr.write(
f"⧗ Input:\n{commit_message}\n\n✖ Found {error_count} error(s).\n\n"
)
for index, error in enumerate(errors):
end_char = "" if index == error_count - 1 else "\n"
sys.stderr.write(f"- {error}\n{end_char}")
sys.stderr.write(f"⧗ Input:\n{commit_message}\n\n")

if skip_detail:
sys.stderr.write(f"{VALIDATION_FAILED}\n")
return

sys.stderr.write(f"✖ Found {error_count} error(s).\n")
for error in errors:
sys.stderr.write(f"- {error}\n")


def _get_commit_message_from_file(filepath: str) -> str:
Expand All @@ -101,41 +119,45 @@ def _get_commit_message_from_file(filepath: str) -> str:
return commit_message


def _handle_commit_message(commit_message: str) -> None:
def _handle_commit_message(commit_message: str, skip_detail: bool) -> None:
"""
Handles a single commit message, checks its validity, and prints the result.

Args:
commit_message (str): The commit message to be handled.
skip_detail (bool): Whether to skip the detailed error linting.

Raises:
SystemExit: If the commit message is invalid.
"""
success, errors = check_commit_message(commit_message)
success, errors = lint_commit_message(commit_message, skip_detail=skip_detail)

if success:
sys.stdout.write(f"{VALIDATION_SUCCESSFUL}\n")
else:
_show_errors(commit_message, errors)
_show_errors(commit_message, errors, skip_detail=skip_detail)
sys.exit(1)


def _handle_multiple_commit_messages(commit_messages: List[str]) -> None:
def _handle_multiple_commit_messages(
commit_messages: List[str], skip_detail: bool
) -> None:
"""
Handles multiple commit messages, checks their validity, and prints the result.

Args:
commit_messages (List[str]): List of commit messages to be handled.
skip_detail (bool): Whether to skip the detailed error linting.

Raises:
SystemExit: If any of the commit messages is invalid.
"""
has_error = False
for commit_message in commit_messages:
success, errors = check_commit_message(commit_message)
success, errors = lint_commit_message(commit_message, skip_detail=skip_detail)
if not success:
has_error = True
_show_errors(commit_message, errors)
_show_errors(commit_message, errors, skip_detail=skip_detail)
sys.stderr.write("\n")

if has_error:
Expand All @@ -153,18 +175,20 @@ def main() -> None:
try:
if args.file:
commit_message = _get_commit_message_from_file(args.file)
_handle_commit_message(commit_message)
_handle_commit_message(commit_message, skip_detail=args.skip_detail)
elif args.hash:
commit_message = get_commit_message_of_hash(args.hash)
_handle_commit_message(commit_message)
_handle_commit_message(commit_message, skip_detail=args.skip_detail)
elif args.from_hash:
commit_messages = get_commit_messages_of_hash_range(
args.from_hash, args.to_hash
)
_handle_multiple_commit_messages(commit_messages)
_handle_multiple_commit_messages(
commit_messages, skip_detail=args.skip_detail
)
else:
commit_message = args.commit_message.strip()
_handle_commit_message(commit_message)
_handle_commit_message(commit_message, skip_detail=args.skip_detail)
except CommitlintException as ex:
sys.stderr.write(f"{ex}\n")
sys.exit(1)
Expand Down
124 changes: 0 additions & 124 deletions src/commitlint/commitlint.py

This file was deleted.

25 changes: 25 additions & 0 deletions src/commitlint/constants.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,28 @@
"""This module defines constants used throughout the application."""

COMMIT_HEADER_MAX_LENGTH = 72

COMMIT_TYPES = (
"build",
"ci",
"docs",
"feat",
"fix",
"perf",
"refactor",
"style",
"test",
"chore",
"revert",
"bump",
)

IGNORE_COMMIT_PATTERNS = (
r"^((Merge pull request)|(Merge (.*?) into (.*?)|(Merge branch (.*?)))(?:\r?\n)*$)|"
r"^(Merge tag (.*?))(?:\r?\n)*$|"
r"^(R|r)evert (.*)|"
r"^(Merged (.*?)(in|into) (.*)|Merged PR (.*): (.*))$|"
r"^Merge remote-tracking branch(\s*)(.*)$|"
r"^Automatic merge(.*)$|"
r"^Auto-merged (.*?) into (.*)$"
)
1 change: 1 addition & 0 deletions src/commitlint/git_helpers.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
"""
This module contains the git related helper functions.
"""

import subprocess
from typing import List

Expand Down
7 changes: 7 additions & 0 deletions src/commitlint/linter/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
"""Main module for commit linters and validators"""

from ._linter import lint_commit_message

__all__ = [
"lint_commit_message",
]
Loading
Loading