基於RabbitMQRPC實現的主機非同步管理
README:
1、需求
- [ ] 利用RibbitMQ進行資料互動
- [ ] 可以對多臺伺服器進行批量操作
- [ ] 執行命令後不等待命令的執行結果,而是直接讓輸入下一條命令,結果出來後自動列印
- [ ] 實現非同步操作
備註
- [ ] RabbitMQ佇列:
①執行命令時,佇列名為“rpc_queue2”
②查詢資料時,用的是回撥時隨機生成的callback_queue名
③conf/settings——Rabbitmq地址“192.168.17.102”,埠:5672,使用者名稱:admin,密碼:admin
- [ ] SSH:
RPC_Server/server.py——paramiko操作連線的測試Linux預設埠22,使用者名稱:root,密碼:123456
- [ ] threading多執行緒:
實現命令執行後不等待執行結果,依然可以輸入新的指令
- [ ] 執行命令格式:
-->>run ifconfig host 192.168.20.22 192.168.20.23
dir server端要執行的命令
host host後可跟一個或多個可以通過rabbitMQ的伺服器地址
- [ ] 檢視後臺所有的TASK_ID資訊:
-->>check_task
顯示結果樣式:TASK_ID【76786】 HOST【192.168.20.22】 COMMAND【dir】
TASK_ID【10307】 HOST【192.168.20.23】 COMMAND【dir】
- [ ] 檢視TASK_ID對應的執行結果:
-->>get_task 10307
程式目錄結構:
├── README.md
├── RPC_Client
│ ├── bin
│ │ ├── __init__.py
│ │ └── start.py #客戶端啟動程式
│ ├── conf
│ │ ├── __init__.py
│ │ ├── __pycache__
│ │ │ ├── __init__.cpython-36.pyc
│ │ │ └── settings.cpython-36.pyc
│ │ └── settings.py
│ ├── core
│ │ ├── __init__.py
│ │ ├── main.py
│ │ └── __pycache__
│ │ ├── __init__.cpython-36.pyc
│ │ └── main.cpython-36.pyc
│ └── modules
│ ├── client.py
│ ├── __init__.py
│ └── __pycache__
│ ├── client.cpython-36.pyc
│ └── __init__.cpython-36.pyc
└── RPC_Server
├── conf
│ ├── __pycache__
│ │ └── settings.cpython-36.pyc
│ └── settings.py
└── server.py #server端啟動程式
程式啟動:
客戶端啟動:RPC_Client/bin/start.py
服務端啟動:RPC_Server/server.py
RPC 客戶端bin目錄start.py
import os,sys,platform
if platform.system() == `Windows`:
BASE_DIR = `\`.join(os.path.abspath(os.path.dirname(__file__)).split(`\`)[:-1])
else:
BASE_DIR = `/`.join(os.path.abspath(os.path.dirname(__file__)).split(`/`)[:-1])
sys.path.append(BASE_DIR)
from core import main
if __name__ == "__main__":
handle = main.Handle()
handle.start()
RPC客戶度conf目錄 settings.py
import os,sys,platform
if platform.system() == `Windows`:
BASE_DIR = `\`.join(os.path.abspath(os.path.dirname(__file__)).split(`\`)[:-1])
else:
BASE_DIR = `/`.join(os.path.abspath(os.path.dirname(__file__)).split(`/`)[:-1])
sys.path.append(BASE_DIR)
RabbitmqHost = `192.168.17.102`
RabbitmqUser = `admin`
RabbitmqPwd = `admin`
credentails = pika.PlainCredentials(RabbitmqUser,RabbitmqPwd)
RPC客戶端主程式core/main.py
import pika
import random
import threading
from modules import client
from conf import settings
class Handle(object):
def __init__(self):
# 建立連線,指定伺服器的ip地址
self.connection = pika.BlockingConnection(pika.ConnectionParameters(host=settings.RabbitmqHost,credentials=settings.credentails))
# 建立一個會話,每個channel代表一個會話任務
self.channel = self.connection.channel()
def run_cmd(self,cmd, host):
rpc_client = client.Client(self.connection,self.channel)
task_id = str(random.randint(1000,9999)) #生成4位的Correlation id
response = rpc_client.call(cmd, host)
self.corr_id = response[1]
print(`Task_ID`,task_id)
self.info[task_id] = [self.corr_id,host,cmd,response[0],response[1]]
def start(self):
self.info = {} #task 返回任務資訊字典
help = ```
命令格式
執行系統命令:run command host eg: run ls 10.10.0.10
檢視所有執行任務:check_task
檢視指定任務結果:get_task id eg:get_task 6723
```
print(help)
while True:
msg = input(">> ").strip()
if msg.startswith(`run`) and len(msg.split()) >= 3:
cmd = msg.split()[1]
#多執行緒執行
th_join = []
for host in msg.split()[2:]:
th = threading.Thread(target=self.run_cmd,args=(cmd,host,))
th.start()
th_join.append(th)
for t in th_join:
t.join()
elif msg == `check_task`:
if not self.info:
print("沒有任務佇列")
continue
else:
for taskid,task in self.info.items():
print(`TaskID [%s] Host [%s] COMMAND [%s]`%(taskid,task[1],task[2]))
elif msg.startswith(`get_task`):
rpc_client = client.Client(self.connection,self.channel)
if msg.split()[1] in self.info:
task_id = msg.split()[1]
callback_queue = self.info[task_id][3]
correlation_id = self.info[task_id][4]
task_result = rpc_client.get_task(callback_queue,correlation_id)
del self.info[task_id]
print(task_result.decode().strip())
else:
print(`輸入的task ID不存在!`)
continue
elif not msg:
continue
else:
print(`輸入錯誤,請重新輸入!`)
continue
RPC客戶端modules
import pika
import random
import uuid
class Client(object):
def __init__(self,connection,channel):
self.connection = connection
self.channel = channel
# 對回撥佇列中的響應進行處理的函式
def on_response(self, ch, method, props, body):
if self.correlation_id == props.correlation_id:
self.response = body
ch.basic_ack(delivery_tag=method.delivery_tag)
def get_task(self,callback_queue,correlation_id):
# 初始化 response
self.response = None
self.correlation_id = correlation_id
# 客戶端訂閱回撥佇列,當回撥佇列中有響應時,呼叫`on_response`方法對響應進行處理;
self.channel.basic_consume(self.on_response,queue=callback_queue)
while self.response is None:
self.connection.process_data_events()
return self.response
# 發出RPC請求
def call(self,cmd,host):
# 宣告回撥佇列,再次宣告的原因是,伺服器和客戶端可能先後開啟,該宣告是冪等的,多次宣告,但只生效一次
result = self.channel.queue_declare(exclusive=True)
# 將次佇列指定為當前客戶端的回撥佇列
self.callback_queue = result.method.queue
msg = cmd + " " + "".join(host)
self.corr_id = str(uuid.uuid4())
#self.corr_id = corr_id
# 傳送RPC請求內容到RPC請求佇列`rpc_queue`,同時傳送的還有`reply_to`和`correlation_id`
self.channel.basic_publish(exchange=``,
routing_key=`rpc_queue2`,
properties=pika.BasicProperties(
reply_to=self.callback_queue,
correlation_id=self.corr_id,
),
body=msg)
return self.callback_queue,self.corr_id
RPC伺服器settings.py
RabbitmqHost = `192.168.17.102`
RabbitmqUser = `admin`
RabbitmqPwd = `admin`
credentails = pika.PlainCredentials(RabbitmqUser,RabbitmqPwd)
RPC服務端主程式:
#!/usr/bin/env python
import pika
import paramiko
import os,sys,platform
if platform.system() == `Windows`:
BASE_DIR = `\`.join(os.path.abspath(os.path.dirname(__file__)).split(`\`)[:-1])
else:
BASE_DIR = `/`.join(os.path.abspath(os.path.dirname(__file__)).split(`/`)[:-1])
sys.path.append(BASE_DIR)
from conf import settings
# 建立連線,伺服器地址為localhost,可指定ip地址
connection = pika.BlockingConnection(pika.ConnectionParameters(host=settings.RabbitmqHost,credentials=settings.credentails))
# 建立會話
channel = connection.channel()
# 宣告RPC請求佇列
channel.queue_declare(queue=`rpc_queue2`)
# 資料處理方法
def exec_cmd(cmd,host):
ssh = paramiko.SSHClient()
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
ssh.connect(hostname=host, port=22, username=`root`, password=`123456`,timeout=10)
stdin, stdout, stderr = ssh.exec_command(cmd)
stdout_result = stdout.read()
stderr_result = stderr.read()
result = stdout_result if stdout_result else stderr_result
return result.decode()
ssh.close()
# 對RPC請求佇列中的請求進行處理
def on_request(ch, method, props, body):
cmd = body.split()[0]
host = body.split()[1]
# 呼叫資料處理方法
response = exec_cmd(cmd,host)
# 將處理結果(響應)傳送到回撥佇列
ch.basic_publish(exchange=``,
routing_key=props.reply_to,
properties=pika.BasicProperties(correlation_id = props.correlation_id),
body=str(response))
ch.basic_ack(delivery_tag=method.delivery_tag)
channel.basic_consume(on_request, queue=`rpc_queue2`)
print(" [x] Awaiting RPC requests")
channel.start_consuming()
本文轉自 baiying 51CTO部落格,原文連結:http://blog.51cto.com/baiying/2065436,如需轉載請自行聯絡原作者
相關文章
- 基於etcd的選主功能實現的主備節點管理
- 架構設計|基於 raft-listener 實現實時同步的主備叢集架構Raft
- 基於itchat實現微信群訊息同步機器人機器人
- vmware基於主機模式實現上網(win10)模式Win10
- 基於rust實現的事件系統,支援同步和非同步Rust事件非同步
- 基於 Redis 的方式實現非同步佇列Redis非同步佇列
- RocketMQ(九):主從同步的實現MQ主從同步
- 基於 swoole 的 websocket 服務實現狀態同步Web
- 揭秘MySQL的主從同步實現方案MySql主從同步
- 【RocketMQ】主從同步實現原理MQ主從同步
- 基於SSH培訓機構管理系統的設計與實現
- 基於RBAC實現許可權管理
- 基於MySql主從分離的程式碼層實現MySql
- python中非同步非阻塞如何實現Python非同步
- 關於java實現同步的方法Java
- 基於zookeeper實現統一配置管理
- ZooKeeper系列(3)--基於ZooKeeper實現主從協作
- Js非同步機制的實現JS非同步
- 基於Linux的虛擬主機搭建Linux
- 基於Guava API實現非同步通知和事件回撥GuavaAPI非同步事件
- 基於Django與Celery實現非同步佇列任務Django非同步佇列
- 基於Redis實現Spring Cloud Gateway的動態管理RedisSpringCloudGateway
- .Net Core實現基於Quart.Net的任務管理
- 基於 mysql 非同步驅動的非阻塞 Mybatis 瞭解一下MySql非同步MyBatis
- 基於Travis CI實現 Gitbook在 Github 和 Coding 的同步部署Github
- linux配置基於ip的虛擬主機Linux
- Nginx配置之基於域名的虛擬主機Nginx
- 基於React 原始碼深入淺出setState:setState非同步實現React原始碼非同步
- 資料庫系列——基於Canal實現MySQL增量資料同步資料庫MySql
- 對於同步、非同步、阻塞、非阻塞的幾點淺薄理解非同步
- 關於.net實現網站模板機制(非標籤替換)網站
- 基於 LNMP 的 Nginx 百萬併發之路 (三)基於域名的虛擬主機LNMPNginx
- 基於vue的簡單許可權管理實現總結Vue
- 基於 Flink CDC 的實時同步系統
- 基於jquery實現的ExceljQueryExcel
- 基於JVMTI的Agent實現JVM
- 實現多臺EXSI主機集中管理——安裝VMware vCenter
- 如何使用irealtime.js實現一個基於websocket的同步畫板JSWeb