httpie-cli/httpie/plugins/manager.py

122 lines
4.0 KiB
Python
Raw Normal View History

import sys
import os
import warnings
from itertools import groupby
2019-09-01 11:45:47 +02:00
from operator import attrgetter
from typing import Dict, List, Type, Iterator, Iterable, Optional, ContextManager
from pathlib import Path
from contextlib import contextmanager, nullcontext
2019-08-31 18:33:54 +02:00
from ..compat import importlib_metadata, find_entry_points, get_dist_name
2019-09-01 11:13:45 +02:00
from ..utils import repr_dict, get_site_paths
from . import AuthPlugin, ConverterPlugin, FormatterPlugin, TransportPlugin
from .base import BasePlugin
2013-09-21 23:46:15 +02:00
ENTRY_POINT_CLASSES = {
'httpie.plugins.auth.v1': AuthPlugin,
'httpie.plugins.converter.v1': ConverterPlugin,
'httpie.plugins.formatter.v1': FormatterPlugin,
'httpie.plugins.transport.v1': TransportPlugin
}
ENTRY_POINT_NAMES = list(ENTRY_POINT_CLASSES.keys())
2013-09-21 23:46:15 +02:00
@contextmanager
def _load_directories(site_dirs: Iterable[Path]) -> Iterator[None]:
plugin_dirs = [
os.fspath(site_dir)
for site_dir in site_dirs
]
sys.path.extend(plugin_dirs)
try:
yield
finally:
for plugin_dir in plugin_dirs:
sys.path.remove(plugin_dir)
def enable_plugins(plugins_dir: Optional[Path]) -> ContextManager[None]:
if plugins_dir is None:
return nullcontext()
else:
return _load_directories(get_site_paths(plugins_dir))
class PluginManager(list):
2019-08-31 18:33:54 +02:00
def register(self, *plugins: Type[BasePlugin]):
for plugin in plugins:
2019-09-01 11:13:45 +02:00
self.append(plugin)
2013-09-21 23:46:15 +02:00
2019-08-31 18:35:24 +02:00
def unregister(self, plugin: Type[BasePlugin]):
2019-09-01 11:13:45 +02:00
self.remove(plugin)
def filter(self, by_type=Type[BasePlugin]):
return [plugin for plugin in self if issubclass(plugin, by_type)]
def iter_entry_points(self, directory: Optional[Path] = None):
with enable_plugins(directory):
eps = importlib_metadata.entry_points()
for entry_point_name in ENTRY_POINT_NAMES:
yield from find_entry_points(eps, group=entry_point_name)
def load_installed_plugins(self, directory: Optional[Path] = None):
for entry_point in self.iter_entry_points(directory):
plugin_name = get_dist_name(entry_point)
try:
plugin = entry_point.load()
except BaseException as exc:
warnings.warn(
f'While loading "{plugin_name}", an error occurred: {exc}\n'
f'For uninstallations, please use either "httpie plugins uninstall {plugin_name}" '
f'or "pip uninstall {plugin_name}" (depending on how you installed it in the first '
'place).'
)
continue
plugin.package_name = plugin_name
self.register(plugin)
# Auth
2019-08-31 18:33:54 +02:00
def get_auth_plugins(self) -> List[Type[AuthPlugin]]:
2019-09-01 11:13:45 +02:00
return self.filter(AuthPlugin)
2013-09-21 23:46:15 +02:00
2019-08-31 18:33:54 +02:00
def get_auth_plugin_mapping(self) -> Dict[str, Type[AuthPlugin]]:
return {
plugin.auth_type: plugin for plugin in self.get_auth_plugins()
}
2013-09-21 23:46:15 +02:00
2019-08-31 18:33:54 +02:00
def get_auth_plugin(self, auth_type: str) -> Type[AuthPlugin]:
2013-09-21 23:46:15 +02:00
return self.get_auth_plugin_mapping()[auth_type]
# Output processing
2019-08-31 18:33:54 +02:00
def get_formatters(self) -> List[Type[FormatterPlugin]]:
2019-09-01 11:13:45 +02:00
return self.filter(FormatterPlugin)
2019-08-31 18:33:54 +02:00
def get_formatters_grouped(self) -> Dict[str, List[Type[FormatterPlugin]]]:
2019-09-01 11:45:47 +02:00
return {
group_name: list(group)
for group_name, group
in groupby(self.get_formatters(), key=attrgetter('group_name'))
}
2019-08-31 18:33:54 +02:00
def get_converters(self) -> List[Type[ConverterPlugin]]:
2019-09-01 11:13:45 +02:00
return self.filter(ConverterPlugin)
# Adapters
2019-08-31 18:33:54 +02:00
def get_transport_plugins(self) -> List[Type[TransportPlugin]]:
2019-09-01 11:13:45 +02:00
return self.filter(TransportPlugin)
def __str__(self):
return repr_dict({
'adapters': self.get_transport_plugins(),
'auth': self.get_auth_plugins(),
'converters': self.get_converters(),
'formatters': self.get_formatters(),
})
2019-09-01 11:13:45 +02:00
def __repr__(self):
return f'<{type(self).__name__} {self}>'