Question

I'm trying to execute a number of functions using eval(), and I need to create some kind of environment for them to run. It is said in documentation that you can pass globals as a second parameter to eval().

But it seems to not work in my case. Here's the simpified example (I tried two approaches, declaring variable global and using globals(), and both do not work):

File script.py:

import test

global test_variable
test_variable = 'test_value'
g = globals()
g['test_variable'] = 'test_value'
eval('test.my_func()', g)

File test.py:

def my_func():
    global test_variable
    print repr(test_variable)

And I'm getting:

NameError: global name 'test_variable' is not defined.

What should I do to pass that test_variable into my_func()? Assuming I can't pass it as a parameter.

Was it helpful?

Solution

test_variable should be global in test.py. You're getting a name error because you're trying to declare a variable global that doesn't yet exist.

So your my_test.py file should be like this:

test_variable = None

def my_func():
    print test_variable

And running this from the command prompt:

>>> import my_test
>>> eval('my_test.my_func()')
None
>>> my_test.test_variable = 'hello'
>>> my_test.test_variable
'hello'
>>> eval('my_test.my_func()')
hello

Generally it's bad form to use eval() and globals, so make sure you know what your doing.

OTHER TIPS

Please correct me Python experts if I am wrong. I am also learning Python. The following is my current understanding of why the NameError exception was thrown.

In Python, you cannot create a variable that can be access across modules without specifying the module name (i.e. to access the global variable test in module mod1 you need to use mod1.test when you in module mod2). The scope of the global variable is pretty much limited to the module itself.

Thus when you have following in test.py:

def my_func():
    global test_variable
    print repr(test_variable)

The test_variable here refers to test.test_variable (i.e. test_variable in the test module namespace).

So setting test_variable in script.py will put the variable in the __main__ namespace (__main__ because this is the top-level module/script you provided to the Python interpreter to execute). Thus, this test_variable will be in a different namespace and not in the test module namespace where it is required to be. Hence, Python generates a NameError because it cannot find the variable after searching the test module global namespace and built-in namespace (local function namespace is skipped because of the global statement).

Therefore, for eval to work, you need to set test_variable in the test module namespace in script.py:

import test
test.test_variable = 'test_value'
eval('test.my_func()')

For more details about Python’s scope and namespaces see: http://docs.python.org/tutorial/classes.html#python-scopes-and-name-spaces

Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top