In this post from 2 years ago I talked about how to create a function from a string in Python. As I explain there, given that eval() only works with expressions and exec() returns nothing, to "return" the function we had to do something a bit tricky with an assignment, particularly tricky due to the limitations of how exec() can interact with variables in its surrounding scope. I'll replicate here the code sample from that post:
fn_st = (
"def multiply(num1, num2):\n"
" print('multiplying numbers')\n"
" return num1 * num2\n"
)
def create_function(fn_st, fn_name):
d = {}
exec(fn_st + "\n" + "d['fn'] = " + fn_name)
return d["fn"]
new_fn = create_function(fn_st, "multiply")
print("created")
print(new_fn(2, 3))
# multiplying numbers
# 6
As I explain in that post code compiled/executed by exec() or eval()) can read variables from the surrounding scope, but if it writes to them or creates new variables, those changes won't be available in the surrounding scope. To circumvent that we set the function as an entry in a dictionary, rather than directly in a variable, so with the extra indirection level it works. After writing that post I had found somewhere another technique a bit cleaner, let's see:
def create_function4(fn_st, fn_name):
exec(fn_st, scope:={})
return scope[fn_name]
print("started option4")
new_fn = create_function4(fn_st, "multiply")
print("created")
print(new_fn(2, 3))
# multiplying numbers
# 6
We can pass to exec() dictionaries representing the global and local variables to use in the block to execute. So in this case we pass a dictionary, that will be used as both local and global scope for the block, so the function that we define in exec gets defined in that dictionary, and we can retrieve the function from the dictionary in the outer scope. This technique corresponds to this in the documentation:
Pass an explicit locals dictionary if you need to see effects of the code on locals after function exec() returns.
On the other hand, the "unable to modify variables in the outer scope" behavior that we experience when we don't explicitly provide the globals and locals arguments corresponds to this in the documentation
In an optimized scope (including functions, generators, and coroutines), each call to locals() instead returns a fresh dictionary containing the current bindings of the function’s local variables and any nonlocal cell references. In this case, name binding changes made via the returned dict are not written back to the corresponding local variables or nonlocal cell references, and assigning, reassigning, or deleting local variables and nonlocal cell references does not affect the contents of previously returned dictionaries.
This brings me back to my post from late December about f_locals
FrameType.f_locals now returns a write-through proxy to the frame’s local and locally referenced nonlocal variables in these scopes.
This means that we can write the above function also this way, passing the f_locals of the current frame:
def create_function5(fn_st, fn_name):
# I can not pass f_locals as globals: a TypeError: exec() globals must be a dict, not FrameLocalsProxy
# but I can pass it as locals:
#exec(compile(fn_st, "", "exec"), locals=inspect.stack()[0].frame.f_locals)
exec(fn_st, locals=inspect.stack()[0].frame.f_locals)
return locals()[fn_name]
print("started option5")
new_fn = create_function5(fn_st, "multiply")
print("created")
print(new_fn(2, 3))
# multiplying numbers
# 6
Notice that we have to pass f_locals as the locals parameter rather than as globals, cause passing it as globals we get an TypeError: exec() globals must be a dict, not FrameLocalsProxy
For this "function creation" case where we just want to retrieve the function, passing as parameter f_locals or passing a new dictionary does not make a particular difference (indeed it's way more verbose), but for cases where we want to modify local variables of the surrounding scope. f_locals is a game changer!
def another_test():
print("another_test")
a1 = "aaa"
exec("a1 += 'bbb'", locals=inspect.stack()[0].frame.f_locals)
print(f"a1: {a1}")
another_test()
# a1: aaabbb