mirror of
https://github.com/VinciGit00/Scrapegraph-ai.git
synced 2026-06-25 21:11:11 +08:00
feat(webdriver-backend): add dynamic import scripts from module and file
This commit is contained in:
parent
2f4fd45700
commit
db2234bf5d
67
scrapegraphai/utils/sys_dynamic_import.py
Normal file
67
scrapegraphai/utils/sys_dynamic_import.py
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
"""high-level module for dynamic importing of python modules at runtime
|
||||||
|
|
||||||
|
source code inspired by https://gist.github.com/DiTo97/46f4b733396b8d7a8f1d4d22db902cfc
|
||||||
|
"""
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import typing
|
||||||
|
|
||||||
|
|
||||||
|
if typing.TYPE_CHECKING:
|
||||||
|
import types
|
||||||
|
|
||||||
|
|
||||||
|
def srcfile_import(modpath: str, modname: str) -> "types.ModuleType":
|
||||||
|
"""imports a python module from its srcfile
|
||||||
|
|
||||||
|
Args:
|
||||||
|
modpath: The srcfile absolute path
|
||||||
|
modname: The module name in the scope
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
The imported module
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ImportError: If the module cannot be imported from the srcfile
|
||||||
|
"""
|
||||||
|
import importlib.util # noqa: F401
|
||||||
|
|
||||||
|
#
|
||||||
|
spec = importlib.util.spec_from_file_location(modname, modpath)
|
||||||
|
|
||||||
|
if spec is None:
|
||||||
|
message = f"missing spec for module at {modpath}"
|
||||||
|
raise ImportError(message)
|
||||||
|
|
||||||
|
if spec.loader is None:
|
||||||
|
message = f"missing spec loader for module at {modpath}"
|
||||||
|
raise ImportError(message)
|
||||||
|
|
||||||
|
module = importlib.util.module_from_spec(spec)
|
||||||
|
|
||||||
|
# adds the module to the global scope
|
||||||
|
sys.modules[modname] = module
|
||||||
|
|
||||||
|
spec.loader.exec_module(module)
|
||||||
|
|
||||||
|
return module
|
||||||
|
|
||||||
|
|
||||||
|
def dynamic_import(modname: str, message: str = "") -> None:
|
||||||
|
"""imports a python module at runtime
|
||||||
|
|
||||||
|
Args:
|
||||||
|
modname: The module name in the scope
|
||||||
|
message: The display message in case of error
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ImportError: If the module cannot be imported at runtime
|
||||||
|
"""
|
||||||
|
if modname not in sys.modules:
|
||||||
|
try:
|
||||||
|
import importlib # noqa: F401
|
||||||
|
|
||||||
|
module = importlib.import_module(modname)
|
||||||
|
sys.modules[modname] = module
|
||||||
|
except ImportError as x:
|
||||||
|
raise ImportError(message) from x
|
||||||
94
tests/utils/test_sys_dynamic_import.py
Normal file
94
tests/utils/test_sys_dynamic_import.py
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from scrapegraphai.utils.sys_dynamic_import import dynamic_import, srcfile_import
|
||||||
|
|
||||||
|
|
||||||
|
def _create_sample_file(filepath: str, content: str):
|
||||||
|
"""creates a sample file at some path with some content"""
|
||||||
|
with open(filepath, "w", encoding="utf-8") as f:
|
||||||
|
f.write(content)
|
||||||
|
|
||||||
|
|
||||||
|
def _delete_sample_file(filepath: str):
|
||||||
|
"""deletes a sample file at some path"""
|
||||||
|
if os.path.exists(filepath):
|
||||||
|
os.remove(filepath)
|
||||||
|
|
||||||
|
|
||||||
|
def test_srcfile_import_success():
|
||||||
|
modpath = "example1.py"
|
||||||
|
modname = "example1"
|
||||||
|
|
||||||
|
_create_sample_file(modpath, "def foo(): return 'bar'")
|
||||||
|
|
||||||
|
module = srcfile_import(modpath, modname)
|
||||||
|
|
||||||
|
assert hasattr(module, "foo")
|
||||||
|
assert module.foo() == "bar"
|
||||||
|
assert modname in sys.modules
|
||||||
|
|
||||||
|
_delete_sample_file(modpath)
|
||||||
|
|
||||||
|
|
||||||
|
def test_srcfile_import_missing_spec():
|
||||||
|
modpath = "nonexistent1.py"
|
||||||
|
modname = "nonexistent1"
|
||||||
|
|
||||||
|
with pytest.raises(FileNotFoundError):
|
||||||
|
srcfile_import(modpath, modname)
|
||||||
|
|
||||||
|
|
||||||
|
def test_srcfile_import_missing_spec_loader(mocker):
|
||||||
|
modpath = "example2.py"
|
||||||
|
modname = "example2"
|
||||||
|
|
||||||
|
_create_sample_file(modpath, "")
|
||||||
|
|
||||||
|
mock_spec = mocker.Mock(loader=None)
|
||||||
|
|
||||||
|
mocker.patch("importlib.util.spec_from_file_location", return_value=mock_spec)
|
||||||
|
|
||||||
|
with pytest.raises(ImportError) as error_info:
|
||||||
|
srcfile_import(modpath, modname)
|
||||||
|
|
||||||
|
assert "missing spec loader for module at" in str(error_info.value)
|
||||||
|
|
||||||
|
_delete_sample_file(modpath)
|
||||||
|
|
||||||
|
|
||||||
|
def test_dynamic_import_success():
|
||||||
|
print(sys.modules)
|
||||||
|
modname = "playwright"
|
||||||
|
assert modname not in sys.modules
|
||||||
|
|
||||||
|
dynamic_import(modname)
|
||||||
|
|
||||||
|
assert modname in sys.modules
|
||||||
|
|
||||||
|
import playwright # noqa: F401
|
||||||
|
|
||||||
|
|
||||||
|
def test_dynamic_import_module_already_imported():
|
||||||
|
modname = "json"
|
||||||
|
|
||||||
|
import json # noqa: F401
|
||||||
|
|
||||||
|
assert modname in sys.modules
|
||||||
|
|
||||||
|
dynamic_import(modname)
|
||||||
|
|
||||||
|
assert modname in sys.modules
|
||||||
|
|
||||||
|
|
||||||
|
def test_dynamic_import_import_error_with_custom_message():
|
||||||
|
modname = "nonexistent2"
|
||||||
|
message = "could not import module"
|
||||||
|
|
||||||
|
with pytest.raises(ImportError) as error_info:
|
||||||
|
dynamic_import(modname, message=message)
|
||||||
|
|
||||||
|
assert str(error_info.value) == message
|
||||||
|
assert modname not in sys.modules
|
||||||
Loading…
Reference in New Issue
Block a user