-
Notifications
You must be signed in to change notification settings - Fork 142
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: add --format flag and *ErrorsFormatter classes (#482)
* move Errors class from file.py to errors.py * feat: add --format flag and *ErrorsFormatter classes
- Loading branch information
Showing
6 changed files
with
173 additions
and
38 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,110 @@ | ||
from __future__ import annotations | ||
|
||
import os | ||
import json | ||
from dataclasses import dataclass, field, asdict | ||
from functools import cmp_to_key | ||
from typing import TYPE_CHECKING, Sequence, Union, Literal, Optional, List | ||
|
||
from norminette.norm_error import NormError, NormWarning, errors as errors_dict | ||
|
||
if TYPE_CHECKING: | ||
from norminette.file import File | ||
|
||
|
||
def sort_errs(a: Error, b: Error): | ||
# TODO Add to Error and Highlight dataclasses be sortable to remove this fn | ||
ah: Highlight = a.highlights[0] | ||
bh: Highlight = b.highlights[0] | ||
if ah.column == bh.column and ah.lineno == bh.lineno: | ||
return 1 if a.name > b.name else -1 | ||
return ah.column - bh.column if ah.lineno == bh.lineno else ah.lineno - bh.lineno | ||
|
||
|
||
@dataclass | ||
class Highlight: | ||
lineno: int | ||
column: int | ||
length: Optional[int] = field(default=None) | ||
hint: Optional[str] = field(default=None) | ||
|
||
|
||
@dataclass | ||
class Error: | ||
name: str | ||
text: str | ||
level: Literal["Error", "Notice"] | ||
highlights: List[Highlight] | ||
|
||
|
||
class Errors: | ||
__slots__ = "_inner" | ||
|
||
def __init__(self) -> None: | ||
self._inner = [] | ||
|
||
def __len__(self) -> int: | ||
return len(self._inner) | ||
|
||
def __iter__(self): | ||
self._inner.sort(key=cmp_to_key(sort_errs)) | ||
return iter(self._inner) | ||
|
||
# TODO Add `add(...)` method to allow creating `Highlight`s and `Error`s easily | ||
|
||
@property | ||
def status(self) -> Literal["OK", "Error"]: | ||
return "OK" if all(it.level == "Notice" for it in self._inner) else "Error" | ||
|
||
def append(self, value: Union[NormError, NormWarning]) -> None: | ||
# TODO Remove NormError and NormWarning since it does not provide `length` data | ||
assert isinstance(value, (NormError, NormWarning)) | ||
level = "Error" if isinstance(value, NormError) else "Notice" | ||
value = Error(value.errno, value.error_msg, level, highlights=[ | ||
Highlight(value.line, value.col, None), | ||
]) | ||
self._inner.append(value) | ||
|
||
|
||
class _formatter: | ||
def __init__(self, files: Union[File, Sequence[File]]) -> None: | ||
if not isinstance(files, list): | ||
files = [files] | ||
self.files = files | ||
|
||
def __init_subclass__(cls) -> None: | ||
cls.name = cls.__name__.rstrip("ErrorsFormatter").lower() | ||
|
||
|
||
class HumanizedErrorsFormatter(_formatter): | ||
def __str__(self) -> str: | ||
output = '' | ||
for file in self.files: | ||
output += f"{file.basename}: {file.errors.status}!" | ||
for error in file.errors: | ||
brief = errors_dict.get(error.name, "Error not found") | ||
highlight = error.highlights[0] | ||
output += f"\n{error.level}: {error.name:<20} " | ||
output += f"(line: {highlight.lineno:>3}, col: {highlight.column:>3}):\t{brief}" | ||
return output | ||
|
||
|
||
class JSONErrorsFormatter(_formatter): | ||
def __str__(self): | ||
files = [] | ||
for file in self.files: | ||
files.append({ | ||
"path": os.path.abspath(file.path), | ||
"status": file.errors.status, | ||
"errors": tuple(map(asdict, file.errors)), | ||
}) | ||
output = { | ||
"files": files, | ||
} | ||
return json.dumps(output, separators=",:") | ||
|
||
|
||
formatters = ( | ||
JSONErrorsFormatter, | ||
HumanizedErrorsFormatter, | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,47 @@ | ||
import json | ||
|
||
import pytest | ||
|
||
from norminette.file import File | ||
from norminette.lexer import Lexer | ||
from norminette.context import Context | ||
from norminette.registry import Registry | ||
from norminette.errors import JSONErrorsFormatter | ||
|
||
tests = [ | ||
{ | ||
"file": File("/nium/test.c", "int\tmain()\n{\n\treturn ;\n}\n"), | ||
"test": { | ||
"files": [ | ||
{ | ||
"path": "/nium/test.c", | ||
"status": "Error", | ||
"errors": [ | ||
{ | ||
"name": "INVALID_HEADER", | ||
"text": "Missing or invalid 42 header", | ||
"level": "Error", | ||
"highlights": [{"lineno": 1, "column": 1, "length": None, "hint": None}], | ||
}, | ||
{ | ||
"name": "NO_ARGS_VOID", | ||
"text": "Empty function argument requires void", | ||
"level": "Error", | ||
"highlights": [{"lineno": 1, "column": 10, "length": None, "hint": None}], | ||
}, | ||
], | ||
}, | ||
], | ||
}, | ||
}, | ||
] | ||
|
||
|
||
@pytest.mark.parametrize("file,test", [it.values() for it in tests]) | ||
def test_json_formatter_errored_file(file, test): | ||
lexer = Lexer(file) | ||
context = Context(file, lexer.get_tokens()) | ||
Registry().run(context) | ||
|
||
formatter = JSONErrorsFormatter(file) | ||
assert str(formatter) == json.dumps(test, separators=",:") |