В настоящее время я использую fab -f check_remote.py func:"arg1","arg2"...
для запуска удаленного пула.
Теперь мне нужно отправить bool arg, но True/False - это строка arg, как установить его как тип bool?
В настоящее время я использую fab -f check_remote.py func:"arg1","arg2"...
для запуска удаленного пула.
Теперь мне нужно отправить bool arg, но True/False - это строка arg, как установить его как тип bool?
Как упоминалось в docs, все аргументы оказываются в виде строк. Простейшая вещь здесь просто проверит аргумент:
def myfunc(arg1, arg2):
arg1 = (arg1 == 'True')
Скобки не требуются, но помогают с читабельностью.
Изменить. По-видимому, я фактически не пробовал свой предыдущий ответ; обновлено. (Два года спустя.)
Я использую это:
from distutils.util import strtobool
def func(arg1="default", arg2=False):
if arg2:
arg2 = bool(strtobool(arg2))
До сих пор работает для меня. он будет анализировать значения (игнорируя случай):
'y', 'yes', 't', 'true', 'on', '1'
'n', 'no', 'f', 'false', 'off', '0'
strtobool возвращает 0 или 1, поэтому bool необходим для преобразования в True/False boolean.
Для полноты, здесь strtobool
реализация:
def strtobool (val):
"""Convert a string representation of truth to true (1) or false (0).
True values are 'y', 'yes', 't', 'true', 'on', and '1'; false values
are 'n', 'no', 'f', 'false', 'off', and '0'. Raises ValueError if
'val' is anything else.
"""
val = val.lower()
if val in ('y', 'yes', 't', 'true', 'on', '1'):
return 1
elif val in ('n', 'no', 'f', 'false', 'off', '0'):
return 0
else:
raise ValueError("invalid truth value %r" % (val,))
Немного лучшая версия (спасибо за комментарии mVChr)
from distutils.util import strtobool
def _prep_bool_arg(arg):
return bool(strtobool(str(arg)))
def func(arg1="default", arg2=False):
arg2 = _prep_bool_arg(arg2)
Я бы использовал функцию:
def booleanize(value):
"""Return value as a boolean."""
true_values = ("yes", "true", "1")
false_values = ("no", "false", "0")
if isinstance(value, bool):
return value
if value.lower() in true_values:
return True
elif value.lower() in false_values:
return False
raise TypeError("Cannot booleanize ambiguous value '%s'" % value)
Затем в задаче:
@task
def mytask(arg):
arg = booleanize(arg)
Если рассматриваемый func использует "if argN:" вместо "если argN is True:", чтобы проверить, истинно ли логическое значение, вы можете использовать "для False и" что угодно" для True.
Смотрите также: http://docs.python.org/library/stdtypes.html#truth-value-testing
Если вы используете шаблон последовательно ( "false", "true" является логическим) для всех ваших задач, вы можете просто обернуть задачу ткани и применить ее на всех
Вы можете использовать этот пакет (написанный мной): https://pypi.python.org/pypi/boolfab/
Вот (по существу) источник:
from fabric.api import task as _task
def fix_boolean(f):
def fix_bool(value):
if isinstance(value, basestring):
if value.lower() == 'false':
return False
if value.lower() == 'true':
return True
return value
@wraps(f)
def wrapper(*args, **kwargs):
args_ = [fix_bool(arg) for arg in args]
kwargs_ = {k: fix_bool(v) for k,v in kwargs.iteritems()}
return f(*args_, **kwargs_)
return wrapper
def task(f):
return _task(fix_boolean(f))
Чтобы он стал:
@task
def my_task(flag_a, flag_b, flag_c)
if flag_a:
....
не загрязняя каждую задачу "booleanizing" args.
Лучшим способом было бы использовать ast.literal_eval
:
from ast import literal_eval
def my_task(flag):
if isinstance(flag, basestring): # also supports calling from other tasks
flag = literal_eval(flag)
Хотя это не учитывает такие значения, как "да" или "нет", оно немного чище и безопаснее, чем eval
...
Ответы Крейга и Ари приведут к Истинному значению, если пользователь пройдет "False" (ответ Ари более ясен об этом)
Если вы используете eval(), строки "True" и "False" будут оценивать их правильные логические значения, но если вы используете значения по умолчанию, вам нужно убедиться, что они являются строками, а не Booleans.
def myfunc(arg1="True", arg2=False):
arg1 = eval(arg1)
arg2 = eval(arg2) #error
В моих файлах я просто делаю:
TRUTHY = [True, 1, '1', 'true', 't', 'yes', 'y']
@task
def my_task(my_arg=True):
if my_arg in TRUTHY:
# do stuff
else:
# do other stuff
Конечно, это означает, что любое значение не в TRUTHY эффективно False
, но до сих пор мне не требовалось ничего более сложного.
Я решил это с помощью декораторов. Мне нравится гибкость и объяснение, которое вы получаете от использования декоратора.
Вот мясо кода:
import ast
from fabric import utils
from fabric.api import task
from functools import wraps
def params(*types, **kwtypes):
def decorator(function):
@wraps(function)
def wrapper(*args, **kwargs):
new_args = ()
for index, arg in enumerate(args):
new_args += __cast(arg, types[index]),
for kwarg in kwargs:
kwargs[kwarg] = __cast(kwargs[kwarg], kwtypes[kwarg])
return function(*new_args, **kwargs)
return wrapper
return decorator
def __evaluate(arg):
try:
return ast.literal_eval(arg)
except:
return str(arg)
def __cast(arg, arg_type):
try:
return arg_type(__evaluate(arg))
except:
utils.abort("Unable to cast '{}' to {}".format(arg, arg_type))
Вот как он выглядит в коде:
@task
@params(int, bool, arg1=int, arg2=bool)
def test(arg1, arg2):
print type(arg1), arg1
print type(arg2), arg2
Вот как это выглядит, чтобы вызвать его через fab с хорошими параметрами:
fab test:0.1,1
<type 'int'> 0
<type 'bool'> True
fab test:5,arg2=False
<type 'int'> 5
<type 'bool'> False
fab test:arg1=0,arg2=false
<type 'int'> 5
<type 'bool'> True
ПРИМЕЧАНИЕ. В последнем примере "false" имеет значение True, это ожидаемое поведение в python, однако оно может быть естественным противником интуитивно понятным. Подобно прохождению False как int, он будет преобразовываться в 0 как int (False) == 0 в python
Вот как это выглядит, чтобы вызвать его через fab с плохими параметрами:
fab test:Test,False
Fatal error: Unable to cast 'Test' to <type 'int'>
Aborting.
Это рабочая версия, основанная на https://gist.github.com/mgedmin/f832eed2ac0f3ce31edf. В отличие от старой версии, это на самом деле отличает все возможные параметры декоратора и псевдонимы задачи:
from functools import wraps
from fabric import tasks
def fix_boolean(f):
true_values = ("yes", "true", "1")
false_values = ("no", "false", "0")
def fix_bool(value):
if isinstance(value, basestring):
if value.lower() in false_values:
return False
if value.lower() in true_values:
return True
return value
@wraps(f)
def wrapper(*args, **kwargs):
args_ = [fix_bool(arg) for arg in args]
kwargs_ = {k: fix_bool(v) for k,v in kwargs.iteritems()}
return f(*args_, **kwargs_)
return wrapper
def task(*args, **kwargs):
"""
The fabric.decorators.task decorator which automatically converts command line task arguments
to a boolean representation if applicable.
:param args:
:param kwargs:
:return: wrapped
"""
invoked = bool(not args or kwargs)
task_class = kwargs.pop("task_class", tasks.WrappedCallableTask)
def wrapper(f):
return task_class(fix_boolean(f), *args, **kwargs)
return wrapper if invoked else wrapper(args[0])
Gist: https://gist.github.com/eltismerino/a8ec8584034c8a7d087e