在 python 中导入挂钩
Import hooks in python
你好,我想在我的 python 副本中为自己实现一个小的库访问控制,我想知道是否有某种方法可以在导入系统上创建一个挂钩,以便我的程序可以检查如果该程序有权导入该库,则每次导入时都会生成一个文件。
我看到了一些关于 sys 库和 pep 302 (https://www.python.org/dev/peps/pep-0302/) 的东西,但我不明白。
您可以通过实现您自己的自定义导入加载器对象来更改模块的导入。可以在此处找到文档的起点:https://docs.python.org/3/library/importlib.html
您需要做的是创建一个加载程序,它将作用于您要检查的包,然后加载它们,或引发所需的异常。对于不在您的访问控制列表中的模块,您应该 return None,这会使导入机器正常加载它们。我已经创建了此类功能的最小示例,您可以从中开始并扩展以构建所需的功能。
import sys
import importlib
class ImportInterceptor(importlib.abc.Loader):
def __init__(self, package_permissions):
self.package_permissions = package_permissions
def find_module(self, fullname, path=None):
if fullname in self.package_permissions:
if self.package_permissions[fullname]:
return self
else:
raise ImportError("Package import was not allowed")
def load_module(self, fullname):
sys.meta_path = [x for x in sys.meta_path[1:] if x is not self]
module = importlib.import_module(fullname)
sys.meta_path = [self] + sys.meta_path
return module
if not hasattr(sys,'frozen'):
sys.meta_path = [ImportInterceptor({'textwrap': True, 'Pathlib': False})] + sys.meta_path
import textwrap
print(textwrap.dedent(' test'))
# Works fine
from pathlib import Path
# Raises exception
请注意,加载程序在加载包时会从 sys.meta_path 中删除自身。这是为了避免无限循环,每次尝试加载模块时它都会不断调用自己 "for real".
你好,我想在我的 python 副本中为自己实现一个小的库访问控制,我想知道是否有某种方法可以在导入系统上创建一个挂钩,以便我的程序可以检查如果该程序有权导入该库,则每次导入时都会生成一个文件。 我看到了一些关于 sys 库和 pep 302 (https://www.python.org/dev/peps/pep-0302/) 的东西,但我不明白。
您可以通过实现您自己的自定义导入加载器对象来更改模块的导入。可以在此处找到文档的起点:https://docs.python.org/3/library/importlib.html
您需要做的是创建一个加载程序,它将作用于您要检查的包,然后加载它们,或引发所需的异常。对于不在您的访问控制列表中的模块,您应该 return None,这会使导入机器正常加载它们。我已经创建了此类功能的最小示例,您可以从中开始并扩展以构建所需的功能。
import sys
import importlib
class ImportInterceptor(importlib.abc.Loader):
def __init__(self, package_permissions):
self.package_permissions = package_permissions
def find_module(self, fullname, path=None):
if fullname in self.package_permissions:
if self.package_permissions[fullname]:
return self
else:
raise ImportError("Package import was not allowed")
def load_module(self, fullname):
sys.meta_path = [x for x in sys.meta_path[1:] if x is not self]
module = importlib.import_module(fullname)
sys.meta_path = [self] + sys.meta_path
return module
if not hasattr(sys,'frozen'):
sys.meta_path = [ImportInterceptor({'textwrap': True, 'Pathlib': False})] + sys.meta_path
import textwrap
print(textwrap.dedent(' test'))
# Works fine
from pathlib import Path
# Raises exception
请注意,加载程序在加载包时会从 sys.meta_path 中删除自身。这是为了避免无限循环,每次尝试加载模块时它都会不断调用自己 "for real".