Detect definition-time mutating expressions

This commit is contained in:
2026-07-02 14:50:03 +02:00
parent 65c3a57052
commit 317788572e
2 changed files with 113 additions and 4 deletions
@@ -617,6 +617,73 @@ NODE_CLASS_MAPPINGS = {
self.assertEqual({}, result["nodes"]) self.assertEqual({}, result["nodes"])
self.assertEqual("no_static_nodes", result["pack"]["status"]) self.assertEqual("no_static_nodes", result["pack"]["status"])
def test_function_default_mutation_invalidates_static_env_value(self):
source = '''
INPUTS = {
"required": {
"image": ("IMAGE",),
},
}
def helper(x=INPUTS.clear()):
pass
class DefaultMutatedInputEnvNode:
RETURN_TYPES = ("IMAGE",)
@classmethod
def INPUT_TYPES(cls):
return INPUTS
NODE_CLASS_MAPPINGS = {
"DefaultMutatedInputEnvNode": DefaultMutatedInputEnvNode,
}
'''
result = self._extract_source(source, "default-mutated-input-env-pack")
self.assertEqual({}, result["nodes"])
self.assertEqual("no_static_nodes", result["pack"]["status"])
def test_function_decorator_mutation_invalidates_static_env_value(self):
source = '''
def decorator(value):
def wrap(fn):
return fn
return wrap
INPUTS = {
"required": {
"image": ("IMAGE",),
},
}
@decorator(INPUTS.clear())
def helper():
pass
class DecoratorMutatedInputEnvNode:
RETURN_TYPES = ("IMAGE",)
@classmethod
def INPUT_TYPES(cls):
return INPUTS
NODE_CLASS_MAPPINGS = {
"DecoratorMutatedInputEnvNode": DecoratorMutatedInputEnvNode,
}
'''
result = self._extract_source(source, "decorator-mutated-input-env-pack")
self.assertEqual({}, result["nodes"])
self.assertEqual("no_static_nodes", result["pack"]["status"])
def test_nested_mutable_env_literal_skips_static_node(self): def test_nested_mutable_env_literal_skips_static_node(self):
source = ''' source = '''
REQ = { REQ = {
@@ -835,6 +902,32 @@ NODE_CLASS_MAPPINGS = {
self.assertEqual({}, result["nodes"]) self.assertEqual({}, result["nodes"])
self.assertEqual("no_static_nodes", result["pack"]["status"]) self.assertEqual("no_static_nodes", result["pack"]["status"])
def test_function_default_mutation_to_return_types_skips_node(self):
source = '''
class DefaultMutatedReturnTypesNode:
RETURN_TYPES = ["IMAGE"]
def helper(self, x=RETURN_TYPES.clear()):
pass
@classmethod
def INPUT_TYPES(cls):
return {
"required": {
"image": ("IMAGE",),
},
}
NODE_CLASS_MAPPINGS = {
"DefaultMutatedReturnTypesNode": DefaultMutatedReturnTypesNode,
}
'''
result = self._extract_source(source, "default-mutated-return-types-pack")
self.assertEqual({}, result["nodes"])
self.assertEqual("no_static_nodes", result["pack"]["status"])
def test_return_types_alias_mutation_skips_node(self): def test_return_types_alias_mutation_skips_node(self):
source = ''' source = '''
class AliasMutatedReturnTypesNode: class AliasMutatedReturnTypesNode:
+20 -4
View File
@@ -181,17 +181,33 @@ def _mutating_call_target_names(stmt):
names = set() names = set()
class MutatingCallVisitor(ast.NodeVisitor): class MutatingCallVisitor(ast.NodeVisitor):
def _visit_function_definition_expressions(self, node):
for decorator in node.decorator_list:
self.visit(decorator)
self.visit(node.args)
if node.returns is not None:
self.visit(node.returns)
for type_param in getattr(node, "type_params", ()):
self.visit(type_param)
def visit_FunctionDef(self, node): def visit_FunctionDef(self, node):
return None self._visit_function_definition_expressions(node)
def visit_AsyncFunctionDef(self, node): def visit_AsyncFunctionDef(self, node):
return None self._visit_function_definition_expressions(node)
def visit_ClassDef(self, node): def visit_ClassDef(self, node):
return None for decorator in node.decorator_list:
self.visit(decorator)
for base in node.bases:
self.visit(base)
for keyword in node.keywords:
self.visit(keyword.value)
for type_param in getattr(node, "type_params", ()):
self.visit(type_param)
def visit_Lambda(self, node): def visit_Lambda(self, node):
return None self.visit(node.args)
def visit_Call(self, node): def visit_Call(self, node):
if isinstance(node.func, ast.Attribute) and node.func.attr in _MUTATING_METHODS: if isinstance(node.func, ast.Attribute) and node.func.attr in _MUTATING_METHODS: