mirror of
https://github.com/JonasunderscoreJones/epr_grader.git
synced 2025-10-24 08:59:17 +02:00
158 lines
7.7 KiB
Python
158 lines
7.7 KiB
Python
__author__ = 'Lukas Horst'
|
|
|
|
import re
|
|
|
|
|
|
class ViolationChecker:
|
|
|
|
__violation_groups = [
|
|
'No deduction',
|
|
'Global statements',
|
|
'Imports',
|
|
'Author variable',
|
|
'Naming',
|
|
'Docstring',
|
|
'Spacing',
|
|
'Classes',
|
|
'Override',
|
|
'Syntax'
|
|
]
|
|
# {violation_name: [amount of the violation, description, violation_group]}
|
|
__violations = None
|
|
_style_check = ''
|
|
__deduction = None
|
|
__module_and_class_docstrings = None
|
|
|
|
def __init__(self, style_check: str, no_deduction: bool):
|
|
self.__module_and_class_docstrings = 0
|
|
self._style_check = style_check
|
|
self.__deduction = no_deduction
|
|
self.__violations = {'W0104': [0, 'Pointless statement', 0],
|
|
'W0201': [0, 'Attribute defined outside init', 7],
|
|
'W0231': [0, 'Super init not called', 7],
|
|
'W0232': [0, 'No init', 7],
|
|
'W0301': [0, 'Unnecessary semicolon', 0],
|
|
'W0311': [0, 'Bad indention', 6],
|
|
'W0401': [0, 'Wildcard import', 2],
|
|
'W0404': [0, 'Reimported', 2],
|
|
'W0603': [0, 'Global statement', 1],
|
|
'W0622': [0, 'Redefined builtin', 8],
|
|
'W0702': [0, 'Bare except', 0],
|
|
'W0705': [0, 'Duplicate except', 0],
|
|
'W0706': [0, 'Try except raise', 0],
|
|
|
|
'C0102': [0, 'Blacklisted name', 4],
|
|
'C0103': [0, 'Invalid name', 4],
|
|
'C0112': [0, 'Empty docstring', 5],
|
|
'C0114': [0, 'Missing module docstring', 5],
|
|
'C0115': [0, 'Missing class docstring', 5],
|
|
'C0116': [0, 'Missing function or method docstring', 5],
|
|
'C0121': [0, 'Singleton-comparison', 0],
|
|
'C0144': [0, 'Non ascii name', 4],
|
|
'C0321': [0, 'Multiple statements', 0],
|
|
'C0325': [0, 'Superfluous-parens', 0],
|
|
'C0410': [0, 'Multiple imports', 2],
|
|
'C0411': [0, 'Wrong import order', 2],
|
|
'C0412': [0, 'Ungrouped imports', 2],
|
|
'C0413': [0, 'Wrong import position', 2],
|
|
'C2100': [0, 'Missing author variable', 3],
|
|
'C2101': [0, 'Malformed author variable', 3],
|
|
'C2102': [0, 'Incorrectly assigned author variable', 3],
|
|
|
|
'E0001': [0, 'Syntax error', 9],
|
|
'E0102': [0, 'Function redefined', 8],
|
|
'E0211': [0, 'No Method argument', 7],
|
|
'E201': [0, 'Whitespace after \'(\'', 6],
|
|
'E202': [0, 'Whitespace before \')\'', 6],
|
|
'E203': [0, 'Whitespace before \':\'', 6],
|
|
'E211': [0, 'Whitespace before \'(\'', 6],
|
|
'E221': [0, 'Multiple spaces before operator', 6],
|
|
'E222': [0, 'Multiple spaces after operator', 6],
|
|
'E223': [0, 'Tab before operator', 6],
|
|
'E224': [0, 'Tab after operator', 6],
|
|
'E225': [0, 'Missing whitespace around operator', 6],
|
|
'E231': [0, 'Missing whitespace after \',\', \';\', or \':\'', 6],
|
|
'E251': [0, 'Unexpected spaces around keyword / parameter equals', 6],
|
|
'E261': [0, 'At least two spaces before inline comment', 6],
|
|
'E262': [0, 'Inline comment should start with \'# \'', 6],
|
|
'E265': [0, 'Block comment should start with \'# \'', 6],
|
|
'E271': [0, 'Multiple space after keyword', 6],
|
|
'E302': [0, 'Expected 2 blank lines', 6],
|
|
'E501': [0, 'Line too long > 99', 6],
|
|
'E502': [0, 'Backslash redundant between brackets', 0],
|
|
'E713': [0, 'Negative membership test should use \'not in\'', 0],
|
|
'E714': [0, 'Negative identity test should use \'is not\'', 0],
|
|
'E721': [0, 'Use \'isinstance\' instead of comparing types', 0]}
|
|
|
|
def check_violations(self):
|
|
"""Method to search for all violations"""
|
|
for violation_name, _ in self.__violations.items():
|
|
all_violations = re.findall(rf',*{violation_name}.*', self._style_check)
|
|
if violation_name == 'C0114' or violation_name == 'C0115':
|
|
self.__module_and_class_docstrings += len(all_violations)
|
|
self.__violations[violation_name][0] = len(all_violations)
|
|
|
|
def list_violation(self):
|
|
"""Method to return a list with all violations and the amount of the violations sort by
|
|
groups"""
|
|
violation_string = ''
|
|
violation_groups_strings = []
|
|
for i in range(10):
|
|
violation_groups_strings.append('')
|
|
for violation_name, value in self.__violations.items():
|
|
violation_groups_strings[value[2]] += f'{violation_name} ({value[1]}): {value[0]}\n'
|
|
for i, violation_group in enumerate(self.__violation_groups):
|
|
if i == 0: continue
|
|
violation_string += f'-----{violation_group}-----\n{violation_groups_strings[i]}'
|
|
violation_amount = self.count_violations(i)
|
|
total_violations = f'\nFehler Insgesamt: {violation_amount}'
|
|
possible_deduction = (f'Möglicher Abzug: {self.count_deduction(i, 100)} '
|
|
f'Punkte')
|
|
deduction = f'Abzug: {self.count_deduction(i, violation_amount)} Punkte\n\n'
|
|
violation_string += f'{total_violations} {possible_deduction} {deduction}'
|
|
if i == 9:
|
|
violation_string += f'-----No deduction-----\n{violation_groups_strings[0]}'
|
|
violation_amount = self.count_violations(0)
|
|
total_violations = f'\nFehler Insgesamt: {violation_amount}'
|
|
possible_deduction = f'Möglicher Abzug: 0 Punkte'
|
|
deduction = 'Abzug: 0 Punkte'
|
|
violation_string += f'{total_violations} {possible_deduction} {deduction}'
|
|
return violation_string
|
|
|
|
def count_violations(self, violation_group: int):
|
|
"""Method to count all violations"""
|
|
all_violations = 0
|
|
if violation_group == -1:
|
|
for _, value in self.__violations.items():
|
|
all_violations += value[0]
|
|
else:
|
|
for _, value in self.__violations.items():
|
|
if value[2] == violation_group or (value[2] == 10 and violation_group == 5):
|
|
all_violations += value[0]
|
|
return all_violations
|
|
|
|
def count_deduction(self, violation_group: int, violation_amount=-1):
|
|
"""Method to count the deduction based on the group and amount"""
|
|
if violation_amount == -1:
|
|
violation_amount = self.count_violations(violation_group)
|
|
if violation_group == 0:
|
|
return 0
|
|
elif violation_group == 3:
|
|
if violation_amount > 0:
|
|
# The deduction for the author variable
|
|
return 2
|
|
else:
|
|
return 0
|
|
elif not self.__deduction:
|
|
return 0
|
|
elif violation_group == 5:
|
|
# Violation for docstrings with a max deduction
|
|
# Module and class docstring gets the half amount of deduction
|
|
return min(violation_amount*0.5 - self.__module_and_class_docstrings*0.25, 2)
|
|
else:
|
|
if violation_amount > 9:
|
|
return 0.5
|
|
elif violation_amount > 1:
|
|
return 0.25
|
|
else:
|
|
return 0
|