mirror of
				https://github.com/JonasunderscoreJones/epr_grader.git
				synced 2025-10-25 09:19:18 +02:00 
			
		
		
		
	Counting violations and removing unnecessary violations
This commit is contained in:
		
							parent
							
								
									84028239d8
								
							
						
					
					
						commit
						6c1381ffb0
					
				
					 3 changed files with 179 additions and 13 deletions
				
			
		|  | @ -28,7 +28,7 @@ blatt0 | ||||||
| Führt jetzt den Startbefehl aus: | Führt jetzt den Startbefehl aus: | ||||||
| ```cmd | ```cmd | ||||||
| cd ...\Tutorium\blatt0 | cd ...\Tutorium\blatt0 | ||||||
| python eprgrader.py begin --table Bewertungstabelle_EPR_08.xlsx --no-stylecheck --tests | python eprgrader.py begin --table Bewertungstabelle_EPR_4.xlsx | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| Zusätzliche Optionen: | Zusätzliche Optionen: | ||||||
|  |  | ||||||
							
								
								
									
										57
									
								
								eprgrader.py
									
										
									
									
									
								
							
							
						
						
									
										57
									
								
								eprgrader.py
									
										
									
									
									
								
							|  | @ -22,12 +22,15 @@ import shutil | ||||||
| import sys | import sys | ||||||
| import unicodedata | import unicodedata | ||||||
| import zipfile | import zipfile | ||||||
|  | import re | ||||||
| 
 | 
 | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
| 
 | 
 | ||||||
| from pylint.lint import Run as RunPylint | from pylint.lint import Run as RunPylint | ||||||
| import pycodestyle | import pycodestyle | ||||||
| 
 | 
 | ||||||
|  | from violation_checker import ViolationChecker | ||||||
|  | 
 | ||||||
