djLint/src/djlint/output.py
2022-01-24 12:03:44 -06:00

173 lines
5.1 KiB
Python

"""Build djLint console output."""
import shutil
from pathlib import Path
from typing import Any, Dict, List
import regex as re
from click import echo
from colorama import Fore, Style
from .settings import Config
def print_output(
config: Config, file_errors: List[Dict[Any, Any]], file_count: int
) -> int:
"""Print results to console."""
file_quantity = build_quantity(file_count)
# format errors
reformat_success_message = ""
lint_success_message = ""
lint_error_count = 0
format_error_count = 0
if config.stdin is False or config.lint:
echo()
for error in sorted(file_errors, key=lambda x: next(iter(list(x.values())[0]))):
if error.get("format_message") and config.stdin is False:
# reformat message
format_error_count += build_check_output(error["format_message"], config)
if error.get("lint_message"):
# lint message
lint_error_count += build_output(error["lint_message"], config)
tense_message = (
build_quantity(format_error_count) + " would be"
if config.check is True
else build_quantity_tense(format_error_count)
)
reformat_success_message = f"{tense_message} updated."
error_case = "error" if lint_error_count == 1 else "errors"
lint_success_message += (
f"Linted {file_quantity}, found {lint_error_count} {error_case}."
)
if config.stdin is False or config.lint:
echo()
if config.stdin is False and (config.reformat or config.check):
reformat_success_color = (
Fore.RED + Style.BRIGHT if (format_error_count) > 0 else Fore.BLUE
)
echo(f"{reformat_success_color}{reformat_success_message}{Style.RESET_ALL}")
if config.lint:
lint_success_color = (
Fore.RED + Style.BRIGHT if (lint_error_count) > 0 else Fore.BLUE
)
echo(f"{lint_success_color}{lint_success_message}{Style.RESET_ALL}")
if config.stdin is False or config.lint:
echo()
return lint_error_count + format_error_count
def build_relative_path(url: str, project_root: Path) -> str:
"""Get path relative to project."""
url_path = Path(url)
if project_root != url_path and project_root in url_path.parents:
return str(url_path.relative_to(project_root.resolve()))
return url
def build_output(error: dict, config: Config) -> int:
"""Build output for file errors."""
errors = sorted(
list(error.values())[0], key=lambda x: tuple(map(int, x["line"].split(":")))
)
width, _ = shutil.get_terminal_size()
if len(errors) == 0:
return 0
filename = build_relative_path(list(error.keys())[0], config.project_root.resolve())
if "{filename}" not in config.linter_output_format:
echo(
f"{Fore.GREEN}{Style.BRIGHT}\n{filename}\n{Style.DIM}"
+ "".join(["" for x in range(1, width)])
+ Style.RESET_ALL
)
for message_dict in errors:
line = Fore.BLUE + message_dict["line"] + Style.RESET_ALL
code = (
(Fore.RED if message_dict["code"][:1] == "E" else Fore.YELLOW)
+ message_dict["code"]
+ Style.RESET_ALL
)
message = message_dict["message"]
match = (
Fore.BLUE
+ re.sub(r"\s{2,}|\n", " ", message_dict["match"])
+ Style.RESET_ALL
)
echo(
config.linter_output_format.format(
filename=filename, line=line, code=code, message=message, match=match
),
err=False,
)
return len(errors)
def build_check_output(errors: dict, config: Config) -> int:
"""Build output for reformat check."""
if len(errors) == 0:
return 0
color = {"-": Fore.YELLOW, "+": Fore.GREEN, "@": Style.BRIGHT + Fore.BLUE}
width, _ = shutil.get_terminal_size()
if config.quiet is True and len(list(errors.values())[0]) > 0:
echo(
Fore.GREEN
+ Style.BRIGHT
+ build_relative_path(list(errors.keys())[0], config.project_root.resolve())
+ Style.DIM
+ Style.RESET_ALL
)
elif config.quiet is False and len(list(errors.values())[0]) > 0:
echo(
Fore.GREEN
+ Style.BRIGHT
+ "\n"
+ build_relative_path(list(errors.keys())[0], config.project_root.resolve())
+ "\n"
+ Style.DIM
+ "".join(["" for x in range(1, width)])
+ Style.RESET_ALL
)
for diff in list(errors.values())[0][2:]:
echo(
f"{ color.get(diff[:1], Style.RESET_ALL)}{diff}{Style.RESET_ALL}",
err=False,
)
return len(list(filter(lambda x: len(x) > 0, errors.values())))
def build_quantity(size: int) -> str:
"""Count files in a list."""
return str(size) + " file" + ("s" if size > 1 or size == 0 else "")
def build_quantity_tense(size: int) -> str:
"""Count files in a list."""
return (
str(size)
+ " file"
+ ("s" if size > 1 or size == 0 else "")
+ " "
+ ("were" if size > 1 or size == 0 else "was")
)