paramiko類Fabric主機管理

環境:Linux python3.5

要求:
類 Fabric 主機管理程序開發:
1. 運行程序列出主機組或者主機列表
2. 選擇指定主機或主機組
3. 選擇讓主機或者主機組執行命令或者向其傳輸文件(上傳/下載)
4. 充分使用多線程或多進程
5. 不一樣主機的用戶名密碼、端口能夠不一樣

結構:
bin-----|
start.py ......啓動目錄,需配置主機列表
core----|
main.py ......主目錄
conf----|
config.py ......配置目錄
system.ini ......配置文件
file ......上傳下載默認目錄

用法:
先在system.ini中配置主機IP組
選擇主機IP,輸入用戶名密碼,登錄
選擇須要執行命令或者傳輸文件
傳輸文件默認目錄爲file目錄

bin:
#!/usr/bin/env python
# -*-coding:utf-8-*-
# Author:zh
import os
import sys
import threading
PATH = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.append(PATH)
from core import main
from conf import config

if __name__ == '__main__':
    conf = config.Configuration()
    ip_list = eval(conf.get_config()[0][1])
    thread = threading.Thread(target=main.main, args=(ip_list,))
    thread.start()
start.py

core:python

#!/usr/bin/env python
# -*-coding:utf-8-*-
# Author:zh

import paramiko
import os
PATH = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))+os.sep+"file"+os.sep


class Fabric(object):
    '''這個類用來初始化SFTPClient和SSHClient'''
    def __init__(self, hostname, port, username, password):
        self.hostname = hostname
        self.port = port
        self.username = username
        self.password = password
        self.transport = paramiko.Transport((self.hostname, self.port))
        self.transport.connect(username=self.username, password=self.password)

    def command(self, cmd_input):
        # 執行命令
        ssh = paramiko.SSHClient()
        ssh._transport = self.transport
        stdin, stdout, stderr = ssh.exec_command(cmd_input)
        res, err = stdout.read(), stderr.read()
        result = res if res else err
        print(result.decode())

    def sftp(self, data):
        # 上傳下載入口
        self.sftp = paramiko.SFTPClient.from_transport(self.transport)
        data_split = data.split()
        sign = data_split[0]
        if len(data_split) == 2:
            file_path = data_split[1]
            if file_path.rfind("/") == -1:
                file_name = file_path
            else:
                file_name = file_path[file_path.rfind("/")+1:]
            local_path = PATH+file_name
        elif len(data_split) == 3:
            if sign == "get":
                local_path = data_split[2]
                file_path = data_split[1]
            if sign == "put":
                file_path = data_split[2]
                local_path = data_split[1]
        else:
            local_path = ''
            file_path = ''
        if hasattr(self, sign):
            func = getattr(self, sign)
            try:
                func(file_path, local_path)
            except Exception as e:
                print("錯誤 %s" % e)
        else:
            print("輸入錯誤")
            self.help()

    def get(self, *args):
        # 下載
        file_path = args[0]
        local_path = args[1]
        self.sftp.get(file_path, local_path)

    def put(self, *args):
        # 上傳
        file_path = args[0]
        local_path = args[1]
        self.sftp.put(local_path, file_path)

    @staticmethod
    def help(self):
        # 展現幫助信息
        show = '''
        get path local_path ......從path下下載到本地目錄
        put local_path path ......從本地上傳到path目錄下
        '''
        print(show)

    def close(self):
        # 關閉鏈接
        self.transport.close()


def logon():
    # 登錄,輸入端口,用戶名,密碼
    while True:
        port = input("請輸入端口:")
        if not port:
            continue
        if not port.isdigit():
            print("請輸入正確的端口")
            continue
        port = int(port)
        name = input("請輸入用戶名:")
        if not name:
            continue
        pwd = input("請輸入密碼:")
        if not pwd:
            continue
        return [port, name, pwd]


def show(show_list):
    '''展現列表給用戶選擇,並返回選擇信息'''
    while True:
        for value, comment in enumerate(show_list):
            print("%s. %s" % (value+1, comment))
        choose = input("請選擇:")
        if not choose:
            continue
        if choose == "exit":
            exit()
        try:
            choose = int(choose)-1
            show_list[choose]
            return choose
        except (ValueError, IndexError) as e:
            print("輸入錯誤:%s ,請從新輸入" % e)
            continue


def main(ip_list):
    # 主方法,程序入口
    ip_list = ip_list
    choose_ip = show(ip_list)
    value_list = logon()
    port = value_list[0]
    name = value_list[1]
    pwd = value_list[2]
    while True:
        action_list = ["執行命令", "上傳下載"]
        choose_action = show(action_list)
        ssh = Fabric(ip_list[choose_ip], port, name, pwd)
        if choose_action == 0:
            func = ssh.command
        else:
            func = ssh.sftp
        while True:
            cmd_input = input("-->")
            if not cmd_input:
                continue
            if cmd_input == 'b':
                ssh.close()
                break
            if cmd_input == 'exit':
                exit()
            func(cmd_input)
main.py

conf:git

#!/usr/bin/env python
# -*-coding:utf-8-*-
# _author_=zh
import os
import configparser
PATH = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))


class Configuration(object):
    def __init__(self):
        self.config = configparser.ConfigParser()
        self.name = PATH+os.sep+"conf"+os.sep+"system.ini"

    def init_config(self):
        # 初始化配置文件,ip :客戶端IP,port:客戶端端口
        if not os.path.exists(self.name):
            self.config["config"] = {"ip_list":['192.168.200.128', '192.168.200.129', '192.168.200.130']}
            self.config.write(open(self.name, "w", encoding="utf-8", ))

    def get_config(self, head="config"):
        '''
        獲取配置文件數據
        :param head: 配置文件的section,默認取初始化文件config的數據
        :return:返回head中的全部數據(列表)
        '''
        self.init_config() # 取文件數據以前生成配置文件
        self.config.read(self.name, encoding="utf-8")
        if self.config.has_section(head):
            section=self.config.sections()
            return self.config.items(section[0])
config.py
相關文章
相關標籤/搜索