| PYLINT_ARGS = [ | PYLINT_ARGS = [ | ||||||
|     '--exit-zero',  # always exit with code 0, even when problems are found |     '--exit-zero',  # always exit with code 0, even when problems are found | ||||||
|     '--load-plugins=eprcheck_2019',  # load plugin for checking __author__ variable |     '--load-plugins=eprcheck_2019',  # load plugin for checking __author__ variable | ||||||
|  | @ -145,7 +148,6 @@ def pylint_context(stdout, workdir): | ||||||
|     sys.path = tmp_storage['path'] |     sys.path = tmp_storage['path'] | ||||||
|     sys.stdout = sys.__stdout__ |     sys.stdout = sys.__stdout__ | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| def lint_files(folders, author_pairs): | def lint_files(folders, author_pairs): | ||||||
|     """Run pylint and pycodestyle on all Python files anywhere within `folders'.""" |     """Run pylint and pycodestyle on all Python files anywhere within `folders'.""" | ||||||
|     count = 0 |     count = 0 | ||||||
|  | @ -180,11 +182,52 @@ def lint_files(folders, author_pairs): | ||||||
|                     print('\n') |                     print('\n') | ||||||
|         with open(folder / 'stylecheck.txt', 'w', encoding='utf-8') as outfile: |         with open(folder / 'stylecheck.txt', 'w', encoding='utf-8') as outfile: | ||||||
|             if lintcache.tell() > 0: |             if lintcache.tell() > 0: | ||||||
|                 outfile.write(lintcache.getvalue()) |                 style_check = remove_unnecessary_violations(lintcache.getvalue()) | ||||||
|  |                 violation_checker = ViolationChecker(style_check) | ||||||
|  |                 violation_checker.check_violations() | ||||||
|  |                 style_check += (f'\n-----Verstöße insgesamt-----' | ||||||
|  |                                 f'\n{violation_checker.list_violation()}') | ||||||
|  |                 outfile.write(style_check) | ||||||
|             else: |             else: | ||||||
|                 outfile.write("Alles sieht gut aus -- weiter so!\n") |                 outfile.write("Alles sieht gut aus -- weiter so!\n") | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def remove_unnecessary_violations(style_check): | ||||||
|  |     """Function to delete all lines with a violation to ignore""" | ||||||
|  |     lines = style_check.splitlines() | ||||||
|  | 
 | ||||||
|  |     filtered_lines = [] | ||||||
|  | 
 | ||||||
|  |     skip_count = 0 | ||||||
|  | 
 | ||||||
|  |     e501_pattern = re.compile(r"E501 line too long \((\d+) > 79 characters\)") | ||||||
|  | 
 | ||||||
|  |     for i, line in enumerate(lines): | ||||||
|  |         if skip_count > 0: | ||||||
|  |             skip_count -= 1 | ||||||
|  |             continue | ||||||
|  |         # Removing lines violations which are shorter than 100 | ||||||
|  |         match = e501_pattern.search(line) | ||||||
|  |         if match: | ||||||
|  |             line_length = int(match.group(1)) | ||||||
|  |             if line_length <= 99: | ||||||
|  |                 skip_count = 2 | ||||||
|  |                 continue | ||||||
|  |         # Upper case violations | ||||||
|  |         elif "C0103" in line and "doesn't conform to UPPER_CASE naming style" in line: | ||||||
|  |             continue | ||||||
|  |         # Allowing variable and argument names with only one char | ||||||
|  |         elif ("C0103" in line and "doesn't conform to snake_case naming style" in line | ||||||
|  |             and ('Argument name "' in line or 'Variable name "' in line)): | ||||||
|  |             start_index = line.find('"') + 1 | ||||||
|  |             end_index = line.find('"', start_index) | ||||||
|  |             argument_name = line[start_index:end_index] | ||||||
|  |             if len(argument_name) == 1: | ||||||
|  |                 continue | ||||||
|  |         filtered_lines.append(line) | ||||||
|  |     return "\n".join(filtered_lines) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def fix_path(path: str) -> str: | def fix_path(path: str) -> str: | ||||||
|     return unicodedata.normalize('NFC', path).replace('U╠ê', 'Ü').replace('u╠ê', 'ü').replace( |     return unicodedata.normalize('NFC', path).replace('U╠ê', 'Ü').replace('u╠ê', 'ü').replace( | ||||||
|         '*', '').replace('"', '') |         '*', '').replace('"', '') | ||||||
|  | @ -234,9 +277,6 @@ def begin_grading(folder: pathlib.Path, ratings_file: pathlib.Path, check_style: | ||||||
|     for f in target_folders: |     for f in target_folders: | ||||||
|         target_name = "Bewertung " + sheet + " " + f.name.split('_')[0] + ratings_file.suffix |         target_name = "Bewertung " + sheet + " " + f.name.split('_')[0] + ratings_file.suffix | ||||||
|         shutil.copy(ratings_file, f / target_name) |         shutil.copy(ratings_file, f / target_name) | ||||||
|         # adding a feedback.txt |  | ||||||
|         with open(f"{f}/Feedback {f.name.split('_')[0]}.txt", "w") as my_file: |  | ||||||
|             pass |  | ||||||
|     print("Done!") |     print("Done!") | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -247,19 +287,12 @@ def finalise_grading(folder: pathlib.Path): | ||||||
|     for f in folders: |     for f in folders: | ||||||
|         target = f.parent / 'korrekturen' |         target = f.parent / 'korrekturen' | ||||||
|         target.mkdir() |         target.mkdir() | ||||||
|         feedback_parent = f.parent / 'feedback' # file for the feedbacks |  | ||||||
|         feedback_parent.mkdir() |  | ||||||
|         for handin in (x for x in f.iterdir() if x.name != '.DS_Store'): |         for handin in (x for x in f.iterdir() if x.name != '.DS_Store'): | ||||||
|             this_target = target / handin.name |             this_target = target / handin.name | ||||||
|             this_target.mkdir() |             this_target.mkdir() | ||||||
|             # copy the stylecheck datas |             # copy the stylecheck datas | ||||||
|             if (handin / 'stylecheck.txt').exists(): |             if (handin / 'stylecheck.txt').exists(): | ||||||
|                 shutil.copy(handin / 'stylecheck.txt', this_target) |                 shutil.copy(handin / 'stylecheck.txt', this_target) | ||||||
|             # copy the feedback datas |  | ||||||
|             feedback_files = list(handin.glob('Feedback*')) |  | ||||||
|             if feedback_files: |  | ||||||
|                 for feedback_file in feedback_files: |  | ||||||
|                     shutil.copy(feedback_file, feedback_parent) |  | ||||||
|             # copy the grading datas |             # copy the grading datas | ||||||
|             glob = list(handin.glob('Bewertung *')) |             glob = list(handin.glob('Bewertung *')) | ||||||
|             if len(glob) == 1: |             if len(glob) == 1: | ||||||
|  |  | ||||||
							
								
								
									
										133
									
								
								violation_checker.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								violation_checker.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,133 @@ | ||||||
|  | __author__ = 'Lukas Horst' | ||||||
|  | 
 | ||||||
|  | import re | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class ViolationChecker: | ||||||
|  | 
 | ||||||
|  |     w0311 = 0  # Bad indention | ||||||
|  |     w0401 = 0   # Wildcard import | ||||||
|  |     w0622 = 0   # Redefined builtin | ||||||
|  |     c0103 = 0   # Invalid name | ||||||
|  |     c0116 = 0   # Missing function or method docstring | ||||||
|  |     c0114 = 0   # Missing module docstring | ||||||
|  |     c0121 = 0   # Singleton-comparison | ||||||
|  |     c0325 = 0   # Superfluous-parens | ||||||
|  |     c0413 = 0   # Wrong import position | ||||||
|  |     c2100 = 0   # Missing author variable | ||||||
|  |     c2101 = 0   # Malformed author variable | ||||||
|  |     c2102 = 0   # Incorrectly assigned author variable | ||||||
|  |     e0001 = 0   # Syntax error | ||||||
|  |     e0102 = 0   # Function redefined | ||||||
|  |     e231 = 0    # Missing whitespace after ',' | ||||||
|  |     e251 = 0    # Unexpected spaces around keyword / parameter equals | ||||||
|  |     e261 = 0    # At least two spaces before inline comment | ||||||
|  |     e265 = 0    # Block comment should start with '# ' | ||||||
|  |     e271 = 0    # Multiple space after keyword | ||||||
|  |     e302 = 0    # Expected 2 blank lines | ||||||
|  |     e501 = 0    # Line too long > 99 | ||||||
|  |     style_check = '' | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     def __init__(self, style_check): | ||||||
|  |         self.style_check = style_check | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     def check_violations(self): | ||||||
|  |         w0311_violations = re.findall(r',*W0311.*', self.style_check) | ||||||
|  |         self.w0311 = len(w0311_violations) | ||||||
|  | 
 | ||||||
|  |         w0401_violations = re.findall(r',*W0401.*', self.style_check) | ||||||
|  |         self.w0401 = len(w0401_violations) | ||||||
|  | 
 | ||||||
|  |         w0622_violations = re.findall(r',*W0622.*', self.style_check) | ||||||
|  |         self.w0622 = len(w0622_violations) | ||||||
|  | 
 | ||||||
|  |         c0103_violations = re.findall(r',*C0103.*', self.style_check) | ||||||
|  |         self.c0103 = len(c0103_violations) | ||||||
|  | 
 | ||||||
|  |         c0114_violations = re.findall(r',*C0114.*', self.style_check) | ||||||
|  |         self.c0114 = len(c0114_violations) | ||||||
|  | 
 | ||||||
|  |         c0116_violations = re.findall(r',*C0116.*', self.style_check) | ||||||
|  |         self.c0116 = len(c0116_violations) | ||||||
|  | 
 | ||||||
|  |         c0121_violations = re.findall(r',*C0121.*', self.style_check) | ||||||
|  |         self.c0121 = len(c0121_violations) | ||||||
|  | 
 | ||||||
|  |         c0325_violations = re.findall(r',*C0325.*', self.style_check) | ||||||
|  |         self.c0325 = len(c0325_violations) | ||||||
|  | 
 | ||||||
|  |         c0413_violations = re.findall(r',*C0413.*', self.style_check) | ||||||
|  |         self.c0413 = len(c0413_violations) | ||||||
|  | 
 | ||||||
|  |         c2100_violations = re.findall(r',*C2100.*', self.style_check) | ||||||
|  |         self.c2100 = len(c2100_violations) | ||||||
|  | 
 | ||||||
|  |         c2101_violations = re.findall(r',*C2101.*', self.style_check) | ||||||
|  |         self.c2101 = len(c2101_violations) | ||||||
|  | 
 | ||||||
|  |         c2102_violations = re.findall(r',*C2102.*', self.style_check) | ||||||
|  |         self.c2102 = len(c2102_violations) | ||||||
|  | 
 | ||||||
|  |         e0001_violations = re.findall(r',*E0001.*', self.style_check) | ||||||
|  |         self.e0001 = len(e0001_violations) | ||||||
|  | 
 | ||||||
|  |         e0102_violations = re.findall(r',*E0102.*', self.style_check) | ||||||
|  |         self.e0102 = len(e0102_violations) | ||||||
|  | 
 | ||||||
|  |         e265_violations = re.findall(r',*E265.*', self.style_check) | ||||||
|  |         self.e265 = len(e265_violations) | ||||||
|  | 
 | ||||||
|  |         e501_violations = re.findall(r',*E501.*', self.style_check) | ||||||
|  |         self.e501 = len(e501_violations) | ||||||
|  | 
 | ||||||
|  |         e302_violations = re.findall(r',*E302.*', self.style_check) | ||||||
|  |         self.e302 = len(e302_violations) | ||||||
|  | 
 | ||||||
|  |         e231_violations = re.findall(r',*E231.*', self.style_check) | ||||||
|  |         self.e231 = len(e231_violations) | ||||||
|  | 
 | ||||||
|  |         e261_violations = re.findall(r',*E261.*', self.style_check) | ||||||
|  |         self.e261 = len(e261_violations) | ||||||
|  | 
 | ||||||
|  |         e271_violations = re.findall(r',*E271.*', self.style_check) | ||||||
|  |         self.e271 = len(e271_violations) | ||||||
|  | 
 | ||||||
|  |         e251_violations = re.findall(r',*E251.*', self.style_check) | ||||||
|  |         self.e251 = len(e251_violations) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     def list_violation(self): | ||||||
|  |         violations = '' | ||||||
|  |         violations += (f'W03111 (Bad indention): {self.w0311}' | ||||||
|  |                        f'\nW0401 (Wildcard import): {self.w0401}' | ||||||
|  |                        f'\nW0622 (Redefined builtin): {self.w0622}' | ||||||
|  |                        f'\nC0103 (Invalid name): {self.c0103}' | ||||||
|  |                        f'\nC0114 (Missing module docstring): {self.c0114}' | ||||||
|  |                        f'\nC0116 (Missing function or method docstring): {self.c0116}' | ||||||
|  |                        f'\nC0121 (Singleton-comparison): {self.c0121}' | ||||||
|  |                        f'\nC0325 (Superfluous-parens): {self.c0325}' | ||||||
|  |                        f'\nC0413 (Wrong import position): {self.c0413}' | ||||||
|  |                        f'\nC2100 (Missing author variable): {self.c2100}' | ||||||
|  |                        f'\nC2101 (Missing author variable): {self.c2101}' | ||||||
|  |                        f'\nC2102 (Incorrectly assigned author variable): {self.c2102}' | ||||||
|  |                        f'\nE0001 (Syntax error): {self.e0001}' | ||||||
|  |                        f'\nE0102 (Function redefined): {self.e0102}' | ||||||
|  |                        f'\nE231 (Missing whitespace after \',\'): {self.e231}' | ||||||
|  |                        f'\nE251 (Unexpected spaces around keyword / parameter equals): {self.e251}' | ||||||
|  |                        f'\nE261 (At least two spaces before inline comment): {self.e261}' | ||||||
|  |                        f'\nE265 (Block comment should start with \'# \'): {self.e265}' | ||||||
|  |                        f'\nE271 (Multiple space after keyword): {self.e271}' | ||||||
|  |                        f'\nE302 (Expected 2 blank lines): {self.e302}' | ||||||
|  |                        f'\nE501 (Line too long > 99): {self.e501}') | ||||||
|  |         return violations | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     with open('G02_Voll/abgaben/Cynthia Celoudis_691452_assignsubmission_file/stylecheck.txt', 'r', | ||||||
|  |               encoding='utf-8') as file: | ||||||
|  |         file_content = file.read() | ||||||
|  |     violation_checker = ViolationChecker(file_content) | ||||||
|  |     violation_checker.check_violations() | ||||||
|  |     print(violation_checker.list_violation()) | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue