Python eval hook时局部和全局变量报错NameError问题

一、eval语法

先介绍一下eval的语法:eval(expression, globals=None, locals=None)

它有三个参数,其中 expression 是一个字符串类型的表达式或代码对象,用于做运算;globals 与 locals 是可选参数,默认值是 None。

globals 用于指定运行时的全局命名空间,类型是字典,缺省时使用的是当前模块的内置命名空间。locals 指定运行时的局部命名空间,类型是字典,缺省时使用 globals 的值。两者都缺省时,则遵循 eval 函数执行时的作用域。值得注意的是,这两者不代表真正的命名空间,只在运算时起作用,运算后则销毁。

二、问题场景

由于业务需要,对eval进行了hook,hook方法大概如下:

import builtins
import os

def eval_hook(func):
    print("eval: ", func)
    def wrapper(*args, **kwargs):
        # 逻辑处理
        return func(*args, **kwargs)
    return wrapper

builtins.eval = eval_hook(builtins.eval)

在脚本中进行调用时会发现;

import test_hook
import os


data = "os.system('ls')"

print(data)
a = "(lambda x:data)(None)"

eval(data)  #正常调用

eval(a) #报错 NameError: name 'data' is not defined
eval(a, None)  #报错 NameError: name 'data' is not defined
eval(a, None, None)   #报错 NameError: name 'data' is not defined

为什么普通的变量data直接使用是正常的,但是在lambda表达式里就报错NameError了呢?

可以从locals()里查看一下当前模块的局部命名空间变量来确定:

在测试前打印locals()可以得到:

{'__name__': '__main__', 
 '__doc__': None, '__package__': None, 
 '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x100750ca0>,
'__spec__': None, '__annotations__': {}, 
'__builtins__': <module 'builtins' (built-in)>, 
'__file__': '/Users/test_str.py', '__cached__': None,
'test_hook': <module 'test_hook' from '/Users/test_hook.py'>,
'os': <module 'os' from '/Library/Developer/CommandLineTools/Library/Frameworks/Python3.framework/Versions/3.9/lib/python3.9/os.py'>,
'data': "os.system('ls')", 
'a': '(lambda x:data)(None)'}

可以看到是有data变量的,但是如果我们在hook 的处理函数eval_hook里打印一下,会发现locals()中确实不含data变量:

{
    'args': ('(lambda x:data)(None)',), 
 	'kwargs': {}, 
 	'func': <built-in function eval>
}

原因其实是变量的作用域问题,在hook的代码里应该在调用原函数前把locals()和globals()还原回hook前的值,这样才能确保局部和全局的变量都能被正确调用。

所以在hook的代码里应该添加下面的处理逻辑:

import builtins
import os
import inspect

def eval_hook(func):
    
    def wrapper(*args, **kwargs):

        frame = inspect.currentframe().f_back
        locals().update(frame.f_locals)
        globals().update(frame.f_globals)

        # 逻辑处理
        return func(*args, **kwargs)

    return wrapper

builtins.eval = eval_hook(builtins.eval)

其中inspect.currentframe().f_back就是eval_hook的上一个栈帧,也就是调用eval的栈。

  • 8
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
你在运行Python代码遇到了NameError: name 'Tuple' is not defined错误。这个错误通常发生在你使用了未导入的模块或未定义的变量类型。在你的情况下,问题可能是因为你没有正确导入所需的模块。 根据你给出的引用内容,我可以看到你的代码中使用了List和Tuple这两个未定义的变量类型。这表明你需要导入相应的模块才能正确使用它们。 要解决这个问题,你需要在你的代码中导入相应的模块。对于List和Tuple这两个变量类型,你需要导入typing模块。修改你的代码如下: ```python from typing import List, Tuple class Solution: def twoSum(self, nums: List[int], target: int) -> List[int]: pass ``` 通过这样的导入声明,你就可以在代码中正确使用List和Tuple了。 希望这个解决方案对你有帮助!如果你还有其他问题,请随提问。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [Python3函数声明报错NameError: name ‘List‘ is not defined](https://blog.csdn.net/mathemagics/article/details/122945469)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *3* [Python eval()报错NameError: name ‘null’ is not defined的解决方法详解](https://blog.csdn.net/weixin_53519100/article/details/113179713)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值