Как заставить Fabric игнорировать автономные хосты в списке env.hosts?
Это связано с моим предыдущим вопросом, но другим.
У меня есть следующий файл:
from fabric.api import *
host1 = '192.168.200.181'
offline_host2 = '192.168.200.199'
host3 = '192.168.200.183'
env.hosts = [host1, offline_host2, host3]
env.warn_only = True
def df_h():
with settings(warn_only=True):
run("df -h | grep sda3")
И результат:
[192.168.200.199] run: df -h | grep sda3
Fatal error: Low level socket error connecting to host 192.168.200.199: No route to host
Aborting.
После того, как выполнение попадает в автономный сервер, он прерывается немедленно, независимо от других серверов в списке env.hosts.
Я использовал настройку env "warn_only = True", но, возможно, я использую ее неправильно.
Как мне изменить это поведение, чтобы оно только печатало ошибку и продолжало выполнение?
Ответы
Ответ 1
В соответствии с документацией по факсу на warn_only,
env.warn_only
"указывает, следует ли предупреждать, а не прерывать, когда run
/sudo
/local
сталкиваются с ошибками.
Это не поможет в случае отказа сервера, поскольку сбой происходит во время попытки SSH до выполнения run
/sudo
/local
.
Одним из решений было бы создать функцию, чтобы проверить, работает ли каждый сервер до выполнения ваших задач. Ниже приведен код, который я использовал.
from __future__ import print_function
from fabric.api import run, sudo, local, env
import paramiko
import socket
host1 = '192.168.200.181'
offline_host2 = '192.168.200.199'
host3 = '192.168.200.183'
env.hosts = [host1, offline_host2, host3]
def df_h():
if _is_host_up(env.host, int(env.port)) is True:
run("df -h | grep sda1")
def _is_host_up(host, port):
# Set the timeout
original_timeout = socket.getdefaulttimeout()
new_timeout = 3
socket.setdefaulttimeout(new_timeout)
host_status = False
try:
transport = paramiko.Transport((host, port))
host_status = True
except:
print('***Warning*** Host {host} on port {port} is down.'.format(
host=host, port=port)
)
socket.setdefaulttimeout(original_timeout)
return host_status
Ответ 2
Начиная с версии 1.4 Fabric имеет опцию --skip-bad-hosts
, которая может быть установлена из командной строки или путем установки переменной в файле fab.
env.skip_bad_hosts = True
Документация для опции:
http://docs.fabfile.org/en/latest/usage/fab.html#cmdoption--skip-bad-hosts
Не забудьте также явно установить значение тайм-аута.
Ответ 3
Вы не используете его неправильно. Вы даже можете предоставить --warn-only=true
в командной строке. Это документированный метод, предложенный командой разработчиков.
Ответ 4
Основываясь на ответе Мэтью, я придумал декоратора, который выполняет именно это:
from __future__ import with_statement
from paramiko import Transport
from socket import getdefaulttimeout, setdefaulttimeout
from fabric.api import run, cd, env, roles
roledefs = {
'greece': [
'alpha',
'beta'
],
'arabia': [
'kha',
'saad'
]
}
env.roledefs = roledefs
def if_host_offline_ignore(fn):
def wrapped():
original_timeout = getdefaulttimeout()
setdefaulttimeout(3)
try:
Transport((env.host, int(env.port)))
return fn()
except:
print "The following host appears to be offline: " + env.host
setdefaulttimeout(original_timeout)
return wrapped
@roles('greece')
@if_host_offline_ignore
def hello_greece():
with cd("/tmp"):
run("touch hello_greece")
@roles('arabia')
@if_host_offline_ignore
def hello_arabia():
with cd("/tmp"):
run("touch hello_arabia")
Это особенно полезно, когда у вас несколько хостов и ролей.