python第六十三天-- 第十一周作业
题目:基于RabbitMQ rpc实现的主机管理
需求:
可以对指定机器异步的执行多个命令
例子:
>>:run "df -h" --hosts 192.168.3.55 10.4.3.4
task id: 45334
>>: check_task 45334
>>:
注意,每执行一条命令,即立刻生成一个任务ID,不需等待结果返回,通过命令check_task TASK_ID来得到任务结果
README
基于RabbitMQ rpc实现的主机管理
可以对指定机器异步的执行多个命令
例子:
>>:run "df -h" --hosts 192.168.3.55 10.4.3.4
task id: 45334
>>: check_task 45334 #查看任务信息 程序结构:
RabbitMQ_PRC/#综合目录
|- - -PRC_CLIENT/#client程序主目录
| |- - -__init__.py
| |- - -bin/#执行程目录
| | |- - -__init__.py
| | |- - -clien_start.py #客户端执行文件
| |
| |
| |- - -core #主逻辑程序目录
| | |- - -__init__.py
| | |- - -clien_class.py#客户端执行主要逻辑 类
| |
| |
|
|
|- - -PRC_SERVER/#服务端程序目录
| |- - -__init__.py
| |- - -bin/#执行目录
| | |- - -__init__.py
| | |- - -server_start.py#服务端程序执行文件
| |
| |
| |- - -core/##主逻辑程序目录
| | |- - -server_class.py#主逻辑 相关类
| |
|
|- - -README
程序结构:
RabbitMQ_PRC/#综合目录
|- - -PRC_CLIENT/#client程序主目录
| |- - -__init__.py
| |- - -bin/#执行程目录
| | |- - -__init__.py
| | |- - -clien_start.py #客户端执行文件
import os ,sys
BASE_DIR=os.path.dirname(os.path.dirname(os.path.abspath(__file__)))#获取相对路径转为绝对路径赋于变量
sys.path.append(BASE_DIR)#增加环境变量 from core.client_class import Threa if __name__ == '__main__':
RPCS=Threa()
response=RPCS.th_start()
| |- - -core #主逻辑程序目录
| | |- - -__init__.py
| | |- - -clien_class.py#客户端执行主要逻辑 类
import pika
import uuid
import threading
import random class FibonacciRpcClient(object):
def __init__(self):
#self.credentials=pika.PlainCredentials("test","test")
self.connection = pika.BlockingConnection(pika.ConnectionParameters(host="localhost"))#生成连接的服务端 ip
#self.connection = pika.BlockingConnection(pika.ConnectionParameters("192.168.11.51",15672,'/',self.credentials))#生成连接的服务端 ip
self.channel = self.connection.channel()#创建一个管道 def get_respon(self,cal_queue,cal_id):#取任务信息
self.response=None
self.callback_id=cal_id#队列名
self.channel.basic_consume(self.on_response,queue=cal_queue)# 使用回调函数
while self.response is None:
self.connection.process_data_events()#非阻塞模式接收消息
return self.response#返回 def on_response(self, ch, method, props, body):#回调函数
if self.callback_id == props.correlation_id:#判断服务端返回的队列名是否与当前所生成的队列名一致
self.response = body# 将服务端的结果赋于返回来的结果变量
ch.basic_ack(delivery_tag = method.delivery_tag)##确保消息被 接收 def call(self, queues,n):#发送消息的函数
result = self.channel.queue_declare(exclusive=False)#随机生成一个队列,收消息后不删除
self.callback_queue = result.method.queue#赋于管道 变量
self.corr_id = str(uuid.uuid4())#生成一个服务端返回消息的队列名
self.channel.basic_publish(exchange='',
routing_key=queues,#队列名
properties=pika.BasicProperties(
reply_to = self.callback_queue,#发送的管道队列名
correlation_id = self.corr_id,#发送给服务端,用于返回消息的队列名
),
body=str(n))#发送的内容数据
return self.callback_queue,self.corr_id#返回管道名 队列id号 class Threa(object):#线程 类
def __init__(self):
self.info={}#生成一个字典
self.help_info=''' 指令示例\033[36;1m
run "df -h" --hosts 192.168.3.55 10.4.3.4
--- ------- ------- ------------ --------
运行 指令 主机 ip 1# ip 2#
check_task_all #查看任务列表
check_task 25413 #查看具体id任务信息,过后删除
helps #查看指令帮助
\033[0m''' def check_task_all(self,cmd):#查看所有任务信息 for i in self.info:
print("任务id:%s,服务端:%s,命令:%s"%(i,self.info[i][0],self.info[i][1]))
def check_task(self,take_id):#查看任务
try:
id=int(take_id.split()[1])#取任务ID
#print(id,'任务ID')
cal_queue=self.info[id][2]#管道名
#print(cal_queue,'队列')
cal_id=self.info[id][3]#消息队列位置
#print(cal_id,'消息位置')
clinets=FibonacciRpcClient()#调用类
rest=clinets.get_respon(cal_queue,cal_id)#取任务信息
print('任务执行结果:',rest.decode())#打印
del self.info[id]#从字典中删除对应任务
except Exception as e:
print(e)
return def run(self,str_l):#run函数
addr_l=self.attr_l(str_l)#获取IP
oreds=self.oreds_(str_l)#获取 命令
#print(oreds,'上传命令')
for i in addr_l:#取出IP
tak_id=random.randint(10000,99999)#任务ID生成
#print(tak_id,'任务ID')
obj=FibonacciRpcClient()#生成连接类
r=obj.call(i,oreds)#ip做队列名 命令
self.info[tak_id]=[i,oreds,r[0],r[1]]#写入字典 tak_id{ ip 命令 管道名 队列名}
return self.info def retf(self,str_l):#反射命令
sl=str_l.split()[0]#取命令开头
if sl=='helps':
self.helps()
if len(str_l.split())==1 and sl!='check_task_all' :
return
if hasattr(self,sl):#是否存在
func=getattr(self,sl)#调用
rer=func(str_l)#执行
#print(rer)
if rer is not None:
for i in rer:
print("任务id:%s"%i) def attr_l(self,n):#命令分解函数
attr=n.split("--")##用--分割
addr=attr[1].split()[1:]#获取IP列表
return addr#返回IP列表 def oreds_(self,n):#获取 命令
oreds=n.split("\"")[1]##用"分割取命令
return oreds#返回 命令 def helps(self):#查看指令帮助
print(self.help_info) def th_start(self):#开始
self.helps()
while True:
str_l=input(">>:").strip()
if not str_l:continue#如果为空重新输入
t1=threading.Thread(target=self.retf,args=(str_l,))#创建新线程 调用反射函数
t1.start()#开始线程
|- - -PRC_SERVER/#服务端程序目录
| |- - -__init__.py
| |- - -bin/#执行目录
| | |- - -__init__.py
| | |- - -server_start.py#服务端程序执行文件
import os ,sys
BASE_DIR=os.path.dirname(os.path.dirname(os.path.abspath(__file__)))#获取相对路径转为绝对路径赋于变量
sys.path.append(BASE_DIR)#增加环境变量 from core.client_class import Threa if __name__ == '__main__':
RPCS=Threa()
response=RPCS.th_start()
| |- - -core/##主逻辑程序目录
| | |- - -server_class.py#主逻辑 相关类
import pika,os class RabbitMQ_PRC(object):
def __init__(self,myaddr):
self.queues=myaddr#用本机IP做队列名
self.connection = pika.BlockingConnection(pika.ConnectionParameters(host='localhost'))#生成消息对队
self.channel = self.connection.channel()#生成管道
self.channel.queue_declare(queue=self.queues)#消息收接队列 def str_run(self,body):#处理 run的函数
msg = os.popen(body.decode()).read()#执行系统命令
if not msg:
msg = '系统命令不存在'
return msg def on_request(self,ch, method, props, body):#回调函数
resp=self.str_run(body)
print('执行完成')
#print(resp)
ch.basic_publish(exchange='',
routing_key=props.reply_to,#收消息的队列
properties=pika.BasicProperties(correlation_id =props.correlation_id),#返回消息的队列
body=str(resp))#返回结果数据
ch.basic_ack(delivery_tag = method.delivery_tag)##确保消息被 客户端接收 def run_(self):
self.channel.basic_qos(prefetch_count=1)#同时只处理一个消息
self.channel.basic_consume(self.on_request, queue=self.queues)#接收消息,自动调用回调函数 print("开始接收数据!")
self.channel.start_consuming()#开始接收
python第六十三天-- 第十一周作业的更多相关文章
- python第六十八天--第十二周作业
主题: 需求: 用户角色,讲师\学员, 用户登陆后根据角色不同,能做的事情不同,分别如下讲师视图 管理班级,可创建班级,根据学员qq号把学员加入班级 可创建指定班级的上课纪录,注意一节上课纪录对应多条 ...
- python练习六十三:文件处理,读取文件内容,按内容生成文件
python练习六十三:文件处理 假设要读取code.txt文件中内容,code.txt文件内容如下 01 CN Chinese 02 US United States of America 03 J ...
- 孤荷凌寒自学python第六十三天学习mongoDB的基本操作并进行简单封装2
孤荷凌寒自学python第六十三天学习mongoDB的基本操作并进行简单封装2 (完整学习过程屏幕记录视频地址在文末) 今天是学习mongoDB数据库的第九天. 今天继续学习mongoDB的简单操作, ...
- 2017-2018-2 20179205《网络攻防技术与实践》第十一周作业 SQL注入攻击与实践
<网络攻防技术与实践>第十一周作业 SQL注入攻击与实践 1.研究缓冲区溢出的原理,至少针对两种数据库进行差异化研究 缓冲区溢出原理 在计算机内部,输入数据通常被存放在一个临时空间内, ...
- 《Linux内核原理与设计》第十一周作业 ShellShock攻击实验
<Linux内核原理与设计>第十一周作业 ShellShock攻击实验 分组: 和20179215袁琳完成实验及博客攥写 实验内容: Bash中发现了一个严重漏洞shellshock, ...
- 2019-2020-1 20199329《Linux内核原理与分析》第十一周作业
<Linux内核原理与分析>第十一周作业 一.本周内容概述: 学习linux安全防护方面的知识 完成实验楼上的<ShellShock 攻击实验> 二.本周学习内容: 1.学习& ...
- 2020-2021-1 20209307 《Linux内核原理与分析》第十一周作业
这个作业属于哪个课程 <2020-2021-1Linux内核原理与分析)> 这个作业要求在哪里 <2020-2021-1Linux内核原理与分析第十一周作业> 这个作业的目标 ...
- 1903021121-刘明伟-java十一周作业-java面向对象编程
项目 内容 课程班级博客链接 19级信计班(本) 作业要求链接 第十一周作业 博客名称 1903021121-刘明伟-java十一周作业-java面向对象 要求 每道题要有题目,代码(使用插入代码,不 ...
- 2017-2018-2 1723《程序设计与数据结构》第十一周作业 & 实验三 & (总体)第三周结对编程 总结
作业地址 第十一次作业:https://edu.cnblogs.com/campus/besti/CS-IMIS-1723/homework/1933 (作业界面已评分,可随时查看,如果对自己的评分有 ...
随机推荐
- Python日常实践(1)——SQL Prompt的Snippets批量整理
引言 个人平时在写sql脚本的时候会使用到SQL Prompt这款插件,除了强大的智能提示和格式化sql语句功能,我还喜欢使用Snippets代码段功能.比如我们可以在查下分析器输入ssf后按Tab键 ...
- PHP-CPP开发扩展(四)
PHP-CPP是一个用于开发PHP扩展的C++库.本节讲解如何在C++中调用PHP函数. 调用PHP函数 调用普通函数 // call a function from user space Php:: ...
- Tensorflow学习笔记(2):tf.nn.dropout 与 tf.layers.dropout
A quick glance through tensorflow/python/layers/core.py and tensorflow/python/ops/nn_ops.pyreveals t ...
- PM2来部署nodejs服务器永久开启
pm2 日常使用 1. pm2 是什么? 日常开发中需要启动一个node项目,需要用npm run …,,如果终端被关掉,程序也就自动停止,有时候几个项目一起跑起来,好几个终端开着,个人不太喜欢, ...
- [Vijos 1676] 陶陶吃苹果
Description curimit知道陶陶很喜欢吃苹果.于是curimit准备在陶陶生日的时候送给他一棵苹果树. curimit准备了一棵这样的苹果树作为生日礼物:这棵苹果树有n个节点,每个节点上 ...
- 记一次安装Nginx+php-fpm安装后无法解析.php文件,状态码200,但显示空白页
安装环境: Nginx:Nginx1.12.2 PHP:PHP 7.2 系统:CentOS 7.4 安装方式: Nginx与PHP都是yum安装的,具体步骤: 1.安装epel源再安装Nginx: r ...
- 在服务器上搭建wordpress个人博客 php7.2+nginx+mysql+wordperss
买了台VPS,准备搭建一个博客.用过几个博客框架还是觉得Wordpress好用.主题多,插件也非常的便利,而且大多还免费开源.搭建也很简单,其实安装好php+mysql+nginx+wordpress ...
- JavaScript类继承
和其他功能一样,ECMAScript 实现继承的方式不止一种.这是因为 JavaScript 中的继承机制并不是明确规定的,而是通过模仿实现的.这意味着所有的继承细节并非完全由解释程序处理.作为开发者 ...
- Java学习笔记之——变量与数据类型、运算符
一.变量 1.变量:变化的值 变量在代码运行期间,开辟了一块空间 .这块空间是有地址的,给这块取了个名字, 这个名字就叫做变量名,这块空间放的东西叫做变量值 2.变量的初始化: (1)先声明再赋值: ...
- 项目开发版本控制----Git
版本控制的工具我早之前用的svn,后来换成了git.同样是版本控制,为什么要换呢?肯定是有原因的啦~ 一.Git和SVN的比较 svn的优缺点 优点: 1.管理方便,逻辑明确,符合一般人思维习惯. 2 ...