Как я могу использовать click.MultiCommand
вместе с командами, определенными как методы класса?
Я пытаюсь настроить систему плагинов для конвертеров, где пользователи библиотеки могут предоставить свои собственные конвертеры.Для этой системы я настраиваю CLI, как показано ниже:
$ myproj convert {converter} INPUT OUTPUT {ARGS}
Каждый конвертер имеет свой собственный класс и все наследуются от BaseConverter
.В BaseConverter
самая простая команда Click, которая принимает только ввод и вывод.
Для преобразователей, которым не нужно больше, им не нужно переопределять этот метод.Если конвертеру требуется нечто большее или требуется предоставить дополнительную документацию, его необходимо переопределить.
При использовании приведенного ниже кода я получаю следующую ошибку при попытке использовать cli:
TypeError: cli () отсутствует 1 обязательный позиционный аргумент: 'cls'
conversion/
├── __init__.py
└── backends/
├── __init__.py
├── base.py
├── bar.py
├── baz.py
└── foo.py
# cli.py
from pydoc import locate
import click
from proj.conversion import AVAILABLE_CONVERTERS
class ConversionCLI(click.MultiCommand):
def list_commands(self, ctx):
return sorted(list(AVAILABLE_CONVERTERS))
def get_command(self, ctx, name):
return locate(AVAILABLE_CONVERTERS[name] + '.cli')
@click.command(cls=ConversionCLI)
def convert():
"""Convert files using specified converter"""
pass
# conversion/__init__.py
from django.conf import settings
AVAILABLE_CONVERTERS = {
'bar': 'conversion.backends.bar.BarConverter',
'baz': 'conversion.backends.baz.BazConverter',
'foo': 'conversion.backends.foo.FooConverter',
}
extra_converters = getattr(settings, 'CONVERTERS', {})
AVAILABLE_CONVERTERS.update(extra_converters)
# conversion/backends/base.py
import click
class BaseConverter():
@classmethod
def convert(cls, infile, outfile):
raise NotImplementedError
@classmethod
@click.command()
@click.argument('infile')
@click.argument('outfile')
def cli(cls, infile, outfile):
return cls.convert(infile, outfile)
# conversion/backends/bar.py
from proj.conversion.base import BaseConverter
class BarConverter(BaseConverter):
@classmethod
def convert(cls, infile, outfile):
# do stuff
# conversion/backends/foo.py
import click
from proj.conversion.base import BaseConverter
class FooConverter(BaseConverter):
@classmethod
def convert(cls, infile, outfile, extra_arg):
# do stuff
@classmethod
@click.command()
@click.argument('infile')
@click.argument('outfile')
@click.argument('extra-arg')
def cli(cls, infile, outfile, extra_arg):
return cls.convert(infile, outfile, extra_arg)