Updates
This commit is contained in:
306
ETB-API/venv/lib/python3.12/site-packages/celery/bin/base.py
Normal file
306
ETB-API/venv/lib/python3.12/site-packages/celery/bin/base.py
Normal file
@@ -0,0 +1,306 @@
|
||||
"""Click customizations for Celery."""
|
||||
import json
|
||||
import numbers
|
||||
from collections import OrderedDict
|
||||
from functools import update_wrapper
|
||||
from pprint import pformat
|
||||
from typing import Any
|
||||
|
||||
import click
|
||||
from click import Context, ParamType
|
||||
from kombu.utils.objects import cached_property
|
||||
|
||||
from celery._state import get_current_app
|
||||
from celery.signals import user_preload_options
|
||||
from celery.utils import text
|
||||
from celery.utils.log import mlevel
|
||||
from celery.utils.time import maybe_iso8601
|
||||
|
||||
try:
|
||||
from pygments import highlight
|
||||
from pygments.formatters import Terminal256Formatter
|
||||
from pygments.lexers import PythonLexer
|
||||
except ImportError:
|
||||
def highlight(s, *args, **kwargs):
|
||||
"""Place holder function in case pygments is missing."""
|
||||
return s
|
||||
LEXER = None
|
||||
FORMATTER = None
|
||||
else:
|
||||
LEXER = PythonLexer()
|
||||
FORMATTER = Terminal256Formatter()
|
||||
|
||||
|
||||
class CLIContext:
|
||||
"""Context Object for the CLI."""
|
||||
|
||||
def __init__(self, app, no_color, workdir, quiet=False):
|
||||
"""Initialize the CLI context."""
|
||||
self.app = app or get_current_app()
|
||||
self.no_color = no_color
|
||||
self.quiet = quiet
|
||||
self.workdir = workdir
|
||||
|
||||
@cached_property
|
||||
def OK(self):
|
||||
return self.style("OK", fg="green", bold=True)
|
||||
|
||||
@cached_property
|
||||
def ERROR(self):
|
||||
return self.style("ERROR", fg="red", bold=True)
|
||||
|
||||
def style(self, message=None, **kwargs):
|
||||
if self.no_color:
|
||||
return message
|
||||
else:
|
||||
return click.style(message, **kwargs)
|
||||
|
||||
def secho(self, message=None, **kwargs):
|
||||
if self.no_color:
|
||||
kwargs['color'] = False
|
||||
click.echo(message, **kwargs)
|
||||
else:
|
||||
click.secho(message, **kwargs)
|
||||
|
||||
def echo(self, message=None, **kwargs):
|
||||
if self.no_color:
|
||||
kwargs['color'] = False
|
||||
click.echo(message, **kwargs)
|
||||
else:
|
||||
click.echo(message, **kwargs)
|
||||
|
||||
def error(self, message=None, **kwargs):
|
||||
kwargs['err'] = True
|
||||
if self.no_color:
|
||||
kwargs['color'] = False
|
||||
click.echo(message, **kwargs)
|
||||
else:
|
||||
click.secho(message, **kwargs)
|
||||
|
||||
def pretty(self, n):
|
||||
if isinstance(n, list):
|
||||
return self.OK, self.pretty_list(n)
|
||||
if isinstance(n, dict):
|
||||
if 'ok' in n or 'error' in n:
|
||||
return self.pretty_dict_ok_error(n)
|
||||
else:
|
||||
s = json.dumps(n, sort_keys=True, indent=4)
|
||||
if not self.no_color:
|
||||
s = highlight(s, LEXER, FORMATTER)
|
||||
return self.OK, s
|
||||
if isinstance(n, str):
|
||||
return self.OK, n
|
||||
return self.OK, pformat(n)
|
||||
|
||||
def pretty_list(self, n):
|
||||
if not n:
|
||||
return '- empty -'
|
||||
return '\n'.join(
|
||||
f'{self.style("*", fg="white")} {item}' for item in n
|
||||
)
|
||||
|
||||
def pretty_dict_ok_error(self, n):
|
||||
try:
|
||||
return (self.OK,
|
||||
text.indent(self.pretty(n['ok'])[1], 4))
|
||||
except KeyError:
|
||||
pass
|
||||
return (self.ERROR,
|
||||
text.indent(self.pretty(n['error'])[1], 4))
|
||||
|
||||
def say_chat(self, direction, title, body='', show_body=False):
|
||||
if direction == '<-' and self.quiet:
|
||||
return
|
||||
dirstr = not self.quiet and f'{self.style(direction, fg="white", bold=True)} ' or ''
|
||||
self.echo(f'{dirstr} {title}')
|
||||
if body and show_body:
|
||||
self.echo(body)
|
||||
|
||||
|
||||
def handle_preload_options(f):
|
||||
"""Extract preload options and return a wrapped callable."""
|
||||
def caller(ctx, *args, **kwargs):
|
||||
app = ctx.obj.app
|
||||
|
||||
preload_options = [o.name for o in app.user_options.get('preload', [])]
|
||||
|
||||
if preload_options:
|
||||
user_options = {
|
||||
preload_option: kwargs[preload_option]
|
||||
for preload_option in preload_options
|
||||
}
|
||||
|
||||
user_preload_options.send(sender=f, app=app, options=user_options)
|
||||
|
||||
return f(ctx, *args, **kwargs)
|
||||
|
||||
return update_wrapper(caller, f)
|
||||
|
||||
|
||||
class CeleryOption(click.Option):
|
||||
"""Customized option for Celery."""
|
||||
|
||||
def get_default(self, ctx, *args, **kwargs):
|
||||
if self.default_value_from_context:
|
||||
self.default = ctx.obj[self.default_value_from_context]
|
||||
return super().get_default(ctx, *args, **kwargs)
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
"""Initialize a Celery option."""
|
||||
self.help_group = kwargs.pop('help_group', None)
|
||||
self.default_value_from_context = kwargs.pop('default_value_from_context', None)
|
||||
super().__init__(*args, **kwargs)
|
||||
|
||||
|
||||
class CeleryCommand(click.Command):
|
||||
"""Customized command for Celery."""
|
||||
|
||||
def format_options(self, ctx, formatter):
|
||||
"""Write all the options into the formatter if they exist."""
|
||||
opts = OrderedDict()
|
||||
for param in self.get_params(ctx):
|
||||
rv = param.get_help_record(ctx)
|
||||
if rv is not None:
|
||||
if hasattr(param, 'help_group') and param.help_group:
|
||||
opts.setdefault(str(param.help_group), []).append(rv)
|
||||
else:
|
||||
opts.setdefault('Options', []).append(rv)
|
||||
|
||||
for name, opts_group in opts.items():
|
||||
with formatter.section(name):
|
||||
formatter.write_dl(opts_group)
|
||||
|
||||
|
||||
class DaemonOption(CeleryOption):
|
||||
"""Common daemonization option"""
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
super().__init__(args,
|
||||
help_group=kwargs.pop("help_group", "Daemonization Options"),
|
||||
callback=kwargs.pop("callback", self.daemon_setting),
|
||||
**kwargs)
|
||||
|
||||
def daemon_setting(self, ctx: Context, opt: CeleryOption, value: Any) -> Any:
|
||||
"""
|
||||
Try to fetch daemonization option from applications settings.
|
||||
Use the daemon command name as prefix (eg. `worker` -> `worker_pidfile`)
|
||||
"""
|
||||
return value or getattr(ctx.obj.app.conf, f"{ctx.command.name}_{self.name}", None)
|
||||
|
||||
|
||||
class CeleryDaemonCommand(CeleryCommand):
|
||||
"""Daemon commands."""
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
"""Initialize a Celery command with common daemon options."""
|
||||
super().__init__(*args, **kwargs)
|
||||
self.params.extend((
|
||||
DaemonOption("--logfile", "-f", help="Log destination; defaults to stderr"),
|
||||
DaemonOption("--pidfile", help="PID file path; defaults to no PID file"),
|
||||
DaemonOption("--uid", help="Drops privileges to this user ID"),
|
||||
DaemonOption("--gid", help="Drops privileges to this group ID"),
|
||||
DaemonOption("--umask", help="Create files and directories with this umask"),
|
||||
DaemonOption("--executable", help="Override path to the Python executable"),
|
||||
))
|
||||
|
||||
|
||||
class CommaSeparatedList(ParamType):
|
||||
"""Comma separated list argument."""
|
||||
|
||||
name = "comma separated list"
|
||||
|
||||
def convert(self, value, param, ctx):
|
||||
return text.str_to_list(value)
|
||||
|
||||
|
||||
class JsonArray(ParamType):
|
||||
"""JSON formatted array argument."""
|
||||
|
||||
name = "json array"
|
||||
|
||||
def convert(self, value, param, ctx):
|
||||
if isinstance(value, list):
|
||||
return value
|
||||
|
||||
try:
|
||||
v = json.loads(value)
|
||||
except ValueError as e:
|
||||
self.fail(str(e))
|
||||
|
||||
if not isinstance(v, list):
|
||||
self.fail(f"{value} was not an array")
|
||||
|
||||
return v
|
||||
|
||||
|
||||
class JsonObject(ParamType):
|
||||
"""JSON formatted object argument."""
|
||||
|
||||
name = "json object"
|
||||
|
||||
def convert(self, value, param, ctx):
|
||||
if isinstance(value, dict):
|
||||
return value
|
||||
|
||||
try:
|
||||
v = json.loads(value)
|
||||
except ValueError as e:
|
||||
self.fail(str(e))
|
||||
|
||||
if not isinstance(v, dict):
|
||||
self.fail(f"{value} was not an object")
|
||||
|
||||
return v
|
||||
|
||||
|
||||
class ISO8601DateTime(ParamType):
|
||||
"""ISO 8601 Date Time argument."""
|
||||
|
||||
name = "iso-86091"
|
||||
|
||||
def convert(self, value, param, ctx):
|
||||
try:
|
||||
return maybe_iso8601(value)
|
||||
except (TypeError, ValueError) as e:
|
||||
self.fail(e)
|
||||
|
||||
|
||||
class ISO8601DateTimeOrFloat(ParamType):
|
||||
"""ISO 8601 Date Time or float argument."""
|
||||
|
||||
name = "iso-86091 or float"
|
||||
|
||||
def convert(self, value, param, ctx):
|
||||
try:
|
||||
return float(value)
|
||||
except (TypeError, ValueError):
|
||||
pass
|
||||
|
||||
try:
|
||||
return maybe_iso8601(value)
|
||||
except (TypeError, ValueError) as e:
|
||||
self.fail(e)
|
||||
|
||||
|
||||
class LogLevel(click.Choice):
|
||||
"""Log level option."""
|
||||
|
||||
def __init__(self):
|
||||
"""Initialize the log level option with the relevant choices."""
|
||||
super().__init__(('DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL', 'FATAL'))
|
||||
|
||||
def convert(self, value, param, ctx):
|
||||
if isinstance(value, numbers.Integral):
|
||||
return value
|
||||
|
||||
value = value.upper()
|
||||
value = super().convert(value, param, ctx)
|
||||
return mlevel(value)
|
||||
|
||||
|
||||
JSON_ARRAY = JsonArray()
|
||||
JSON_OBJECT = JsonObject()
|
||||
ISO8601 = ISO8601DateTime()
|
||||
ISO8601_OR_FLOAT = ISO8601DateTimeOrFloat()
|
||||
LOG_LEVEL = LogLevel()
|
||||
COMMA_SEPARATED_LIST = CommaSeparatedList()
|
||||
Reference in New Issue
Block a user