diff --git a/CHANGELOG.md b/CHANGELOG.md index f21963b37..7615cdd54 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -12,6 +12,7 @@ and adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). - Added `include_frames` filter to `snapshot` - Added `exclude_vars` filter to `snapshot` - Added new `python_ta.debug` module with an `SnapshotTracer` context manager for generating memory models +- Added `z3` option to `inconsistent-or-missing-returns`, `redundant-assignment`, and `possibly-undefined` checkers to only check for feasible code blocks based on edge z3 constraints - Included the name of redundant variable in `E9959 redundant-assignment` message - Update to pylint v3.3 and and astroid v3.3. This added support for Python 3.13 and dropped support for Python 3.8. - Added a STRICT_NUMERIC_TYPES configuration to `python_ta.contracts` allowing to enable/disable stricter type checking of numeric types diff --git a/python_ta/__init__.py b/python_ta/__init__.py index 6bbecd56a..82abd35c6 100644 --- a/python_ta/__init__.py +++ b/python_ta/__init__.py @@ -136,7 +136,9 @@ def _check( global PYLINT_PATCHED if not PYLINT_PATCHED: - patch_all(messages_config) # Monkeypatch pylint (override certain methods) + patch_all( + messages_config, linter.config.z3 + ) # Monkeypatch pylint (override certain methods) PYLINT_PATCHED = True # Try to check file, issue error message for invalid files. diff --git a/python_ta/cfg/graph.py b/python_ta/cfg/graph.py index d074ebc32..f442b6bb8 100644 --- a/python_ta/cfg/graph.py +++ b/python_ta/cfg/graph.py @@ -166,11 +166,16 @@ def multiple_link_or_merge(self, source: CFGBlock, targets: List[CFGBlock]) -> N for target in targets: self.link(source, target) - def get_blocks(self) -> Generator[CFGBlock, None, None]: - """Generate a sequence of all blocks in this graph.""" - yield from self._get_blocks(self.start, set()) + def get_blocks(self, only_feasible: bool = False) -> Generator[CFGBlock, None, None]: + """Generate a sequence of all blocks in this graph. - def _get_blocks(self, block: CFGBlock, visited: Set[int]) -> Generator[CFGBlock, None, None]: + When only_feasible is True, only generate blocks feasible from start based on edge z3 constraints. + """ + yield from self._get_blocks(self.start, set(), only_feasible) + + def _get_blocks( + self, block: CFGBlock, visited: Set[int], only_feasible: bool + ) -> Generator[CFGBlock, None, None]: if block.id in visited: return @@ -178,20 +183,27 @@ def _get_blocks(self, block: CFGBlock, visited: Set[int]) -> Generator[CFGBlock, visited.add(block.id) for edge in block.successors: - yield from self._get_blocks(edge.target, visited) + if not only_feasible or edge.is_feasible: + yield from self._get_blocks(edge.target, visited, only_feasible) - def get_blocks_postorder(self) -> Generator[CFGBlock, None, None]: + def get_blocks_postorder(self, only_feasible: bool = False) -> Generator[CFGBlock, None, None]: """Return the sequence of all blocks in this graph in the order of - a post-order traversal.""" - yield from self._get_blocks_postorder(self.start, set()) + a post-order traversal. + + When only_feasible is True, only generate blocks feasible from start based on edge z3 constraints. + """ + yield from self._get_blocks_postorder(self.start, set(), only_feasible) - def _get_blocks_postorder(self, block: CFGBlock, visited) -> Generator[CFGBlock, None, None]: + def _get_blocks_postorder( + self, block: CFGBlock, visited: Set[int], only_feasible: bool + ) -> Generator[CFGBlock, None, None]: if block.id in visited: return visited.add(block.id) for succ in block.successors: - yield from self._get_blocks_postorder(succ.target, visited) + if not only_feasible or succ.is_feasible: + yield from self._get_blocks_postorder(succ.target, visited, only_feasible) yield block @@ -353,6 +365,10 @@ def jump(self) -> Optional[NodeNG]: if len(self.statements) > 0: return self.statements[-1] + @property + def is_feasible(self) -> bool: + return any(edge.is_feasible for edge in self.predecessors) + def is_jump(self) -> bool: """Returns True if the block has a statement that branches the control flow (ex: `break`)""" diff --git a/python_ta/checkers/inconsistent_or_missing_returns_checker.py b/python_ta/checkers/inconsistent_or_missing_returns_checker.py index 527a41475..21c3f2541 100644 --- a/python_ta/checkers/inconsistent_or_missing_returns_checker.py +++ b/python_ta/checkers/inconsistent_or_missing_returns_checker.py @@ -25,6 +25,17 @@ class InconsistentReturnChecker(BaseChecker): "Used when a function does not have a return statement and whose return type is not None", ), } + options = ( + ( + "z3", + { + "default": False, + "type": "yn", + "metavar": "", + "help": "Use Z3 to restrict control flow checks to paths that are logically feasible.", + }, + ), + ) def __init__(self, linter: Optional[PyLinter] = None) -> None: super().__init__(linter=linter) @@ -71,6 +82,15 @@ def _check_return_statements(self, node) -> None: if has_return_annotation or has_return_value: for block, statement in return_statements.items(): if statement is None: + # ignore unfeasible edges for missing return if z3 option is on + if self.linter.config.z3 and ( + not block.is_feasible + or not any( + edge.is_feasible for edge in block.successors if edge.target is end + ) + ): + continue + # For rendering purpose: # line: the line where the error occurs, used to calculate indentation # end_line: the line to insert the error message diff --git a/python_ta/checkers/possibly_undefined_checker.py b/python_ta/checkers/possibly_undefined_checker.py index c9e576667..65f23b462 100644 --- a/python_ta/checkers/possibly_undefined_checker.py +++ b/python_ta/checkers/possibly_undefined_checker.py @@ -26,6 +26,17 @@ class PossiblyUndefinedChecker(BaseChecker): "Reported when a statement uses a variable that might not be assigned.", ) } + options = ( + ( + "z3", + { + "default": False, + "type": "yn", + "metavar": "", + "help": "Use Z3 to restrict control flow checks to paths that are logically feasible.", + }, + ), + ) def __init__(self, linter=None) -> None: super().__init__(linter=linter) @@ -56,7 +67,7 @@ def _analyze(self, node: Union[nodes.Module, nodes.FunctionDef]) -> None: out_facts = {} cfg = ControlFlowGraph() cfg.start = node.cfg_block - blocks = list(cfg.get_blocks_postorder()) + blocks = list(cfg.get_blocks_postorder(only_feasible=self.linter.config.z3)) blocks.reverse() all_assigns = self._get_assigns(node) @@ -66,7 +77,11 @@ def _analyze(self, node: Union[nodes.Module, nodes.FunctionDef]) -> None: worklist = blocks while len(worklist) != 0: b = worklist.pop() - outs = [out_facts[p.source] for p in b.predecessors if p.source in out_facts] + outs = [ + out_facts[p.source] + for p in b.predecessors + if p.source in out_facts and (not self.linter.config.z3 or p.is_feasible) + ] if outs == []: in_facts = set() else: @@ -74,7 +89,13 @@ def _analyze(self, node: Union[nodes.Module, nodes.FunctionDef]) -> None: temp = self._transfer(b, in_facts, all_assigns) if temp != out_facts[b]: out_facts[b] = temp - worklist.extend([succ.target for succ in b.successors]) + worklist.extend( + [ + succ.target + for succ in b.successors + if not self.linter.config.z3 or succ.is_feasible + ] + ) def _transfer(self, block: CFGBlock, in_facts: set[str], local_vars: set[str]) -> set[str]: gen = in_facts.copy() diff --git a/python_ta/checkers/redundant_assignment_checker.py b/python_ta/checkers/redundant_assignment_checker.py index 7e3207b25..8f8f1157d 100644 --- a/python_ta/checkers/redundant_assignment_checker.py +++ b/python_ta/checkers/redundant_assignment_checker.py @@ -41,6 +41,17 @@ class RedundantAssignmentChecker(BaseChecker): " You can remove the assignment(s) without changing the behaviour of this code.", ) } + options = ( + ( + "z3", + { + "default": False, + "type": "yn", + "metavar": "", + "help": "Use Z3 to restrict control flow checks to paths that are logically feasible.", + }, + ), + ) def __init__(self, linter=None) -> None: super().__init__(linter=linter) @@ -88,7 +99,7 @@ def _analyze(self, node: Union[nodes.Module, nodes.FunctionDef]) -> None: out_facts = {} cfg = ControlFlowGraph() cfg.start = node.cfg_block - worklist = list(cfg.get_blocks_postorder()) + worklist = list(cfg.get_blocks_postorder(only_feasible=self.linter.config.z3)) worklist.reverse() all_assigns = self._get_assigns(node) @@ -97,7 +108,11 @@ def _analyze(self, node: Union[nodes.Module, nodes.FunctionDef]) -> None: while len(worklist) != 0: b = worklist.pop() - outs = [out_facts[p.target] for p in b.successors if p.target in out_facts] + outs = [ + out_facts[p.target] + for p in b.successors + if p.target in out_facts and (not self.linter.config.z3 or p.is_feasible) + ] if outs == []: in_facts = set() else: @@ -105,7 +120,13 @@ def _analyze(self, node: Union[nodes.Module, nodes.FunctionDef]) -> None: temp = self._transfer(b, in_facts) if b in out_facts and temp != out_facts[b]: out_facts[b] = temp - worklist.extend([pred.source for pred in b.predecessors if pred.source.reachable]) + worklist.extend( + [ + pred.source + for pred in b.predecessors + if pred.source.reachable and (not self.linter.config.z3 or pred.is_feasible) + ] + ) def _transfer(self, block: CFGBlock, out_facts: set[str]) -> set[str]: gen = out_facts.copy() diff --git a/python_ta/patches/__init__.py b/python_ta/patches/__init__.py index cdeea5f75..a61749154 100644 --- a/python_ta/patches/__init__.py +++ b/python_ta/patches/__init__.py @@ -7,9 +7,9 @@ from .transforms import patch_ast_transforms -def patch_all(messages_config: dict): +def patch_all(messages_config: dict, z3: bool): """Execute all patches defined in this module.""" patch_checkers() - patch_ast_transforms() + patch_ast_transforms(z3) patch_messages() patch_error_messages(messages_config) diff --git a/python_ta/patches/transforms.py b/python_ta/patches/transforms.py index 97b4f135e..10a5f2607 100644 --- a/python_ta/patches/transforms.py +++ b/python_ta/patches/transforms.py @@ -4,15 +4,18 @@ from pylint.lint import PyLinter from ..cfg.visitor import CFGVisitor +from ..transforms.z3_visitor import Z3Visitor -def patch_ast_transforms(): +def patch_ast_transforms(z3: bool): old_get_ast = PyLinter.get_ast def new_get_ast(self, filepath, modname, data): ast = old_get_ast(self, filepath, modname, data) if ast is not None: try: + if z3: + ast = Z3Visitor().visitor.visit(ast) ast.accept(CFGVisitor()) except: pass diff --git a/tests/test_custom_checkers/test_inconsistent_returns.py b/tests/test_custom_checkers/test_inconsistent_returns.py index af169c70b..0adef7e78 100644 --- a/tests/test_custom_checkers/test_inconsistent_returns.py +++ b/tests/test_custom_checkers/test_inconsistent_returns.py @@ -6,6 +6,7 @@ from python_ta.checkers.inconsistent_or_missing_returns_checker import ( InconsistentReturnChecker, ) +from python_ta.transforms.z3_visitor import Z3Visitor class TestInconsistentReturnChecker(pylint.testutils.CheckerTestCase): @@ -153,3 +154,86 @@ def func(): ignore_position=True, ): self.checker.visit_functiondef(func_node) + + +class TestInconsistentReturnCheckerZ3Option(pylint.testutils.CheckerTestCase): + CHECKER_CLASS = InconsistentReturnChecker + CONFIG = {"z3": True} + + def test_z3_unfeasible_inconsistent_return(self): + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x > 5 + ''' + if x < 0: + return + return x + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = next(mod.nodes_of_class(nodes.FunctionDef)) + inconsistent_return_node, _ = mod.nodes_of_class(nodes.Return) + + with self.assertAddsMessages( + pylint.testutils.MessageTest( + msg_id="inconsistent-returns", + node=inconsistent_return_node, + ), + ignore_position=True, + ): + self.checker.visit_functiondef(func_node) + + def test_z3_partially_feasible_inconsistent_return(self): + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x > 5 + ''' + if x < 0: + print(x) + return + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = next(mod.nodes_of_class(nodes.FunctionDef)) + inconsistent_return_node = next(mod.nodes_of_class(nodes.Return)) + + with self.assertAddsMessages( + pylint.testutils.MessageTest( + msg_id="inconsistent-returns", + node=inconsistent_return_node, + ), + ignore_position=True, + ): + self.checker.visit_functiondef(func_node) + + def test_z3_feasible_inconsistent_return(self): + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x > 5 + ''' + if x > 0: + return + return x + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = next(mod.nodes_of_class(nodes.FunctionDef)) + inconsistent_return_node, _ = mod.nodes_of_class(nodes.Return) + + with self.assertAddsMessages( + pylint.testutils.MessageTest( + msg_id="inconsistent-returns", + node=inconsistent_return_node, + ), + ignore_position=True, + ): + self.checker.visit_functiondef(func_node) diff --git a/tests/test_custom_checkers/test_missing_return_statements.py b/tests/test_custom_checkers/test_missing_return_statements.py index 81c475db5..e935e0b19 100644 --- a/tests/test_custom_checkers/test_missing_return_statements.py +++ b/tests/test_custom_checkers/test_missing_return_statements.py @@ -6,6 +6,7 @@ from python_ta.checkers.inconsistent_or_missing_returns_checker import ( InconsistentReturnChecker, ) +from python_ta.transforms.z3_visitor import Z3Visitor class TestMissingReturnChecker(pylint.testutils.CheckerTestCase): @@ -348,3 +349,101 @@ def division_missing_return(x, y) -> int: ), ): self.checker.visit_functiondef(func_node) + + +class TestMissingReturnCheckerZ3Option(pylint.testutils.CheckerTestCase): + CHECKER_CLASS = InconsistentReturnChecker + CONFIG = {"z3": True} + + def test_z3_unfeasible_missing_return(self): + src = """ + def unfeasible_missing_return(x: str) -> str: + ''' + Preconditions: + - x[0:2] == "ab" + ''' + if x in "cd": + print("empty string") + else: + return x + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = next(mod.nodes_of_class(nodes.FunctionDef)) + + with self.assertNoMessages(): + self.checker.visit_functiondef(func_node) + + def test_z3_partially_feasible_missing_return(self): + src = """ + def feasible_missing_return(x: int) -> int: + ''' + Preconditions: + - x in [1, 2, 3, 4, 5] + ''' + if x > 5: + print(x) + print(x) + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = next(mod.nodes_of_class(nodes.FunctionDef)) + + with self.assertAddsMessages( + pylint.testutils.MessageTest( + msg_id="missing-return-statement", + node=func_node, + args="feasible_missing_return", + ), + ignore_position=True, + ): + self.checker.visit_functiondef(func_node) + + def test_z3_feasible_missing_return(self): + src = """ + def feasible_missing_return(x: bool, y: int) -> int: + ''' + Preconditions: + - x + - y > 5 + ''' + while not x: + print(1) + y += 1 + if y > 10: + return y + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = next(mod.nodes_of_class(nodes.FunctionDef)) + + with self.assertAddsMessages( + pylint.testutils.MessageTest( + msg_id="missing-return-statement", + node=func_node, + args="feasible_missing_return", + ), + ignore_position=True, + ): + self.checker.visit_functiondef(func_node) + + def test_z3_return_not_missing_by_precondition(self): + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x > 0 + ''' + if x > 0: + return x + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = next(mod.nodes_of_class(nodes.FunctionDef)) + + with self.assertNoMessages(): + self.checker.visit_functiondef(func_node) diff --git a/tests/test_custom_checkers/test_possibly_undefined_checker.py b/tests/test_custom_checkers/test_possibly_undefined_checker.py index 0466ee984..344b3dcde 100644 --- a/tests/test_custom_checkers/test_possibly_undefined_checker.py +++ b/tests/test_custom_checkers/test_possibly_undefined_checker.py @@ -4,6 +4,7 @@ from python_ta.cfg import CFGVisitor from python_ta.checkers.possibly_undefined_checker import PossiblyUndefinedChecker +from python_ta.transforms.z3_visitor import Z3Visitor class TestPossiblyUndefinedChecker(pylint.testutils.CheckerTestCase): @@ -469,3 +470,146 @@ def g(a: int) -> int: with self.assertNoMessages(): self.checker.visit_functiondef(func_node) + + +class TestPossiblyUndefinedCheckerZ3Option(pylint.testutils.CheckerTestCase): + CHECKER_CLASS = PossiblyUndefinedChecker + CONFIG = {"z3": True} + + def test_z3_unfeasible_branch(self) -> None: + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x > 5 + ''' + if x > 3: + a = 10 + else: + pass + return a + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = mod.body[0] + *_, name_node = mod.nodes_of_class(nodes.Name) + + self.checker.visit_functiondef(func_node) + with self.assertNoMessages(): + self.checker.visit_name(name_node) + + def test_z3_unfeasible_assignment(self) -> None: + src = """ + def func(x: str) -> int: + ''' + Preconditions: + - x in ["a", "b", "c"] + ''' + if x == "a": + a = 5 + elif x == "d": + a = 10 + return a + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = mod.body[0] + *_, name_node = mod.nodes_of_class(nodes.Name) + + self.checker.visit_functiondef(func_node) + with self.assertAddsMessages( + pylint.testutils.MessageTest(msg_id="possibly-undefined", node=name_node), + ignore_position=True, + ): + self.checker.visit_name(name_node) + + def test_z3_unfeasible_deletion(self) -> None: + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x > 5 + ''' + a = 0 + if x < 0: + del a + return a + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = mod.body[0] + *_, name_node = mod.nodes_of_class(nodes.Name) + + self.checker.visit_functiondef(func_node) + with self.assertNoMessages(): + self.checker.visit_name(name_node) + + def test_z3_unfeasible_variable_use(self) -> None: + src = """ + def func(x: int, y: int) -> int: + ''' + Preconditions: + - x > 10 + ''' + if y > 10: + a = 0 + if x < 10: + return a + return 0 + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = mod.body[0] + *_, name_node = mod.nodes_of_class(nodes.Name) + + self.checker.visit_functiondef(func_node) + with self.assertNoMessages(): + self.checker.visit_name(name_node) + + def test_z3_variable_defined_by_precondition(self) -> None: + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x == 0 + ''' + if x == 0: + a = 10 + return a + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = mod.body[0] + *_, name_node = mod.nodes_of_class(nodes.Name) + + self.checker.visit_functiondef(func_node) + with self.assertNoMessages(): + self.checker.visit_name(name_node) + + def test_z3_partially_unfeasible_assignment(self) -> None: + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x in {1, 2} + ''' + if x == 3: + a = 5 + else: + a = 10 + return a + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + func_node = mod.body[0] + *_, name_node = mod.nodes_of_class(nodes.Name) + + self.checker.visit_functiondef(func_node) + with self.assertNoMessages(): + self.checker.visit_name(name_node) diff --git a/tests/test_custom_checkers/test_redundant_assignment_checker.py b/tests/test_custom_checkers/test_redundant_assignment_checker.py index fa6997616..0268e7272 100644 --- a/tests/test_custom_checkers/test_redundant_assignment_checker.py +++ b/tests/test_custom_checkers/test_redundant_assignment_checker.py @@ -4,6 +4,7 @@ from python_ta.cfg import CFGVisitor from python_ta.checkers.redundant_assignment_checker import RedundantAssignmentChecker +from python_ta.transforms.z3_visitor import Z3Visitor class TestRedundantAssignmentChecker(pylint.testutils.CheckerTestCase): @@ -334,3 +335,57 @@ def test_multiple_target_assign_one_variable_redundant(self): ignore_position=True, ): self.checker.visit_assign(assign_node) + + +class TestRedundantAssignmentCheckerZ3Option(pylint.testutils.CheckerTestCase): + CHECKER_CLASS = RedundantAssignmentChecker + CONFIG = {"z3": True} + + def test_z3_unfeasible_variable_use(self): + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x > 10 + ''' + a = 10 + if x < 5: + print(a) + a = 20 + return a + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + assign_1, *_ = mod.nodes_of_class(nodes.Assign) + + self.checker.visit_functiondef(mod.body[0]) + with self.assertAddsMessages( + pylint.testutils.MessageTest(msg_id="redundant-assignment", node=assign_1, args="a"), + ignore_position=True, + ): + self.checker.visit_assign(assign_1) + + def test_z3_redundant_reassignment_by_precondition(self): + src = """ + def func(x: int) -> int: + ''' + Preconditions: + - x > 10 + ''' + a = 10 + if x > 0: + a = 20 + return a + """ + z3v = Z3Visitor() + mod = z3v.visitor.visit(astroid.parse(src)) + mod.accept(CFGVisitor()) + assign_1, _ = mod.nodes_of_class(nodes.Assign) + + self.checker.visit_functiondef(mod.body[0]) + with self.assertAddsMessages( + pylint.testutils.MessageTest(msg_id="redundant-assignment", node=assign_1, args="a"), + ignore_position=True, + ): + self.checker.visit_assign(assign_1)