"""
Module that contains the command line app.
Why does this file exist, and why not put this in __main__?
You might be tempted to import things from __main__ later, but that will cause
problems: the code will get executed twice:
- When you run `python -mvoca` python will execute
``__main__.py`` as a script. That means there won't be any
``voca.__main__`` in ``sys.modules``.
- When you import __main__ it will get executed again (as a module) because
there's no ``voca.__main__`` in ``sys.modules``.
Also see (1) from http://click.pocoo.org/5/setuptools/#setuptools-integration
"""
import sys
import types
import os
import click
import eliot
from voca import app
from voca import listen
from voca import manager
from voca import worker
from voca import log
from voca import config
CONTEXT_SETTINGS = {"auto_envvar_prefix": "VOCA"}
[docs]def log_cli_call(f):
def wrap(*args, **kwargs):
with eliot.start_action(
action_type="cli_command", pid=os.getpid(), argv=sys.argv
):
return log.log_call(f)(*args, **kwargs)
return wrap
@click.group(context_settings=CONTEXT_SETTINGS)
@click.option("--log/--no-log", "should_log", is_flag=True, default=True)
@click.pass_context
def cli(ctx, **kwargs):
ctx.obj = types.SimpleNamespace()
ctx.obj.kwargs = kwargs.copy()
app.main(**kwargs)
@cli.command("mic")
def _detect_mic(**kwargs):
del sys.argv[1]
import voca.mic
@cli.command("listen")
@click.option("--server", "-s", default="127.0.0.1")
@click.option("--port", "-p", default=8019, type=int)
@click.option("--device", "-d", default=7, type=int)
@click.option("--hypotheses", type=bool, default=True)
@click.option("--save-adaptation-state", type=bool)
@click.option("--send-adaptation-state", type=bool)
@click.option("--audio-gate", default=0, type=int)
@log_cli_call
def _listen(**kwargs):
content_type = "audio/x-raw, layout=(string)interleaved, rate=(int)16000, format=(string)S16LE, channels=(int)1"
path = "client/ws/speech"
listen.run(
args=types.SimpleNamespace(**kwargs), path=path, content_type=content_type
)
@cli.command("manage")
@click.option("--import-path", "-i", "module_names", multiple=True, default=None)
@click.option("--num-workers", type=int, default=5)
@click.pass_obj
@log_cli_call
def _manage(obj, **kwargs):
log_filename = config.get_config_dir() / "log.jsonl"
with open(log_filename, "w") as log_file:
eliot.add_destinations(log.json_to_file(log_file))
manager.main(**obj.kwargs, **kwargs)
@cli.command("worker")
@click.option("import_paths", "-i", multiple=True)
@click.option("--patch-caster", is_flag=True, default=False, envvar="VOCA_PATCH_CASTER")
@click.option(
"--backup-modules/--no-backup-modules",
"use_backup_modules",
is_flag=True,
default=True,
)
@click.pass_obj
@log_cli_call
def _worker(obj, patch_caster, **kwargs):
eliot.add_destinations(log.json_to_file(sys.stdout))
if patch_caster:
from voca import caster_adapter
caster_adapter.patch_all()
worker.main(**kwargs)