用python实现与小米网关通讯
python 与小米网关通讯的三块内容:
以下内容的理解需要配合《绿米网关局域网通讯协议》使用
1、监听网关发出的组播信息:(有网关及连接设备的生命信号,事件信息)
2、读取需要获得的信息
3、控制连接设备(涉及了token加密部分)
1、upd广播监听小米网关的组播信息
#!/usr/bin/env python
# -*- coding:utf-8 -*-
import socket
def get_gateway_heart():
SENDERIP = "0.0.0.0"
MYPORT = 9898
MYGROUP = '224.0.0.50'
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, socket.IPPROTO_UDP)
#allow multiple sockets to use the same PORT number
sock.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
#Bind to the port that we know will receive multicast data
sock.bind((SENDERIP,MYPORT))
#tell the kernel that we are a multicast socket
#sock.setsockopt(socket.IPPROTO_IP, socket.IP_MULTICAST_TTL, 255)
#Tell the kernel that we want to add ourselves to a multicast group
#The address for the multicast group is the third param
status = sock.setsockopt(socket.IPPROTO_IP,
socket.IP_ADD_MEMBERSHIP,
socket.inet_aton(MYGROUP) + socket.inet_aton(SENDERIP));
#sock.setblocking(0)
#ts = time.time()
data, addr = sock.recvfrom(1024)
data_str=str(data,encoding='utf-8')
# sock.close()
return data_str
if __name__=='__main__':
while True:
tmp=get_gateway_heart()
print(tmp)
2、小米网关的初始密码向量 转换为字符串 的计算方法
from binascii import b2a_hex, a2b_hex import sys s='17996d093d28ddb3ba695a2e6f58562e' #初始向量 m=a2b_hex(s) print(m) #转换后的初始向量 #b'\x17\x99m\t=(\xdd\xb3\xbaiZ.oXV.'
3、加密token的方法
from Crypto.Cipher import AES
from binascii import b2a_hex, a2b_hex
class prpcrypt():
def __init__(self, key='cy5zmrpqws05vsqj'):
self.key = key #
self.mode = AES.MODE_CBC
# 加密函数,如果text不足16位就用空格补足为16位,
# 如果大于16当时不是16的倍数,那就补足为16的倍数。
def encrypt(self, text): #text是要加密的内容
cryptor = AES.new(self.key, self.mode,b'\x17\x99m\t=(\xdd\xb3\xbaiZ.oXV.')
# 这里密钥key 长度必须为16(AES-),
# (AES-),或者32 (AES-)Bytes 长度
# 目前AES- 足够目前使用
length =
count = len(text)
if count < length:
add = (length - count)
# \ backspace
text = text + ('\0' * add)
elif count > length:
add = (length - (count % length))
text = text + ('\0' * add)
self.ciphertext = cryptor.encrypt(text)
# 因为AES加密时候得到的字符串不一定是ascii字符集的,输出到终端或者保存时候可能存在问题
# 所以这里统一把加密后的字符串转化为16进制字符串
return str(b2a_hex(self.ciphertext),encoding='utf-8')
#return self.ciphertext
# 解密后,去掉补足的空格用strip() 去掉 b'
def decrypt(self, text):
cryptor = AES.new(self.key, self.mode, b'\x17\x99m\t=(\xdd\xb3\xbaiZ.oXV.')
plain_text = cryptor.decrypt(a2b_hex(text))
return plain_text.rstrip('\0')
if __name__ == '__main__':
pc = prpcrypt('0987654321qwerty') # 初始化密钥
e = pc.encrypt('1234567890abcdef') # 加密
# d = pc.decrypt(e) # 解密
print("加密:", e)
#print("解密:", d)
4、获取token,并对获取的token进行加密
#!/usr/bin/env python
# -*- coding:utf-8 -*-
import socket
import json
from xm_gw.encrypty import prpcrypt
def get_token(): #通过get_id_list 获得token
ip_port_single = ("192.168.31.150", 9898)
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)
comd = {'cmd': 'get_id_list'}
order = json.dumps(comd)
s.sendto(bytes(order, encoding="utf-8"), ip_port_single)
data,addr=s.recvfrom(1024)
data_str=str(data,encoding='utf-8')
token=json.loads(data_str).get('token')
s.close()
return token
def get_token_encrypty():
tok = get_token() # 拿到当前token,要进行加密的内容
k = prpcrypt()
key_encrypt = k.encrypt(tok)
return key_encrypt
if __name__=='__main__':
tok=get_token()
tok_encry=get_token_encrypty()
print(tok)
print(tok_encry)
5、建立网关通讯并执行控制命令
#!/usr/bin/env python
# -*- coding:utf-8 -*-
import socket
import json
from xm_gw import udp_token_key
class udp_gw():
def __init__(self, ip_gateway='192.168.8.100'):
self.ip_port_zu43 = ('224.0.0.50', 4321)
self.ip_port_single = (ip_gateway, 9898)
self.ip_port_zu9898=('224.0.0.50', 9898)
def whois(self):
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)
comd = {'cmd': 'whois'}
order = json.dumps(comd)
s.sendto(bytes(order, encoding="utf-8"), self.ip_port_zu43)
data_bytes, addr = s.recvfrom(1024)
data_dic = json.loads(str(data_bytes, encoding='utf-8'))
s.close()
return data_dic
def get_id_list(self):
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)
comd = {'cmd': 'get_id_list'}
order = json.dumps(comd)
s.sendto(bytes(order, encoding="utf-8"), self.ip_port_single)
data_bytes, addr = s.recvfrom(1024)
data_dic = json.loads(str(data_bytes, encoding='utf-8'))
s.close()
return data_dic
def read_sid(self, sid):
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)
comd = {'cmd': 'read', 'sid': sid}
order = json.dumps(comd)
s.sendto(bytes(order, encoding="utf-8"), self.ip_port_single)
data_bytes, addr = s.recvfrom(1024)
data_dic = json.loads(str(data_bytes, encoding='utf-8'))
s.close()
return data_dic
def write_plug(self, status):
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)
key_encrypt = udp_token_key.get_token_encrypty()
comd = {"cmd": "write", "model": "plug", "sid": "158d0001b84d9a", "short_id": 46384,
"data": {"status": status, 'key': key_encrypt}}
order = json.dumps(comd)
s.sendto(bytes(order, encoding="utf-8"), self.ip_port_single)
data_bytes, addr = s.recvfrom(1024)
data_dic = json.loads(str(data_bytes, encoding='utf-8'))
s.close()
return data_dic
def read_all_sid(self):
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)
ls = json.loads(self.get_id_list().get('data'))
ls_sensor_state = []
for sid in ls:
comd = {'cmd': 'read', 'sid': sid}
order = json.dumps(comd)
s.sendto(bytes(order, encoding="utf-8"), self.ip_port_single)
data_bytes, addr = s.recvfrom(1024)
data_dic = json.loads(str(data_bytes, encoding='utf-8'))
# print(data_dic)
ls_sensor_state.append(data_dic)
s.close()
return ls_sensor_state
def get_dict_model_sid(self):
dic_gw=self.whois()
ls=self.read_all_sid()
dic_model_sid = {}
for dic in ls:
model = dic.get('model')
sid = dic.get('sid')
dic_model_sid[model] = sid
dic_model_sid['gateway'] = dic_gw.get('sid')
return dic_model_sid
if __name__=='__main__':
import time
#{'plug': '158d0001b84d9a', 'switch': '158d0001c10bd7', 'sensor_ht': '158d0001e87bd9',
# 'magnet': '158d0001bb3daf', 'motion': '158d0001c2f110', 'gateway': '7811dcb38599'}
gw=udp_gw('192.168.31.150')
tmp = gw.read_sid('158d0001b84d9a')
# print(tmp1)
# time.sleep(5)
# gw.write_plug('off')
# time.sleep(5)
# tmp=gw.read_sid('158d0001b84d9a')
print(tmp)
用python实现与小米网关通讯的更多相关文章
- python网络编程:TCP通讯模板、粘包及解决方案、自定义报头
一.TCP通讯模板 二.远程CMD程序 三.解决粘包问题 四.解决粘包问题2 一.TCP通讯模板 TCP客户端 import socket c = socket.socket() # 连接服务器 c. ...
- [Python]实现XMPP协议即时通讯发送消息功能
#-*- coding: utf-8 -*- __author__ = 'tsbc' import xmpp import time #注意帐号信息,必须加@域名格式 from_user = 'che ...
- Python基础(四) socket简单通讯
socket:我们通常听过的套接字: 服务端: 1.创建socket对象 2.bing 绑定ip及端口 3.对该端口进行监听 4.消息阻塞(等待客户端消息) 客户端: 1.创建socket对象 2.连 ...
- python处理多线程之间事件通讯方法
一.什么是事件 每执行一个事情,肯定有该事情的执行后状态,那事件就是该事情发生的信号 在程序中,多线程之间需要通讯,而事件就是方便线程之间的通讯 案例: 1.服务器启动需要5秒 2.客服端启动后去链接 ...
- 用Robot Framework+python来测试基于socket通讯的C/S系统(网络游戏)
项目终于换了方案,改用socket来实现而不是之前的http了,所以测试工具就不能用以前的了,因为测试人手少,逼不得已的必须要挖掘更多的自动化方案来弥补.于是先研究了下python的socket解决方 ...
- 小米网关api
http://bbs.xiaomi.cn/t-13198850 https://github.com/snOOrz/homebridge-aqara/blob/master/README.md htt ...
- TCP/IP协议,TCP与平台通信,通讯协议压力测试(python)
最近的项目来了一个需求,要求测试tcp网关通讯协议: 1.液压井盖通过TCP/IP TCP与平台通信: 2.硬件定期发送心跳包(10S)给平台,是平台与硬件保持长连接: 3.每台硬件有一个12字节的唯 ...
- python中利用队列asyncio.Queue进行通讯详解
python中利用队列asyncio.Queue进行通讯详解 本文主要给大家介绍了关于python用队列asyncio.Queue通讯的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细 ...
- 【XMLRPC实现跨语言编程】Tcl <----> python
转载请声明出处,谢谢合作. # 期望一种能实现tcl.python两者解释器能双向通讯的结合体# py->tcl: from Tkinter import Tcl; tcl = Tcl(); t ...
随机推荐
- 重磅消息-Service Fabric 正式开源
微软的Azure Service Fabric的官方博客在2017.3.24日发布了一篇博客 Service Fabric .NET SDK goes open source ,介绍了社区呼声最高的S ...
- webstorm提交版本时,忽略特定文件
项目提交时,部分本地配置文件,不需要提交,这时候需要在整个版本控制中忽略掉文件的提交. 操作如下: File -> Settings -> Version Control -> Ig ...
- 在 WinForm 中 如何实现 加载等待功能
1,需要一个动态的londing文件:在项目中我们新建一个文件夹来存放它: 2,在需要出现londing状态的窗体上加上一个Panel: 黄色区域是Panel,灰色的是需要被加载的区域.当需要触发lo ...
- Java的类C结构体的实现,以及对类的某一属性进行排序
public static class Foo { public int x1; public int x2; public int day; } public static Foo[] bridge ...
- webrtc起步 - apprtc服务器搭建
简介 apprtc 是什么,webrtc.org官方指定体验app 原料: ubuntu14.04,其他linux版本不限,官方并没特殊说明 chrome M51+ stunnle ...
- html、css简述面试题
hTML, HTTP,web综合问题 1.前端需要注意哪些SEO 合理的title.description.keywords:搜索对着三项的权重逐个减小,title值强调重点即可,重要关键词出现不要超 ...
- .Net调用钉钉接口,实现发送企业消息功能
我在钉钉官网上看了下,关于调用钉钉接口,发送企业消息通知,他给的核心代码总共几行. 用到的一个接口引用 TopSDK.dll 下载网址https://open-doc.dingtalk.com/doc ...
- C语言--第0次作业
1.你认为大学的学习生活.同学关系.师生应该是怎样?请一个个展开描写. 学习生活 大学是一个过渡时间,它不同于高中与社会.我希望自己可以养成自主学习的习惯,也希望能在大学学习中找到属于自己的节奏,不被 ...
- Redis+Restful 构造序列号和压力测试【原创】
[本人原创],欢迎交流和分享技术,转载请附上如下内容:如果你觉得这篇文章对你有帮助,请记得帮我点赞, 谢谢!作者:kevin[转自]http://www.cnblogs.com/itshare/ 很多 ...
- 开源一套基于vue全家桶的webapp
一.设计初衷 原本今年就是有一个打算要做一套商业的作品,恰巧目前离职,在找工作的过程中,所以有时间闲下来沉淀对原本的知识进行梳理. 说一个题外话,就是由于博主之前是很早一批使用vue的用户,也就是距今 ...