需要先了解绿米网关局域网通讯协议

地址:绿米网关局域网通讯协议 · GitBook​legacy.gitbook.com

主要就是三个步骤:1、监听网关发出的组播信息:(有网关及连接设备的信号,事件信息)

2、读取需要获得的信息(读取温度、湿度什么的)

3、控制连接设备(写入命令需要加密,涉及了token加密部分)

主要用到python进行socket

1、upd广播监听小米网关的组播信息(设备参数变化会自动上传,这一块我测试失败了,不知道是因为什么,比如温度上升了十度也收不到信息)

1 #!/usr/bin/env python

2 # -*- coding:utf-8 -*-

3

4 import socket

5

6 def get_gateway():

7 SENDERIP = "0.0.0.0"

8 MYPORT = 9898

9 MYGROUP = '224.0.0.50'

10

11 sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, socket.IPPROTO_UDP)#创建socket,声明UDP协议

12 #allow multiple sockets to use the same PORT number

13 sock.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)

14 #Bind to the port that we know will receive multicast data

15 sock.bind((SENDERIP,MYPORT))#绑定ip和端口

16 #tell the kernel that we are a multicast socket

17 #sock.setsockopt(socket.IPPROTO_IP, socket.IP_MULTICAST_TTL, 255)

18 #Tell the kernel that we want to add ourselves to a multicast group

19 #The address for the multicast group is the third param

20 status = sock.setsockopt(socket.IPPROTO_IP,

21 socket.IP_ADD_MEMBERSHIP,

22 socket.inet_aton(MYGROUP) + socket.inet_aton(SENDERIP));#将程序运行ip和小米网关的ip设置一个网段下

26 data, addr = sock.recvfrom(1024)#收到上报信息,最大1024

27 data_str=str(data,encoding='utf-8')#发出的是bytes格式,转化成utf-8

28 # sock.close()

29 return data_str

30

31

32 if __name__=='__main__':

33 while True:

34 tmp=get_gateway()

35 print(tmp)

2、小米网关的初始密码向量 转换为字符串 的计算方法

1 from binascii import b2a_hex, a2b_hex

2 import sys

3

4 s='17996d093d28ddb3ba695a2e6f58562e' #初始向量

5 m=a2b_hex(s)

6 print(m)

7

8 #转换后的初始向量

9 #b'\x17\x99m\t=(\xdd\xb3\xbaiZ.oXV.' #下边直接用就可以了

3、加密token的方法

1 from Crypto.Cipher import AES

2 from binascii import b2a_hex, a2b_hex

3

4 class pt():

5 def __init__(self, key='小米网关的key,在米家APP的开发者模式中获取,找到网关的关于,狂点下边的空白处,奇迹发生'):

6 self.key = key

7 self.mode = AES.MODE_CBC

8

9 # 加密函数,如果text不足16位就用空格补足为16位,

10 # 如果大于16当时不是16的倍数,那就补足为16的倍数。

11 def encrypt(self, text): #text是要加密的内容

12 cryptor = AES.new(self.key, self.mode,b'\x17\x99m\t=(\xdd\xb3\xbaiZ.oXV.')

13 # 这里密钥key 长度必须为16(AES-128),

14 # 24(AES-192),或者32 (AES-256)Bytes 长度

15 # 目前AES-128 足够目前使用

16 length = 16

17 count = len(text)

18 if count < length:

19 add = (length - count)

20 # \0 backspace

21 text = text + ('\0' * add)

22 elif count > length:

23 add = (length - (count % length))

24 text = text + ('\0' * add)

25 self.ciphertext = cryptor.encrypt(text)

26 # 因为AES加密时候得到的字符串不一定是ascii字符集的,输出到终端或者保存时候可能存在问题

27 # 所以这里统一把加密后的字符串转化为16进制字符串

28 return str(b2a_hex(self.ciphertext),encoding='utf-8')

29

30

31 # 解密后,去掉补足的空格用strip() 去掉 b'0000000000000000'

32 def decrypt(self, text):

33 cryptor = AES.new(self.key, self.mode, b'\x17\x99m\t=(\xdd\xb3\xbaiZ.oXV.')

34 plain_text = cryptor.decrypt(a2b_hex(text))

35 return plain_text.rstrip('\0')

36

37

38 if __name__ == '__main__':

39 p = pt('0987654321qwerty') # 初始化密钥

40 e = p.encrypt('1234567890abcdef') # 加密

41 d = p.decrypt(e) # 解密

42 print("加密:", e)

43 print("解密:", d)

4、获取token,并对获取的token进行加密

#!/usr/bin/env python

# -*- coding:utf-8 -*-

import socket

import json

