Skip to content
Snippets Groups Projects
Commit c19c45be authored by Nicholas Robinson's avatar Nicholas Robinson
Browse files

centralize class & interface logic

parent a41f5afb
No related branches found
No related tags found
No related merge requests found
...@@ -4,8 +4,61 @@ from Environment import Env ...@@ -4,8 +4,61 @@ from Environment import Env
# types: class , interface # types: class , interface
class ClassInterNode(ASTNode):
def checkHierarchy(self):
# 3. An interface must not be repeated in an implements clause
if self.superInter:
unique = []
for inter in self.superInter:
if inter.name in unique:
raise Exception("ERROR: Class/Interface '{}' implements duplicate interfaces '{}'".format(self.name, inter.name))
unique.append(inter.name)
# 7. A class or interface must not declare two methods with the same signature (name and parameter types).
unique = []
for method in self.methods:
key = (method.name, method.paramTypes)
if key in unique:
raise Exception("ERROR: Class/Interface '{}' declares 2 methods with the same signature '{}'".format(self.name, key[0]))
unique.append(key)
# 6. The hierarchy must be acyclic
# 9. A class or interface must not contain (declare or inherit) two methods with the same signature but different return types
# 11. A nonstatic method must not replace a static method
# 13. A protected method must not replace a public method
# 14. A method must not replace a final method
return self.getContains([])
def getContains(self, hierarchy):
# check if not acyclic
if self.name in hierarchy:
raise Exception("ERROR: The hierarchy is not acyclic '{}', saw '{}'".format(hierarchy, self.name))
hierarchy.append(self.name)
# get contains
contains = self.methods
for inter in self.superInter:
superContains = inter.getContains(hierarchy)
for con in superContains:
conOverwritten = False
for method in self.methods:
if (method == con):
# cannot have same signiture but different return types
if (method.methodType != con.methodType):
raise Exception("ERROR: Class '{}' contains 2 methods '{}' with the same signature but different return types".format(self.name, method.name))
# protected must not replace public
if 'protected' in method.mods and 'public' in con.mods:
raise Exception("ERROR: Protected method '{}' in class '{}' replaces public method '{}' in class {}".format(method.name, self.name, con.name, inter.name))
conOverwritten = True
break
if not conOverwritten:
contains.append(con)
return contains
# class # class
class ClassNode(ASTNode): class ClassNode(ClassInterNode):
# always list all fields in the init method to show the class structure # always list all fields in the init method to show the class structure
def __init__(self, parseTree): def __init__(self, parseTree):
self.parseTree = parseTree self.parseTree = parseTree
...@@ -88,17 +141,7 @@ class ClassNode(ASTNode): ...@@ -88,17 +141,7 @@ class ClassNode(ASTNode):
for inter in self.superInter: for inter in self.superInter:
if not isinstance(inter, InterNode): if not isinstance(inter, InterNode):
raise Exception("ERROR: Class '{}' implements non-interface '{}'".format(self.name, inter.name)) raise Exception("ERROR: Class '{}' implements non-interface '{}'".format(self.name, inter.name))
if inter.name in unique:
raise Exception("ERROR: Class '{}' implements duplicate interfaces '{}'".format(self.name, inter.name))
unique.append(inter.name) unique.append(inter.name)
# 7. A class or interface must not declare two methods with the same signature (name and parameter types).
unique = []
for method in self.methods:
key = (method.name, method.paramTypes)
if key in unique:
raise Exception("ERROR: Class '{}' declares 2 methods with the same signature '{}'".format(self.name, key[0]))
unique.append(key)
# 8. A class must not declare two constructors with the same parameter types # 8. A class must not declare two constructors with the same parameter types
unique = [] unique = []
...@@ -108,12 +151,9 @@ class ClassNode(ASTNode): ...@@ -108,12 +151,9 @@ class ClassNode(ASTNode):
raise Exception("ERROR: Class '{}' declares 2 constructors with the same parameter types".format(self.name)) raise Exception("ERROR: Class '{}' declares 2 constructors with the same parameter types".format(self.name))
unique.append(key) unique.append(key)
# 6. The hierarchy must be acyclic # centralized point for overlapping class & interface logic
# 9. A class or interface must not contain (declare or inherit) two methods with the same signature but different return types contains = super().checkHierarchy()
# 11. A nonstatic method must not replace a static method
# 13. A protected method must not replace a public method
# 14. A method must not replace a final method
contains = self.getContains([])
# 10. A class that contains (declares or inherits) any abstract methods must be abstract. # 10. A class that contains (declares or inherits) any abstract methods must be abstract.
for con in contains: for con in contains:
if 'abstract' in con.mods and 'abstract' not in self.mods: if 'abstract' in con.mods and 'abstract' not in self.mods:
...@@ -121,20 +161,16 @@ class ClassNode(ASTNode): ...@@ -121,20 +161,16 @@ class ClassNode(ASTNode):
# hierarchy: string[] # hierarchy: string[]
def getContains(self, hierarchy): def getContains(self, hierarchy):
# TODO: move this to a more centralized spot for both ClassNode and InterNode? # centralized logic
# TODO: need to check all of the implemented interfaces as well *** contains = super().getContains(hierarchy)
# check if not acyclic # get contains from extends class
if self.name in hierarchy:
raise Exception("ERROR: The hierarchy is not acyclic '{}', saw '{}'".format(hierarchy, self.name))
hierarchy.append(self.name)
# get contains
contains = self.methods
if self.superClass: if self.superClass:
addToContains = []
superContains = self.superClass.getContains(hierarchy) superContains = self.superClass.getContains(hierarchy)
for con in superContains: for con in superContains:
conOverwritten = False conOverwritten = False
for method in self.methods: for method in contains:
if (method == con): if (method == con):
# cannot have same signiture but different return types # cannot have same signiture but different return types
if (method.methodType != con.methodType): if (method.methodType != con.methodType):
...@@ -151,7 +187,8 @@ class ClassNode(ASTNode): ...@@ -151,7 +187,8 @@ class ClassNode(ASTNode):
conOverwritten = True conOverwritten = True
break break
if not conOverwritten: if not conOverwritten:
contains.append(con) addToContains.append(con)
contains += addToContains
return contains return contains
def getConstructor(self, argTypes): def getConstructor(self, argTypes):
...@@ -161,7 +198,7 @@ class ClassNode(ASTNode): ...@@ -161,7 +198,7 @@ class ClassNode(ASTNode):
##################################################################### #####################################################################
# interface # interface
class InterNode(ASTNode): class InterNode(ClassInterNode):
# always list all fields in the init method to show the class structure # always list all fields in the init method to show the class structure
def __init__(self, parseTree): def __init__(self, parseTree):
self.parseTree = parseTree self.parseTree = parseTree
...@@ -214,41 +251,10 @@ class InterNode(ASTNode): ...@@ -214,41 +251,10 @@ class InterNode(ASTNode):
raise Exception("ERROR: Interface '{}' extends duplicate interfaces '{}'".format(self.name, inter.name)) raise Exception("ERROR: Interface '{}' extends duplicate interfaces '{}'".format(self.name, inter.name))
unique.append(inter.name) unique.append(inter.name)
# 7. A class or interface must not declare two methods with the same signature (name and parameter types). # centralized point for overlapping class & interface logic
unique = [] contains = super().checkHierarchy()
for method in self.methods:
key = (method.name, method.paramTypes)
if key in unique:
raise Exception("ERROR: Interface '{}' declares 2 methods with the same signature '{}'".format(self.name, key[0]))
unique.append(key)
# 6. The hierarchy must be acyclic
# 9. A class or interface must not contain (declare or inherit) two methods with the same signature but different return types
# 13. A protected method must not replace a public method
contains = self.getContains([])
# hierarchy: string[] # hierarchy: string[]
def getContains(self, hierarchy): def getContains(self, hierarchy):
# check if not acyclic # centralized logic
if self.name in hierarchy: return super().getContains(hierarchy)
raise Exception("ERROR: The hierarchy is not acyclic '{}'".format(hierarchy))
hierarchy.append(self.name)
# get contains
contains = self.methods
for inter in self.superInter:
superContains = inter.getContains(hierarchy)
for con in superContains:
conOverwritten = False
for method in self.methods:
if (method == con):
# cannot have same signiture but different return types
if (method.methodType != con.methodType):
raise Exception("ERROR: Interface '{}' contains 2 methods '{}' with the same signature but different return types".format(self.name, method.name))
# protected must not replace public
if 'protected' in method.mods and 'public' in con.mods:
raise Exception("ERROR: Protected method '{}' in class '{}' replaces public method '{}' in class {}".format(method.name, self.name, con.name, inter.name))
conOverwritten = True
break
if not conOverwritten:
contains.append(con)
return contains
\ No newline at end of file
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment