IotCoreController.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. import os
  4. import hashlib
  5. import json
  6. import time
  7. import uuid
  8. import boto3
  9. import requests
  10. from django.views import View
  11. from Ansjer.config import BASE_DIR
  12. from base64 import b64encode, encodebytes
  13. from Controller.DeviceConfirmRegion import Device_Region
  14. from Model.models import Device_User, Device_Info, iotdeviceInfoModel, UIDCompanySerialModel, \
  15. SerialNumberModel
  16. from Object.IOTCore.IotObject import IOTClient
  17. from Object.ResponseObject import ResponseObject
  18. from Object.TokenObject import TokenObject
  19. from Service.CommonService import CommonService
  20. import OpenSSL.crypto as ct
  21. class IotCoreView(View):
  22. def get(self, request, *args, **kwargs):
  23. request.encoding = 'utf-8'
  24. request_dict = request.GET
  25. operation = kwargs.get('operation', None)
  26. return self.validate(operation, request_dict, request)
  27. def post(self, request, *args, **kwargs):
  28. request.encoding = 'utf-8'
  29. request_dict = request.POST
  30. operation = kwargs.get('operation', None)
  31. return self.validate(operation, request_dict, request)
  32. def validate(self, operation, request_dict, request):
  33. token = TokenObject(request_dict.get('token', None))
  34. response = ResponseObject()
  35. if operation == 'createKeysAndCertificate':
  36. return self.create_keys_and_certificate(request_dict, response, request)
  37. if operation == 'requestPublishMessage':
  38. return self.request_publish_message(request_dict, response, request)
  39. elif operation == 'thingRegroup':
  40. return self.thing_regroup(request_dict, response, request)
  41. else:
  42. if token.code != 0:
  43. return response.json(token.code)
  44. if operation == 'clearIotCerm':
  45. return self.clear_Iot_Cerm(request_dict, response)
  46. else:
  47. return response.json(404)
  48. # CVM注册 :正使用
  49. def create_keys_and_certificate(self, request_dict, response, request):
  50. serial_number = request_dict.get('serial_number', None)
  51. serial_number_code = request_dict.get('serial_number_code', None)
  52. token = request_dict.get('token', None)
  53. time_stamp = request_dict.get('time_stamp', None)
  54. device_version = request_dict.get('device_version', None).replace('.', '_') # 物品组命名不能包含'.'
  55. language = request_dict.get('language', None)
  56. if serial_number and token and time_stamp and serial_number_code and device_version and language:
  57. serial_number_code = CommonService.decode_data(serial_number_code)
  58. token = int(CommonService.decode_data(token))
  59. time_stamp = int(time_stamp)
  60. now_time = int(time.time())
  61. distance = now_time - time_stamp
  62. thingGroup = device_version + '_' + language
  63. if token != time_stamp or distance > 60000 or distance < -60000 or serial_number != serial_number_code: # 为了全球化时间控制在一天内
  64. return response.json(404)
  65. serial = serial_number[0:6]
  66. iotqs = iotdeviceInfoModel.objects.filter(serial_number=serial)
  67. # 判断设备是否已注册证书
  68. if not iotqs.exists():
  69. ip = CommonService.get_ip_address(request)
  70. region_id = Device_Region().get_device_region(ip)
  71. iotClient = IOTClient(region_id)
  72. res = iotClient.create_keys_and_certificate(serial_number, thingGroup, response)
  73. nowTime = int(time.time())
  74. token_iot_number = hashlib.md5((str(uuid.uuid1()) + str(nowTime)).encode('utf-8')).hexdigest()
  75. # sn = SerialNumberModel.objects.get(serial_number=serial)
  76. try:
  77. sn = SerialNumberModel.objects.get(serial_number=serial)
  78. except:
  79. return response.json(444)
  80. iotdeviceInfoModel.objects.create(serial_number=serial,
  81. endpoint=res[0]['endpoint'],
  82. certificate_id=res[0]['certificateId'],
  83. certificate_pem=res[0]['certificatePem'],
  84. public_key=res[0]['publicKey'],
  85. private_key=res[0]['privateKey'],
  86. thing_name=res[1]['ThingName'],
  87. token_iot_number=token_iot_number
  88. )
  89. res = {
  90. 'certificateId': res[0]['certificateId'],
  91. 'certificatePem': res[0]['certificatePem'],
  92. 'publicKey': res[0]['publicKey'],
  93. 'privateKey': res[0]['privateKey'],
  94. 'endpoint': res[0]['endpoint']
  95. }
  96. return response.json(0, {'res': res})
  97. else:
  98. iot = iotqs[0]
  99. res = {
  100. 'certificateId': iot.certificate_id,
  101. 'certificatePem': iot.certificate_pem,
  102. 'publicKey': iot.public_key,
  103. 'privateKey': iot.private_key,
  104. 'endpoint': iot.endpoint
  105. }
  106. # print('此设备已注册证书')
  107. return response.json(0, {'res': res})
  108. else:
  109. return response.json(444)
  110. def thing_regroup(self, request_dict, response, request):
  111. # 物品重新分组
  112. token = request_dict.get('token', None)
  113. time_stamp = request_dict.get('time_stamp', None)
  114. serial_number = request_dict.get('serial_number', None)
  115. device_version = request_dict.get('device_version', None)
  116. language = request_dict.get('language', None)
  117. if not all([serial_number, device_version, language, token, time_stamp]):
  118. return response.json(444)
  119. # 封装token认证
  120. token = int(CommonService.decode_data(token))
  121. time_stamp = int(time_stamp)
  122. now_time = int(time.time())
  123. distance = now_time - time_stamp
  124. if token != time_stamp or distance > 60000 or distance < -60000: # 为了全球化时间控制在一天内
  125. return response.json(404)
  126. ip = CommonService.get_ip_address(request)
  127. region_id = Device_Region().get_device_region(ip)
  128. thingName = 'Ansjer_Device_' + serial_number
  129. new_thingGroupName = (device_version + '_' + language).replace('.', '_') # 物品组命名不能包含'.'
  130. # 调试参数
  131. # thingName = 'Ansjer_Device_00EBEX'
  132. # new_thingGroupName = 'C1Pro_V1_0_1_cn'
  133. iotClient = IOTClient(region_id)
  134. try:
  135. # 获取旧物品组
  136. list_groups_res = iotClient.client.list_thing_groups_for_thing(thingName=thingName, maxResults=1)
  137. old_thingGroupName = list_groups_res['thingGroups'][0]['groupName']
  138. # 没有新物品组则创建
  139. list_thing_groups_res = iotClient.client.list_thing_groups(namePrefixFilter=new_thingGroupName
  140. , maxResults=1, recursive=False)
  141. if not list_thing_groups_res['thingGroups']:
  142. attributes = {
  143. "update_time": "0"
  144. }
  145. thingGroupProperties = {
  146. "thingGroupDescription": "OTA",
  147. "attributePayload": {
  148. "attributes": attributes,
  149. "merge": False # 更新时覆盖掉而不是合并
  150. }
  151. }
  152. iotClient.client.create_thing_group(thingGroupName=new_thingGroupName
  153. , thingGroupProperties=thingGroupProperties)
  154. iotClient.client.update_thing_groups_for_thing(thingName=thingName
  155. , thingGroupsToAdd=[new_thingGroupName]
  156. , thingGroupsToRemove=[old_thingGroupName])
  157. # 更新设备版本信息
  158. Device_Info.objects.filter(serial_number=serial_number).update(version=device_version)
  159. return response.json(0)
  160. except Exception as e:
  161. print(e)
  162. return response.json(500, repr(e))
  163. def clear_Iot_Cerm(self, userID, request_dict, response):
  164. serial_number = request_dict.get('serial_number', None)
  165. if serial_number:
  166. iot = iotdeviceInfoModel.objects.filter(thing_name="Ansjer_Device_" + serial_number)
  167. if iot.exists():
  168. iot.delete()
  169. return response.json(0)
  170. else:
  171. return response.json(444)
  172. def request_publish_message(self, request_dict, response, request):
  173. UID = request_dict.get('UID', None)
  174. MSG = request_dict.get('MSG', None)
  175. if not all([UID, MSG]):
  176. return response.json(444)
  177. try:
  178. # 获取检查uid的序列号,如果没有序列号,不使用MQTT下发消息
  179. device_info_qs = Device_Info.objects.filter(UID=UID).values('serial_number')
  180. serial_number = device_info_qs[0]['serial_number']
  181. if serial_number == '':
  182. return response.json(10043)
  183. # 获取数据组织将要请求的url
  184. thing_name = 'Ansjer_Device_' + serial_number
  185. iot = iotdeviceInfoModel.objects.filter(thing_name__contains=thing_name).values('thing_name', 'endpoint',
  186. 'token_iot_number')
  187. thing_name = iot[0]['thing_name'] # IoT core上的物品名: Ansjer_Device_+序列号+企业编码
  188. endpoint = iot[0]['endpoint']
  189. Token = iot[0]['token_iot_number']
  190. # Token = '297a601b3925e04daab5a60280650e09'
  191. topic_name = thing_name + '_rtsp_topic'
  192. # rtsp://rtsp.zositech.org:8554/ZFdqWldXRFpMTkVaYVZEaEJXRXhUV0RFeE1VRT1B
  193. # api doc: https://docs.aws.amazon.com/zh_cn/iot/latest/developerguide/http.html
  194. # https://IoT_data_endpoint/topics/url_encoded_topic_name?qos=1
  195. # POST请求url来发布消息
  196. url = 'https://{}/topics/{}?rtsp_command={}'.format(endpoint, topic_name, MSG)
  197. authorizer_name = 'Ansjer_Iot_Auth'
  198. signature = self.rsa_sign(Token) # Token签名
  199. headers = {'x-amz-customauthorizer-name': authorizer_name, 'Token': Token,
  200. 'x-amz-customauthorizer-signature': signature}
  201. params = {'rtsp_command': MSG}
  202. r = requests.post(url=url, headers=headers, json=params, timeout=2)
  203. if r.status_code == 200:
  204. return response.json(0)
  205. else:
  206. # print('发布失败')
  207. return response.json(10044)
  208. except Exception as e:
  209. # print(e)
  210. return response.json(500, repr(e))
  211. def rsa_sign(self, Token):
  212. # 私钥签名Token
  213. private_key_file = '''-----BEGIN RSA PRIVATE KEY-----
  214. MIIEpQIBAAKCAQEA5iJzEDPqtGmFMggekVro6C0lrjuC2BjunGkrFNJWpDYzxCzE
  215. X5jf4/Fq7hcIaQd5sqHugDxPVollSLPe9zNilbrd0sZfU+Ed8gRVuKW9KwfE9XFr
  216. L0pt6bKRQ0IIRfiZ9TuR0tsQysvcO1GZSXcYfPue3tGM1zOnWFThWDqZ06+sOxzt
  217. RMRl4yNfbpCG4MfxG3itNXOfrjZv2OMLSXrxmzubSvRpUYSvQPs4fm9302SAnySY
  218. 0MKzx6H6528ZQm/IDDSZy6EmNBIyTRDfxC56vnYcXvqedAQh7jJnjdvt6Q4MhASH
  219. eIYi1FBSdu2NT6wgpnrqXzx5pq9kR/lnsLID0wIDAQABAoIBAQCiF4GT1/1oNSpr
  220. ouxk1PNXFPWFUsVGD8mAwVJmx//eiY7MjfuCmdqYYmI+cFqsH2fIOeYSzGfVO9Dq
  221. 9EYHN1oovAWhf7eFDPpajFMUSyiCNmazub8VAAeKowtNpCTPo9pMsDh1m3aoYA4u
  222. ebrN0+Sbo16y8kWRDgDAZoiR7DSMs8lczk16hwfv5mw8XpNDbaL3Coi4Koe2S1Yh
  223. 2SX3vWFlpd7qF1ZYXuZIp+b8JPrV7n9eUKoFgzj0gqgwQK80CoexIjiOrNMPvkQa
  224. q+8kCvFjAzKxOK7e8gjM8lMRiGodb61kmYZkkJzFwWO4EaGbl34lfVECd1Ixp3tF
  225. be0OWAGBAoGBAPSteXDzzToD8ovM7LL11x0jWwI6HOiHu89kZtW566rIezjWBuA2
  226. TxrcYKM3h9jQRXS3CsMdoIv6XGk5lqM8ADtjn23FBWe/THYLh8bm8JOgh5RRWQDg
  227. SvkLfi9Ih2mM4NJfmuuDOh3Nze2efLM7+kOZWUQwF2Zx9mL5jvRBk351AoGBAPDI
  228. sYmT2Li+i5+0vykA2m5uPF8ZOW8BGtAfCZv0suW7BNzSgin78g9WapRd/4p0NNiL
  229. /nVMqPPCpd1akCUpV+GDWQt0hV+HZjxANE0KWhciQRyo2qvo51j8SWILJSgh0tXC
  230. aTF8qt6oGw3VN3m57vKhbrlDaz0J/NDJFci6msAnAoGBAOuG6bXPGijUj+//DYKf
  231. n7jOxdZ49kboEePrtAncdHzri6IEdI3z+WXT6bpzw/LzWUimwldb96WHFNm9s8Hi
  232. Ch8hIODbnP5naUTgiIzw1XhmONyPCewL/F+LrqX5XVA/alNX8JrwsUrrR2WLAGLQ
  233. Q3I69XDsEjptTU2tCO0bCs3ZAoGBAJ2lCHfm0JHET230zONvp5N9oREyVqQSuRdh
  234. +syc3TQDyh85w/bw+X6JOaaCFHj1tFPC9Iqf8k4GNspCLPXnp54CfR4+38O3xnvU
  235. HWoDSRC0YKT++IxtJGriYrlKSr2Hx54kdvLriIPW1D+uRW/xCDza7L9nIKMKEvgv
  236. b4/IfOEpAoGAeKM9Te7T1VzlAkS0CJOwanzwYV/zrex84WuXxlsGgPQ871lTs5AP
  237. H1QLfLfFXH+UVrCEC2yv4eml/cqFkpB3gE5i4MQ8GPVIOSs5tsIyl8YUA03vdNdB
  238. GCqvlyw5dfxNA+EtxNE2wCW/LW7ENJlACgcfgPlBZtpLheWoZB/maw4=
  239. -----END RSA PRIVATE KEY-----'''
  240. # 使用密钥文件方式
  241. # private_key_file_path = os.path.join(BASE_DIR, 'static/iotCore/private.pem')#.replace('\\', '/')
  242. # private_key_file = open(private_key_file_path, 'r')
  243. private_key = ct.load_privatekey(ct.FILETYPE_PEM, private_key_file)
  244. signature = ct.sign(private_key, Token.encode('utf8'), 'sha256')
  245. signature = encodebytes(signature).decode('utf8').replace('\n', '')
  246. # print('signature:', signature)
  247. return signature