Сделать виртуальный пакет доступным через sys.modules - PullRequest
11 голосов
/ 15 декабря 2008

Скажите, у меня есть пакет "mylibrary".

Я хочу сделать «mylibrary.config» доступным для импорта, либо как динамически создаваемый модуль, либо как модуль, импортированный из совершенно другого места, которое затем будет «подключено» внутри пространства имен «mylibrary».

То есть, я делаю:

import sys, types
sys.modules['mylibrary.config'] = types.ModuleType('config')

Учитывая эту настройку:

>>> import mylibrary.config    # -> works

>>> from mylibrary import config
<type 'exceptions.ImportError'>: cannot import name config

Даже незнакомец:

>>> import mylibrary.config as X
<type 'exceptions.ImportError'>: cannot import name config

Похоже, что использование прямого импорта работает, а другие формы - нет. Можно ли заставить их работать так же?

Ответы [ 3 ]

14 голосов
/ 15 декабря 2008

Вам необходимо подключить модуль не только к sys.modules, но и к его родительскому модулю:

>>> import sys,types,xml
>>> xml.config = sys.modules['xml.config'] = types.ModuleType('xml.config')
>>> import xml.config
>>> from xml import config
>>> from xml import config as x
>>> x
<module 'xml.config' (built-in)>
2 голосов
/ 15 декабря 2008

А также следующее:

import sys, types
config = types.ModuleType('config')
sys.modules['mylibrary.config'] = config

Вам также нужно сделать:

import mylibrary
mylibrary.config = config
1 голос
/ 15 декабря 2008

Вы можете попробовать что-то вроде этого:

class VirtualModule(object):
  def __init__(self, modname, subModules):
    try:
      import sys
      self._mod = __import__(modname)
      sys.modules[modname] = self
      __import__(modname)
      self._modname = modname
      self._subModules = subModules
    except ImportError, err:
      pass  # please signal error in some useful way :-)
  def __repr__(self):
    return "Virtual module for " + self._modname
  def __getattr__(self, attrname):
    if attrname in self._subModules.keys():
      import sys
      __import__(self._subModules[attrname])
      return sys.modules[self._subModules[attrname]]
    else:
      return self._mod.__dict__[attrname]


VirtualModule('mylibrary', {'config': 'actual_module_for_config'})

import mylibrary
mylibrary.config
mylibrary.some_function
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...