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实现与小米网关通讯的更多相关文章

  1. python网络编程:TCP通讯模板、粘包及解决方案、自定义报头

    一.TCP通讯模板 二.远程CMD程序 三.解决粘包问题 四.解决粘包问题2 一.TCP通讯模板 TCP客户端 import socket c = socket.socket() # 连接服务器 c. ...

  2. [Python]实现XMPP协议即时通讯发送消息功能

    #-*- coding: utf-8 -*- __author__ = 'tsbc' import xmpp import time #注意帐号信息,必须加@域名格式 from_user = 'che ...

  3. Python基础(四) socket简单通讯

    socket:我们通常听过的套接字: 服务端: 1.创建socket对象 2.bing 绑定ip及端口 3.对该端口进行监听 4.消息阻塞(等待客户端消息) 客户端: 1.创建socket对象 2.连 ...

  4. python处理多线程之间事件通讯方法

    一.什么是事件 每执行一个事情,肯定有该事情的执行后状态,那事件就是该事情发生的信号 在程序中,多线程之间需要通讯,而事件就是方便线程之间的通讯 案例: 1.服务器启动需要5秒 2.客服端启动后去链接 ...

  5. 用Robot Framework+python来测试基于socket通讯的C/S系统(网络游戏)

    项目终于换了方案,改用socket来实现而不是之前的http了,所以测试工具就不能用以前的了,因为测试人手少,逼不得已的必须要挖掘更多的自动化方案来弥补.于是先研究了下python的socket解决方 ...

  6. 小米网关api

    http://bbs.xiaomi.cn/t-13198850 https://github.com/snOOrz/homebridge-aqara/blob/master/README.md htt ...

  7. TCP/IP协议,TCP与平台通信,通讯协议压力测试(python)

    最近的项目来了一个需求,要求测试tcp网关通讯协议: 1.液压井盖通过TCP/IP TCP与平台通信: 2.硬件定期发送心跳包(10S)给平台,是平台与硬件保持长连接: 3.每台硬件有一个12字节的唯 ...

  8. python中利用队列asyncio.Queue进行通讯详解

    python中利用队列asyncio.Queue进行通讯详解 本文主要给大家介绍了关于python用队列asyncio.Queue通讯的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细 ...

  9. 【XMLRPC实现跨语言编程】Tcl <----> python

    转载请声明出处,谢谢合作. # 期望一种能实现tcl.python两者解释器能双向通讯的结合体# py->tcl: from Tkinter import Tcl; tcl = Tcl(); t ...

随机推荐

  1. 关系型数据库工作原理-数据结构(翻译自Coding-Geek文章)

    本文翻译自Coding-Geek文章:< How does a relational database work>. 原文链接:http://coding-geek.com/how-dat ...

  2. epoll 实现回射服务器

    epoll是I/O复用模型中相对epoll和select更高效的实现对套接字管理的函数. epoll有两种模式 LT 和 ET 二者的差异在于 level-trigger 模式下只要某个 socket ...

  3. [sharepoint]rest api文档库文件上传,下载,拷贝,剪切,删除文件,创建文件夹,修改文件夹属性,删除文件夹,获取文档列表

    写在前面 最近对文档库的知识点进行了整理,也就有了这篇文章,当时查找这些接口,并用在实践中,确实废了一些功夫,也为了让更多的人走更少的弯路. 系列文章 sharepoint环境安装过程中几点需要注意的 ...

  4. 使用外置无线网卡来切换mac地址

    mac地址被別人過濾了!小樣,既然內置網卡不能修改mac,那我就用外置usb無線網卡,你以為你很厲害嗎,看我怎麼破解了你.

  5. window.open打开文件乱码

    问题:刚开始使用window.open在IE兼容模式下打开文件下载出现乱码. 一开始以为是文件名是中文导致的.然后使用a标签的download属性更改文件名解决. <a class=" ...

  6. c++ dynamic_cast 和 static_cast 的区别

    今天在看王道宝典的时候看到dynamic_cast ,一直都没用过,也不了解,今天来总结一下. dynamic_cast 和 static_cast 都可以用来强制转换指针类型,但不同的是dynami ...

  7. 笔记:Maven 生命周期与命令行详解

    Maven 拥有三套相互独立的生命周期,分别是 clean.default和site,clean 生命周期的目的是清理项目,default 生命周期的目的是构建项目,而site生命周期的目的是建立项目 ...

  8. centos7下更改docker镜像和容器的默认路径

    笔者近期在服务器上搭建docker环境,可由于笔者是普通用户,在安装的时候就跳了很多坑,现在记录一下. 一.docker权限问题 据官方解释,搭建docker环境必须使用root权限,或者sudo装, ...

  9. Algorithm --> 最长公共子序列(LCS)

      一.什么是最长公共子序列     什么是最长公共子序列呢?举个简单的例子吧,一个数列S,若分别是两个或多个已知序列的子序列,且是所有符合条件序列中最长的,则S称为已知序列的最长公共子序列. 举例如 ...

  10. mysql新手入门随笔3

    #求最高工资的员工信息 SELECT * FROM emp WHERE sal = (SELECT max(sal) FROM emp); #删除工资最低的员工信息 DELETE FROM emp W ...