-
Notifications
You must be signed in to change notification settings - Fork 280
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
Add JSON output format for output #684
Conversation
updated commit to satisfy linters |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hello Oscar, thanks for this contribution!
I see a problem: the format this outputs is not valid JSON:
{"file": "a.yaml", "level": "error", "line": 2, "column": 4, "desc": "trailing spaces", "rule": "trailing-spaces"}
{"file": "a.yaml", "level": "error", "line": 3, "column": 4, "desc": "no new line character at the end of file", "rule": "new-line-at-end-of-file"}
… while it should be:
[
{"file": "a.yaml", "level": "error", "line": 2, "column": 4, "desc": "trailing spaces", "rule": "trailing-spaces"},
{"file": "a.yaml", "level": "error", "line": 3, "column": 4, "desc": "no new line character at the end of file", "rule": "new-line-at-end-of-file"}
]
Fortunately a valid format seems achievable easily: you can define and call Format.json_pre_all()
and Format.json_post_all()
to print [\n
and \n]\n
at start and end, and use first
to know whether to print ,\n
(see the other formats, they use first
).
yamllint/cli.py
Outdated
line = { | ||
"file": filename, | ||
"level": problem.level, | ||
"line": problem.line, | ||
"column": problem.column, | ||
"desc": problem.desc, | ||
"rule": problem.rule | ||
} | ||
return json.dumps(line) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I propose these changes:
- Return directly (like in
Format.parsable()
above). - Change
desc
todescription
. - Use single quotes for consistency with the rest of the code.
line = { | |
"file": filename, | |
"level": problem.level, | |
"line": problem.line, | |
"column": problem.column, | |
"desc": problem.desc, | |
"rule": problem.rule | |
} | |
return json.dumps(line) | |
return json.dumps({ | |
'file': filename, | |
'level': problem.level, | |
'line': problem.line, | |
'column': problem.column, | |
'description': problem.desc, | |
'rule': problem.rule, | |
}) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Took this suggestion. Given there are other suggestions, I will squash this commit in with the other suggestions at the end as well.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
regarding the json list structure. I am a bit confused by the suggestion, given that we would actually want ,\n
for every entry in the list except the last entry.
What would your suggestion be for that? Given my limited python knowledge, it may require an update to the for loop to use the built-in enumerate()
to maintain a current index and check if the current index is the last to know when we have arrived at the last element.
Something like the below:
size = len(problems)
for index, problem in enumerate(problems):
# do some formatting for every other element
if index == (size -1):
# do some formatting for the last element
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok. I have updated the PR to include the suggestions of json_pre_all and json_post_all, but I'm struggling a bit with printing the ,\n
for all except the last element given that it uses a generator and not an actual list.
I have tried casting to a list to get a length reference and current index, but received many errors in most test cases as a result. Please provide any guidance you may have.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Following up. The most recent revision seems to work as I have built it, but it is probably not the best implementation
9e6297a
to
6beddbe
Compare
Hello Oscar, thanks for following up and squashing. Sorry to be direct but I lack time:
Once it's good to go you can squash again so it's ready to merge 👍 |
No need for apology, Adrien. Your suggestion is far simpler and more readable. It was me who didn't understand it the first time. I have applied the suggested code and removed any double quotes I may have added that are not related to validating the json format. Thanks for you time reviewing the change request. |
It occurred to me that this implementation only works for a single yaml file. When providing a directory and allowing the tool to find multiple files, the opening and closing braces are added once for every file, rather than once around all "problems" Here is a quick example output when scanning multiple files where only one has an issue:
I'd expect to see a single json array as output. I'm going to close this merge request and re-open when the implementation works a bit better |
This introduces JSON output format. I based this MR on one that was existing from a few years back under the label, #68
It follows the same pattern, but rather than exposing some extra keys that suited a particular use case, it prints the fields out as they appear in the LintProblem class with the filepath added in to each record.