def get_token(): #通过get_id_list 获得token

ip_port_single = ("小米网关ip", 9898)#和key值一样获取,米家app里

s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)

comd = {'cmd': 'get_id_list'}#设置json格式命令

order = json.dumps(comd)

s.sendto(bytes(order, encoding="utf-8"), ip_port_single)#发送命令,转化为16进制

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 = pt()

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、建立网关通讯并执行控制命令

1 #!/usr/bin/env python

2 # -*- coding:utf-8 -*-

3 import socket

4 import json

5 from xm_gw import udp_token_key

6

7

8 class udp_gw():

9 def __init__(self, ip_gateway='192.168.1.1'):

10 self.ip_port_43 = ('224.0.0.50', 4321)

11 self.ip_port_single = (ip_gateway, 9898)

12 self.ip_port_zu9898=('224.0.0.50', 9898)

13

14 def whois(self):#发现设备命令

15 s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)

16 comd = {'cmd': 'whois'}

17 order = json.dumps(comd)

18 s.sendto(bytes(order, encoding="utf-8"), self.ip_port_43)

19 data_bytes, addr = s.recvfrom(1024)

20 data_dic = json.loads(str(data_bytes, encoding='utf-8'))

21 s.close()

22 return data_dic

23

24 def get_id_list(self):#列出设备命令

25 s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)

26 comd = {'cmd': 'get_id_list'}

27 order = json.dumps(comd)

28 s.sendto(bytes(order, encoding="utf-8"), self.ip_port_single)

29 data_bytes, addr = s.recvfrom(1024)

30 data_dic = json.loads(str(data_bytes, encoding='utf-8'))

31 s.close()

32 return data_dic

33

34 def read_sid(self, sid):#读取设备ID

35 s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)

36 comd = {'cmd': 'read', 'sid': sid}

37 order = json.dumps(comd)

38 s.sendto(bytes(order, encoding="utf-8"), self.ip_port_single)

39 data_bytes, addr = s.recvfrom(1024)

40 data_dic = json.loads(str(data_bytes, encoding='utf-8'))

41 s.close()

42 return data_dic

43

44 def write_plug(self, status):#写命令

45 s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)

46 key_encrypt = udp_token_key.get_token_encrypty()

47 comd = {"cmd": "write", "model": "plug", "sid": "设备id", "short_id": 设备short_id,

48 "data": {"status": status, 'key': key_encrypt}}

49 order = json.dumps(comd)

50 s.sendto(bytes(order, encoding="utf-8"), self.ip_port_single)

51 data_bytes, addr = s.recvfrom(1024)

52 data_dic = json.loads(str(data_bytes, encoding='utf-8'))

53 s.close()

54 return data_dic

55

56 def read_all_sid(self):#读取所有设备

57 s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)

58 ls = json.loads(self.get_id_list().get('data'))

59 ls_sensor_state = []

60 for sid in ls:

61 comd = {'cmd': 'read', 'sid': sid}

62 order = json.dumps(comd)

63 s.sendto(bytes(order, encoding="utf-8"), self.ip_port_single)

64 data_bytes, addr = s.recvfrom(1024)

65 data_dic = json.loads(str(data_bytes, encoding='utf-8'))

66

67 ls_sensor_state.append(data_dic)

68 s.close()

69 return ls_sensor_state

70

71 def get_dict_model_sid(self):

72 dic_gw=self.whois()

73 ls=self.read_all_sid()

74 dic_model_sid = {}

75 for dic in ls:

76 model = dic.get('model')

77 sid = dic.get('sid')

78 dic_model_sid[model] = sid

79 dic_model_sid['gateway'] = dic_gw.get('sid')

80 return dic_model_sid

81

82

83 if __name__=='__main__':

84 import time

87 gw=udp_gw('你的网关ip')

88 tmp = gw.read_sid('子设备ID')

89 print(tmp)

完美!

得到json数据:

