11

これは私の前の質問に関連していますが、別の質問です。

私は次のfabfileを持っています:

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」を使用しましたが、不適切に使用している可能性があります。

エラーを出力して実行を継続するように、この動作を変更するにはどうすればよいですか?

4

4 に答える 4

15

warn_only に関するファブリックのドキュメントによると、

env.warn_only"エラー条件が発生したときにrun、中止する代わりに警告するsudoかどうかを指定します。local

run/ sudo/を実行する前の SSH の試行中に障害が発生するため、サーバーがダウンしている場合、これは役に立ちませんlocal

1 つの解決策は、タスクを実行する前に各サーバーが稼働しているかどうかを確認する関数を作成することです。以下は私が使用したコードです。

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
于 2010-04-01T15:53:25.750 に答える
1

あなたはそれを不適切に使用していません。--warn-only=trueコマンドラインで 指定することもできます。これは、開発チームによって提案された文書化された方法です。

于 2009-12-24T05:15:54.183 に答える
1

マシューの答えに基づいて、まさにそれを達成するデコレータを思いつきました:

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")

複数のホストとロールがある場合に特に便利です。

于 2012-10-04T12:54:54.380 に答える