ID: py/import-of-mutable-attribute Kind: problem Security severity: Severity: warning Precision: medium Tags: - quality - reliability - correctness Query suites: - python-security-and-quality.qls
Click to see the query in the CodeQL repository
Explicitly importing an attribute from a module into the current namespace means that the value of that attribute will not be updated if the value in the original module changes.
This can mean that changes in global state are not observed locally, which may lead to inconsistencies and possible errors.
Recommendation¶
Instead of using from module import attr, simply import the module using import module and replace all uses of attr with module.attr.
Example¶
In the first of the two modules shown below, from sys import stdout is used to import the stdout attribute, rather than using import sys to import the module. Then stdout is used in the main() function.
from sys import stdout def main(): stdout.write("Hello World!")
In the second module, below, a function, redirect_to_file is defined to collect the output from sys.stdout and save it to a file. However, redirect_to_file will not work correctly when passed the main() function. This is because the main() function will not see the change to sys.stdout, as it uses its own version of stdout that was defined when the module was loaded.
import sys def redirect_to_file(function, args, kwargs, filename): with open(filename) as out: orig_stdout = sys.stdout sys.stdout = out try: function(*args, **kwargs) finally: sys.stdout = orig_stdout
The problem can be fixed by rewriting the first module to import the sys module and write to sys.stdout, as shown below.
import sys def main(): sys.stdout.write("Hello World!")
References¶
Python Language Reference: The import statement.
Python Tutorial: Modules.