Source code for meerkat.tools.lazy_loader

# coding=utf-8
# Copyright 2020 The TF-Agents Authors.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

"""Lazy loader class."""

from __future__ import absolute_import, division, print_function

import importlib
import logging
import types

logger = logging.getLogger(__name__)


[docs]class LazyLoader(types.ModuleType): """Lazily import a module, mainly to avoid pulling in large dependencies. `contrib`, and `ffmpeg` are examples of modules that are large and not always needed, and this allows them to only be loaded when they are used. """ # The lint error here is incorrect. def __init__( self, local_name, parent_module_globals=None, name=None, warning=None, error=None, ): # pylint: disable=super-on-old-class if parent_module_globals is None: parent_module_globals = globals() self._local_name = local_name self._parent_module_globals = parent_module_globals self._warning = warning self._error = error if name is None: name = local_name super(LazyLoader, self).__init__(name) def _load(self): """Load the module and insert it into the parent's globals.""" # Import the target module and insert it into the parent's namespace try: module = importlib.import_module(self.__name__) except ImportError as e: raise ImportError(self._error) if self._error else e self._parent_module_globals[self._local_name] = module # Emit a warning if one was specified if self._warning: logger.warning(self._warning) # Make sure to only warn once. self._warning = None # Update this object's dict so that if someone keeps a reference to the # LazyLoader, lookups are efficient (__getattr__ is only called on lookups # that fail). self.__dict__.update(module.__dict__) return module def __getattr__(self, item): module = self._load() return getattr(module, item) def __dir__(self): module = self._load() return dir(module)