{'cmd': 'read_ack', 'model': 'sensor_ht', 'sid': '########', 'short_id': #####, 'data': '{"temperature":"3041","humidity":"5664"}'}

分别是:cmd:命令,model:设备,sid:设备ID,short_id:#,data: 温度,湿度,3041就是30.41度,5664就是湿度56.64%

python与传感器交互_Python-socket实现与小米传感器通信相关推荐

  1. python中和js交互_python和js交互调用的方法

    后台代码都是利用的 1.[get方式]使用jquery的get json与后台交互 前端js代码片段 var data= { 'a': $('input[name="a"]').v ...

  2. python与word交互_Python基础——与Python的指令交互

    Python是近几年兴起的一门编程语言,也是人工智能的主要语言,在TIOBE排行榜上占据着不错的位置,是许多程序员学习的对象,对于孩子来说,学习Python还是有些吃力的,我们需要从头开始一点点学起. ...

  3. python socket双向通信_Python socket实现多对多全双工通信的方法

    服务器:#server.py #!/usr/bin/env python #-*-coding:utf-8-*- import sys import struct#将字符串打包为二进制流进行网络传输 ...

  4. python多线程数据交互_python 多线程 通信

    一篇文章搞定Python多进程(全) 公众号:pythonislover 前面写了三篇关于python多线程的文章,大概概况了多线程使用中的方法,文章链接如下: 一篇文章搞懂Python多线程简单实现 ...

  5. 建立通讯录python增删改查_python 字典使用实例:创建通信录并完成修改、查找操作...

    练习目的:巩固python 字典的创建,合并,修改及使用. 学了python字典后,同学们想创建一个自己的通信录,小明是这么做的: 先根据三位舍友的联系方式创建一个字典dicTXL 然后将隔壁舍长已创 ...

  6. python实现邮件客户端_python socket实现web服务器,邮件客户端

    本博客是针对,<计算机网络自顶向下方法>一书第二章后面套接字编程作业, 所有代码均已上传至我的github:https://github.com/inspurer/ComputerNetw ...

  7. python web页面输出_python+socket+jq实现web页面实时输出结果

    例如有这样一个需求: 在终端上进行ping操作,现在想把这个这个操作放到web页面上进行,并且实现实时输出的效果. 来分析下具体实现过程 第一步,传统的http请求实现这个有点不太友好,因为这里边是一 ...

  8. python转换函数使用_Python Socket模块中的IP转换函数

    Python Socket模块中包含一些有用IP转换函数,说明如下: socket.ntohl(x) // 类似于C语言的ntohl(x) 把32位正整数从网络序转换成主机字节序. socket.nt ...

  9. python tcp服务器模板_python socket之tcp服务器与客户端示例

    Python中用socket编写服务器和客户端... 先说下我的问题,本来预想是,客户端连上服务器,然后向服务器发送信息服务器端代码 while 1: buf = s.recv(1024) 改成 wh ...

  10. 好玩的python代码聊天客户端_python socket实现客户端与服务器端对话

    最近巩固下python语言,说真的,使用了这么多脚本语言,还是这门比较好玩点,快速,简洁,容易上手更不用说了. socket 套接字是为特定网络协议(例如TCP/IP,ICMP/IP,UDP/IP等) ...

最新文章

  1. .NET Core 2.1预览分层编译特性
  2. AB1601GPIO不支持较高频率的脉冲中断
  3. 【文字识别小程序】快速识别文字,一款用了就再也离不开的宝藏神器~(出道即巅峰永久免费)
  4. Xamarin的坑 - 绑定(一) - 拿微信iOS SDK 简单说起
  5. springboot创建子模块时遇到子模块覆盖父模块问题解决
  6. IntersectionObserver对象
  7. 【数据结构和算法笔记】KMP算法介绍
  8. 11.策略模式(Strategy Pattern)
  9. java replace会替换吗,java替换字符_java replace字符替换函数
  10. virtualbox 搭建opnsense 防火墙
  11. 武汉市最好的计算机专科学校,武汉中专排名前十位_武汉最好的3加2学校_武汉中专技校有哪些_计算机平面设计_365导学网...
  12. [ArcGIS] Python计算Geometrical interval自然间断点 | jenkspy库
  13. 【爬虫】 静态网页抓取
  14. java namevaluepair_NameValuePair方式传参数
  15. 【js】:js中的确认框,提示框和确认框的使用
  16. tableau地图城市数据_Tableau 中的地图工作区
  17. 前端知识体系1:【css/js/vue/es6/手写/安全/优化】
  18. 德勤《全球AI发展白皮书》出炉!八大新趋势,三个关键技术一文扫尽【附下载】...
  19. Day4 - 函数、方法
  20. 配置TP5在Linux环境下域名映射tp5

热门文章

  1. 2022年CSDN粉丝总数前100名博主
  2. 【数据结构】链表相关OJ题 (万字详解)
  3. 火狐浏览器插件开发小试
  4. python图案绘制解锁_PythonAppium实现安卓手机图形解锁详解
  5. 前端三刺客---JS(基础语法)
  6. 2023认证杯论文格式注意事项
  7. 微软远程桌面升级到RDP8的方法
  8. 树莓派raspberry Pi 4B安装Ubuntu 20.04 LTS系统后如何连接WiFi
  9. OpenGL:gl_ClipDistance和gl_CullDiatance
  10. 972信息检索 | 第六章 专业性书目信息检索系统