在 python 中导入钩子

Inf*_*tic 0 python import hook python-import

你好,我想在我的 python 副本中为自己实现一个小的库访问控制,我想知道是否有某种方法可以在导入系统上创建一个钩子,这样我的程序可以在每次导入时检查一个文件,如果该程序有权导入该库。我看到了一些有关 sys 库和 pep 302 ( https://www.python.org/dev/peps/pep-0302/ ) 的信息,但我不明白。

Vin*_*ent 5

您可以通过实现自己的自定义导入加载器对象来更改模块的导入。可以在此处找到文档中的起点:https : //docs.python.org/3/library/importlib.html

您需要做的是创建一个加载器,它将作用于您要检查的包,然后加载它们,或引发所需的异常。如果模块不在您的访问控制列表中,您应该返回 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
Run Code Online (Sandbox Code Playgroud)

请注意,加载程序在加载包时会从 sys.meta_path 中删除自身。这是为了避免无限循环,每次尝试“真正”加载模块时,它都会不断调用自己。