Use a Plugin-Based Approach to Provide a Linter-Inspired Interface
**This PR adopts a plugin-based approach to provide a linter-inspired interface.**
This means:
-- A person using GatorGrader will be able to introduce their own checks dynamically
-- The command-line interface will feature positional required arguments for all checks
This PR also introduces a significant number of bug fixes and improvements to the tool. For instance, there were a number of problems in the way that GatorGrader previously handled the combination of wildcards with the use of `--exact`. These issues have been resolved, at the expense of making the code more complex. I suggest that we still merge this PR and then revisit this complexity in a future PR. For instance, the `invoke_all_comment_checks` function in the `invoke` module is now more complex that is appropriate, in part because it now correctly makes diagnostics for a wide variety of edge cases.
What is the current behavior?
GatorGrader currently does not:
-- Allow a person to specify their own checks. As such, the tool is currently not extensible.
-- Allow the specification of checks in an easy-to-document and easy-to-use fashion.
What is the new behavior if this PR is merged?
GatorGrader will now support a plugin-based and linter-inspired interface. It does so by using the PluginBase package described at: https://github.com/mitsuhiko/pluginbase
jjumadinova and amohangit and dluman and obonhamcarter please note that once this PR is merged, the way in which you will write checks in a `gatorgrader.yml` file will change. If you want to continue to use the old approach, you can use a `version:` tag in the YAML header of your `gatorgrader.yml` file, using a method that Michionlion will finish implementing before the Fall 2019 semester starts. With that said, Michionlion and all of the other core maintainers of GatorGrader suggest that you adopt this new approach instead as it will be easier to specify your own checks and add checks that are not currently provided by GatorGrader.
If jjumadinova or amohangit or dluman or obonhamcarter have questions about the implications of merging this PR to GatorGrader, they should communicate with gkapfham.
Unit Testing Details
There are currently a few branches and lines of code that are not covered by the unit test test suite. With that said, this PR has 99% coverage of the statements and the branches and I think that this is high enough to support the merge of the PR.
Here are the details from the coverage report:
Test session starts (platform: linux, Python 3.7.3, pytest 5.0.1, pytest-sugar 0.9.2)
rootdir: /home/gkapfham/working/source/gatorgrader, inifile: pytest.ini
plugins: sugar-0.9.2, cov-2.7.1
collecting ...
tests/test_arguments.py ✓✓✓✓✓✓✓✓✓✓ 1% ▎
tests/test_checkers.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 4% ▍
tests/test_comments.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 17% █▊
✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 20% ██
tests/test_constants.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 24% ██▌
tests/test_display.py ✓✓✓✓✓✓ 25% ██▌
tests/test_files.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 27% ██▊
tests/test_fragments.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 40% ████
✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 46% ████▋
tests/test_invoke.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 50% █████
tests/test_leave.py ✓✓✓✓✓✓✓✓✓✓ 51% █████▏
tests/test_markdown.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 53% █████▍
tests/test_orchestrate.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 55% █████▌
tests/test_report.py ✓✓✓✓✓✓ 56% █████▋
tests/test_repository.py ✓✓✓✓✓✓✓✓✓✓✓ 57% █████▊
tests/test_run.py ✓✓✓✓✓✓ 58% █████▊
tests/test_util.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 63% ██████▍
tests/checks/test_check_ConfirmFileExists.py ✓✓✓✓✓✓✓✓✓✓✓ 64% ██████▍
tests/checks/test_check_CountCommandOutput.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 66% ██████▋
tests/checks/test_check_CountCommits.py ✓✓✓✓✓✓✓✓✓✓✓ 67% ██████▊
tests/checks/test_check_CountFileLines.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 70% ██████▉
tests/checks/test_check_CountFileParagraphs.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 72% ███████▎
tests/checks/test_check_CountFileWords.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 74% ███████▍
tests/checks/test_check_CountMarkdownTags.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 77% ███████▋
tests/checks/test_check_CountMultipleLineComments.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 80% ████████
tests/checks/test_check_CountParagraphWords.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 82% ████████▎
tests/checks/test_check_CountSingleLineComments.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 86% ████████▋
tests/checks/test_check_ExecuteCommand.py ✓✓✓✓✓✓✓✓✓✓ 87% ████████▊
tests/checks/test_check_ListChecks.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 89% ████████▉
tests/checks/test_check_MatchCommandFragment.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 92% █████████▎
tests/checks/test_check_MatchCommandRegex.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 95% █████████▌
tests/checks/test_check_MatchFileFragment.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 98% █████████▊
tests/checks/test_check_MatchFileRegex.py ✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓✓ 100% ██████████
----------- coverage: platform linux, python 3.7.3-final-0 -----------
Name Stmts Miss Branch BrPart Cover Missing
---------------------------------------------------------------------------------------------
gator/__init__.py 0 0 0 0 100%
gator/arguments.py 31 0 4 0 100%
gator/checkers.py 84 0 36 0 100%
gator/checks/__init__.py 0 0 0 0 100%
gator/checks/check_ConfirmFileExists.py 16 0 0 0 100%
gator/checks/check_CountCommandOutput.py 19 0 0 0 100%
gator/checks/check_CountCommits.py 18 0 0 0 100%
gator/checks/check_CountFileLines.py 22 0 0 0 100%
gator/checks/check_CountFileParagraphs.py 21 0 0 0 100%
gator/checks/check_CountFileWords.py 23 0 0 0 100%
gator/checks/check_CountMarkdownTags.py 24 0 0 0 100%
gator/checks/check_CountMultipleLineComments.py 24 0 0 0 100%
gator/checks/check_CountParagraphWords.py 23 0 0 0 100%
gator/checks/check_CountSingleLineComments.py 24 0 0 0 100%
gator/checks/check_ExecuteCommand.py 14 0 0 0 100%
gator/checks/check_ListChecks.py 28 0 4 0 100%
gator/checks/check_MatchCommandFragment.py 21 0 0 0 100%
gator/checks/check_MatchCommandRegex.py 21 0 0 0 100%
gator/checks/check_MatchFileFragment.py 24 0 0 0 100%
gator/checks/check_MatchFileRegex.py 24 0 0 0 100%
gator/comments.py 26 0 0 0 100%
gator/constants.py 25 0 0 0 100%
gator/display.py 20 0 0 0 100%
gator/entities.py 29 0 6 0 100%
gator/files.py 39 0 8 0 100%
gator/fragments.py 114 0 40 0 100%
gator/invoke.py 200 4 80 4 96% 186->202, 193->197, 197->198, 198-201, 330->336
gator/leave.py 5 0 2 0 100%
gator/markdown.py 21 0 6 0 100%
gator/orchestrate.py 67 0 14 0 100%
gator/report.py 34 0 2 0 100%
gator/repository.py 24 0 2 0 100%
gator/run.py 25 0 4 0 100%
gator/util.py 145 0 54 0 100%
---------------------------------------------------------------------------------------------
TOTAL 1235 4 262 4 99%
Results (11.27s):
922 passed
Integration Testing Details
This PR was tested with an \"integration test suite\" involving a series of checks in a `gatorgrader.yml` file that is included below. Please note that all of these checks are designed to pass correctly. There are also some comments in the file that explain the types of checks that would not, in fact, pass correctly.
---
name: cmpsc-203-spring-2019-practical3
break: true
indent: 4
revision: feature/use-linters-plugins
---
--> check the source code for various characteristics
note that without an \"--exact\" the check is an \"at least\"
termfrequency:
compute_tf_cookbook.py:
ConfirmFileExists
CountFileLines --count 10
CountFileLines --count 86 --exact
MatchFileFragment --fragment \"TODO\" --count 0 --exact
MatchFileFragment --fragment \"for tf in word_freqs\" --count 1
MatchFileFragment --fragment \"word_freqs\" --count 1
CountSingleLineComments --language Python --count 12
CountMultipleLineComments --language Python --count 7 --exact
tests:
test_compute_tf_cookbook.py:
ConfirmFileExists
MatchFileFragment --fragment \"from termfrequency import compute_tf_cookbook\" --count 1
MatchFileFragment --fragment \"test_\" --count 5
CountFileLines --count 20
termfrequency:
*.py:
MatchFileFragment --fragment \"TODO\" --count 0 --exact
CountFileLines --count 0
tests:
*.py:
MatchFileFragment --fragment \"TODO\" --count 0 --exact
CountFileLines --count 0
--> check the technical writing, specify a specific file
mdl is a Markdown linting tool
proselint checks writing for common mistakes
writing/reflection.md:
mdl
proselint
ConfirmFileExists
CountParagraphWords --count 100
CountParagraphWords --count 100 --exact
CountFileWords --count 200
CountFileWords --count 600 --exact
MatchFileFragment --fragment \"\" --count 2 --exact
MatchFileRegex --regex \"\" --count 2 --exact
MatchFileRegex --regex \" Explain [.]*\" --count 4 --exact
CountMarkdownTags --tag \"heading\" --count 8 --exact
CountMarkdownTags --tag \"code\" --count 3 --exact
CountMarkdownTags --tag \"code_block\" --count 1 --exact
CountFileParagraphs --count 6 --exact
--> check the technical writing, specify a specific file
Note that third-party tools like mdl and proselint may not work with
wildcards and thus, since they are not an official part of GatorGrader
they cannot be specified with a wildcard filename.
Also note that in this case there is only one file in the writing/
directory and thus the checks are exactly the same as the prior block.
writing/*.md:
CountParagraphWords --count 100
CountParagraphWords --count 100 --exact
CountFileWords --count 200
CountFileWords --count 600 --exact
MatchFileFragment --fragment \"\" --count 2 --exact
MatchFileRegex --regex \"\" --count 2 --exact
MatchFileRegex --regex \" Explain [.]*\" --count 4 --exact
CountMarkdownTags --tag \"heading\" --count 8 --exact
CountMarkdownTags --tag \"code\" --count 3 --exact
CountMarkdownTags --tag \"code_block\" --count 1 --exact
CountFileParagraphs --count 6 --exact
--> check the number of commits beyond a threshold
CountCommits --count 10
--> check that the program executes and produces 4 lines of output
ExecuteCommand --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\"
CountCommandOutput --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\" --count 8 --exact
MatchCommandFragment --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\" --count 0 --fragment \"the\" --exact
MatchCommandFragment --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\" --count 1 --fragment \"live\" --exact
MatchCommandFragment --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\" --count 6 --fragment \"1\" --exact
MatchCommandRegex --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\" --count 0 --regex \"(the)+\" --exact
MatchCommandRegex --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\" --count 1 --regex \"(live)+\" --exact
MatchCommandRegex --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\" --count 6 --regex \"1\" --exact
this would not work correctly because it matches everything
MatchCommandRegex --command \"pipenv run python3 termfrequency/compute_tf_cookbook.py inputs/input.txt\" --count 1 --regex \"(the)*\" --exact
--> check that the test suite executes and does not fail
ExecuteCommand --command \"pipenv run pytest\"
ExecuteCommand --command \"pipenv run pytest -x -s --cov-config pytest.cov --cov-report term-missing --cov\"
Here is the output from running these checks:
gkapfham in solutions/cs203-S2019-practical3-solution masterU8 C6 gradle grade
> Configure project :