IotCoreController.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  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, AWS_IOT_GETS3_PULL_CHINA_ID, AWS_IOT_GETS3_PULL_CHINA_SECRET, \
  12. AWS_IOT_GETS3_PULL_FOREIGN_ID, AWS_IOT_GETS3_PULL_FOREIGN_SECRET, AWS_ARN
  13. from base64 import b64encode, encodebytes
  14. from Controller.DeviceConfirmRegion import Device_Region
  15. from Model.models import Device_User, Device_Info, iotdeviceInfoModel, UIDCompanySerialModel, \
  16. SerialNumberModel
  17. from Object.IOTCore.IotObject import IOTClient
  18. from Object.ResponseObject import ResponseObject
  19. from Object.TokenObject import TokenObject
  20. from Service.CommonService import CommonService
  21. import OpenSSL.crypto as ct
  22. class IotCoreView(View):
  23. def get(self, request, *args, **kwargs):
  24. request.encoding = 'utf-8'
  25. request_dict = request.GET
  26. operation = kwargs.get('operation', None)
  27. return self.validate(operation, request_dict, request)
  28. def post(self, request, *args, **kwargs):
  29. request.encoding = 'utf-8'
  30. request_dict = request.POST
  31. operation = kwargs.get('operation', None)
  32. return self.validate(operation, request_dict, request)
  33. def validate(self, operation, request_dict, request):
  34. response = ResponseObject()
  35. lang = request_dict.get('lang', 'en')
  36. response.lang = lang
  37. if operation == 'createKeysAndCertificate':
  38. return self.create_keys_and_certificate(request_dict, response, request)
  39. elif operation == 'requestPublishMessage':
  40. return self.request_publish_message(request_dict, response, request)
  41. elif operation == 'getS3PullKey':
  42. return self.get_s3_pull_key(request_dict, response, request)
  43. elif operation == 'thingRegroup':
  44. return self.thing_regroup(request_dict, response, request)
  45. else:
  46. token = TokenObject(request_dict.get('token', None))
  47. if token.code != 0:
  48. return response.json(token.code)
  49. response.lang = token.lang
  50. if operation == 'clearIotCerm':
  51. return self.clear_Iot_Cerm(request_dict, response)
  52. else:
  53. return response.json(404)
  54. # CVM注册 :正使用
  55. def create_keys_and_certificate(self, request_dict, response, request):
  56. uid = request_dict.get('uid', '')
  57. token = request_dict.get('token', None)
  58. uid_code = request_dict.get('uid_code', None)
  59. language = request_dict.get('language', None)
  60. time_stamp = request_dict.get('time_stamp', None)
  61. device_version = request_dict.get('device_version', None).replace('.', '_') # 物品组命名不能包含'.'
  62. if not all([token, time_stamp, device_version, language]):
  63. return response.json(444, {'param': 'token, uid_code, time_stamp, device_version, language'})
  64. # token时间戳校验
  65. token = int(CommonService.decode_data(token))
  66. time_stamp = int(time_stamp)
  67. now_time = int(time.time())
  68. distance = now_time - time_stamp
  69. if token != time_stamp or distance > 60000 or distance < -60000: # 为了全球化时间控制在一天内
  70. return response.json(404)
  71. if not uid:
  72. # 使用序列号
  73. serial_number = request_dict.get('serial_number', None)
  74. serial_number_code = request_dict.get('serial_number_code', None)
  75. if not all([serial_number, serial_number_code]):
  76. return response.json(444, {'param': 'serial_number, serial_number_code'})
  77. # 序列号编码解码校验
  78. serial_number_code = CommonService.decode_data(serial_number_code)
  79. if serial_number != serial_number_code:
  80. return response.json(404)
  81. serial = serial_number[0:6]
  82. try:
  83. SerialNumberModel.objects.get(serial_number=serial)
  84. except:
  85. return response.json(444)
  86. ThingNameSuffix = serial_number # 物品名后缀
  87. iotdeviceInfo_qs = iotdeviceInfoModel.objects.filter(serial_number=serial)
  88. else:
  89. # 使用uid
  90. # uid编码解码校验
  91. uid_code = CommonService.decode_data(uid_code)
  92. if uid != uid_code:
  93. return response.json(404)
  94. serial = '' # iot_deviceInfo表写入serial_number为''
  95. ThingNameSuffix = uid # 物品名后缀
  96. iotdeviceInfo_qs = iotdeviceInfoModel.objects.filter(uid=uid)
  97. # 判断设备是否已注册证书
  98. if not iotdeviceInfo_qs.exists():
  99. thingGroup = device_version + '_' + language
  100. ip = CommonService.get_ip_address(request)
  101. region_id = Device_Region().get_device_region(ip)
  102. iotClient = IOTClient(region_id)
  103. res = iotClient.create_keys_and_certificate(ThingNameSuffix, thingGroup, response)
  104. token_iot_number = hashlib.md5((str(uuid.uuid1()) + str(now_time)).encode('utf-8')).hexdigest()
  105. iotdeviceInfoModel.objects.create(uid=uid,
  106. serial_number=serial,
  107. endpoint=res[0]['endpoint'],
  108. certificate_id=res[0]['certificateId'],
  109. certificate_pem=res[0]['certificatePem'],
  110. public_key=res[0]['publicKey'],
  111. private_key=res[0]['privateKey'],
  112. thing_name=res[1]['ThingName'],
  113. thing_groups=res[1]['thingGroupName'],
  114. token_iot_number=token_iot_number
  115. )
  116. res = {
  117. 'certificateId': res[0]['certificateId'],
  118. 'certificatePem': res[0]['certificatePem'],
  119. 'publicKey': res[0]['publicKey'],
  120. 'privateKey': res[0]['privateKey'],
  121. 'endpoint': res[0]['endpoint']
  122. }
  123. return response.json(0, {'res': res})
  124. else:
  125. iot = iotdeviceInfo_qs[0]
  126. res = {
  127. 'certificateId': iot.certificate_id,
  128. 'certificatePem': iot.certificate_pem,
  129. 'publicKey': iot.public_key,
  130. 'privateKey': iot.private_key,
  131. 'endpoint': iot.endpoint
  132. }
  133. # print('此设备已注册证书')
  134. return response.json(0, {'res': res})
  135. def thing_regroup(self, request_dict, response, request):
  136. # 物品重新分组
  137. uid = request_dict.get('uid', '')
  138. token = request_dict.get('token', None)
  139. language = request_dict.get('language', None)
  140. time_stamp = request_dict.get('time_stamp', None)
  141. device_version = request_dict.get('device_version', None)
  142. if not all([token, language, time_stamp, device_version]):
  143. return response.json(444, {'param: token, language, time_stamp, device_version'})
  144. # 封装token认证
  145. token = int(CommonService.decode_data(token))
  146. time_stamp = int(time_stamp)
  147. now_time = int(time.time())
  148. distance = now_time - time_stamp
  149. if token != time_stamp or distance > 60000 or distance < -60000: # 为了全球化时间控制在一天内
  150. return response.json(404)
  151. ip = CommonService.get_ip_address(request)
  152. region_id = Device_Region().get_device_region(ip)
  153. iotClient = IOTClient(region_id)
  154. ThingNameSuffix = uid
  155. if not uid:
  156. # 使用序列号
  157. serial_number = request_dict.get('serial_number', None)
  158. if not serial_number:
  159. return response.json(444)
  160. ThingNameSuffix = serial_number
  161. thingName = 'Ansjer_Device_' + ThingNameSuffix
  162. new_thingGroupName = (device_version + '_' + language).replace('.', '_') # 物品组命名不能包含'.'
  163. try:
  164. # 获取旧物品组
  165. list_groups_res = iotClient.client.list_thing_groups_for_thing(thingName=thingName, maxResults=1)
  166. old_thingGroupName = list_groups_res['thingGroups'][0]['groupName']
  167. # 没有新物品组则创建
  168. list_thing_groups_res = iotClient.client.list_thing_groups(namePrefixFilter=new_thingGroupName
  169. , maxResults=1, recursive=False)
  170. if not list_thing_groups_res['thingGroups']:
  171. attributes = {
  172. "update_time": "0"
  173. }
  174. thingGroupProperties = {
  175. "thingGroupDescription": "OTA",
  176. "attributePayload": {
  177. "attributes": attributes,
  178. "merge": False # 更新时覆盖掉而不是合并
  179. }
  180. }
  181. iotClient.client.create_thing_group(thingGroupName=new_thingGroupName
  182. , thingGroupProperties=thingGroupProperties)
  183. iotClient.client.update_thing_groups_for_thing(thingName=thingName
  184. , thingGroupsToAdd=[new_thingGroupName]
  185. , thingGroupsToRemove=[old_thingGroupName])
  186. # 更新设备版本信息
  187. if uid:
  188. Device_Info.objects.filter(UID=uid).update(version=device_version)
  189. else:
  190. Device_Info.objects.filter(serial_number=serial_number).update(version=device_version)
  191. return response.json(0)
  192. except Exception as e:
  193. print(e)
  194. return response.json(500, repr(e))
  195. def clear_Iot_Cerm(self, userID, request_dict, response):
  196. serial_number = request_dict.get('serial_number', None)
  197. if serial_number:
  198. iot = iotdeviceInfoModel.objects.filter(thing_name="Ansjer_Device_" + serial_number)
  199. if iot.exists():
  200. iot.delete()
  201. return response.json(0)
  202. else:
  203. return response.json(444)
  204. def request_publish_message(self, request_dict, response, request):
  205. # Alexa请求IoT Core下发MQTT消息通知设备开始或停止推流,或唤醒设备
  206. UID = request_dict.get('UID', None)
  207. MSG = request_dict.get('MSG', None)
  208. if not all([UID, MSG]):
  209. return response.json(444)
  210. try:
  211. # 获取检查uid的序列号,如果没有序列号,不使用MQTT下发消息
  212. device_info_qs = Device_Info.objects.filter(UID=UID).values('UID', 'serial_number')
  213. if not device_info_qs.exists():
  214. return response.json(10043)
  215. uid = device_info_qs[0]['UID']
  216. serial_number = device_info_qs[0]['serial_number']
  217. # 如果device_info表的serial_number不为空,物品名为'Ansjer_Device_序列号'
  218. thing_name_suffix = serial_number if serial_number != '' else uid
  219. # 获取数据组织将要请求的url
  220. iot = iotdeviceInfoModel.objects.filter(thing_name__contains=thing_name_suffix).values('thing_name',
  221. 'endpoint',
  222. 'token_iot_number')
  223. if not iot.exists():
  224. return response.json(10043)
  225. thing_name = iot[0]['thing_name'][14:] # IoT core上的物品名: Ansjer_Device_ + 序列号+企业编码/uid
  226. endpoint = iot[0]['endpoint']
  227. Token = iot[0]['token_iot_number']
  228. # Token = '297a601b3925e04daab5a60280650e09'
  229. topic_suffix = '_power_topic' if 'Turn' in MSG else '_rtsp_topic'
  230. topic_name = thing_name + topic_suffix # MQTT主题
  231. # api doc: https://docs.aws.amazon.com/zh_cn/iot/latest/developerguide/http.html
  232. # url: https://IoT_data_endpoint/topics/url_encoded_topic_name?qos=1
  233. # post请求url来发布MQTT消息
  234. url = 'https://{}/topics/{}'.format(endpoint, topic_name)
  235. authorizer_name = 'Ansjer_Iot_Auth'
  236. signature = self.rsa_sign(Token) # Token签名
  237. headers = {'x-amz-customauthorizer-name': authorizer_name, 'Token': Token,
  238. 'x-amz-customauthorizer-signature': signature}
  239. params = {'command': MSG}
  240. r = requests.post(url=url, headers=headers, json=params, timeout=2)
  241. if r.status_code == 200:
  242. res = r.json()
  243. if res['message'] == 'OK':
  244. return response.json(0)
  245. return response.json(10044)
  246. else:
  247. # print('发布失败')
  248. return response.json(10044)
  249. except Exception as e:
  250. # print(e)
  251. return response.json(500, repr(e))
  252. def request_publish_mqtt(self, request_dict, response, request):
  253. # 通用发布主题通知
  254. UID = request_dict.get('UID', None)
  255. MSG = request_dict.get('MSG', None)
  256. return_topic_name = request_dict.get('return_topic_name', None)
  257. if not all([UID, MSG]):
  258. return response.json(444)
  259. try:
  260. # 获取检查uid的序列号,如果没有序列号,不使用MQTT下发消息
  261. device_info_qs = Device_Info.objects.filter(UID=UID).values('UID', 'serial_number')
  262. if not device_info_qs.exists():
  263. return response.json(10043)
  264. uid = device_info_qs[0]['UID']
  265. serial_number = device_info_qs[0]['serial_number']
  266. # 如果device_info表的serial_number不为空,物品名为'Ansjer_Device_序列号'
  267. thing_name_suffix = serial_number if serial_number != '' else uid
  268. # 获取数据组织将要请求的url
  269. iot = iotdeviceInfoModel.objects.filter(thing_name__contains=thing_name_suffix).values('thing_name',
  270. 'endpoint',
  271. 'token_iot_number')
  272. if not iot.exists():
  273. return response.json(10043)
  274. thing_name = iot[0]['thing_name'][14:] # IoT core上的物品名: Ansjer_Device_ + 序列号+企业编码/uid
  275. endpoint = iot[0]['endpoint']
  276. Token = iot[0]['token_iot_number']
  277. # Token = '297a601b3925e04daab5a60280650e09'
  278. topic_name = thing_name + return_topic_name # MQTT主题
  279. if return_topic_name == 'get_s3_key_return':
  280. MSG = self.get_s3_key_return_msg(endpoint)
  281. # api doc: https://docs.aws.amazon.com/zh_cn/iot/latest/developerguide/http.html
  282. # url: https://IoT_data_endpoint/topics/url_encoded_topic_name?qos=1
  283. # post请求url来发布MQTT消息
  284. url = 'https://{}/topics/{}'.format(endpoint, topic_name)
  285. authorizer_name = 'Ansjer_Iot_Auth'
  286. signature = self.rsa_sign(Token) # Token签名
  287. headers = {'x-amz-customauthorizer-name': authorizer_name, 'Token': Token,
  288. 'x-amz-customauthorizer-signature': signature}
  289. params = {'command': MSG}
  290. r = requests.post(url=url, headers=headers, json=params, timeout=2)
  291. if r.status_code == 200:
  292. res = r.json()
  293. if res['message'] == 'OK':
  294. return response.json(0)
  295. return response.json(10044)
  296. else:
  297. # print('发布失败')
  298. return response.json(10044)
  299. except Exception as e:
  300. # print(e)
  301. return response.json(500, repr(e))
  302. def get_s3_pull_key(self, request_dict, response, request):
  303. # 通用发布主题通知
  304. UID = request_dict.get('UID', None)
  305. if not all([UID]):
  306. return response.json(444)
  307. try:
  308. # 获取检查uid的序列号,如果没有序列号,不使用MQTT下发消息
  309. device_info_qs = Device_Info.objects.filter(UID=UID).values('UID', 'serial_number')
  310. if not device_info_qs.exists():
  311. return response.json(10043)
  312. uid = device_info_qs[0]['UID']
  313. serial_number = device_info_qs[0]['serial_number']
  314. # 如果device_info表的serial_number不为空,物品名为'Ansjer_Device_序列号'
  315. thing_name_suffix = serial_number if serial_number != '' else uid
  316. # 获取数据组织将要请求的url
  317. iot = iotdeviceInfoModel.objects.filter(thing_name__contains=thing_name_suffix).values('thing_name',
  318. 'endpoint',
  319. 'token_iot_number')
  320. if not iot.exists():
  321. return response.json(10043)
  322. endpoint = iot[0]['endpoint']
  323. MSG = self.get_s3_key_return_msg(endpoint)
  324. return response.json(MSG)
  325. except Exception as e:
  326. # print(e)
  327. return response.json(500, repr(e))
  328. def get_s3_key_return_msg(self,endpoint):
  329. MSG = {}
  330. if endpoint == 'cn-northwest-1':
  331. key = AWS_IOT_GETS3_PULL_CHINA_ID
  332. secret = AWS_IOT_GETS3_PULL_CHINA_SECRET
  333. arn = AWS_ARN[0]
  334. else:
  335. key = AWS_IOT_GETS3_PULL_FOREIGN_ID
  336. secret = AWS_IOT_GETS3_PULL_FOREIGN_SECRET
  337. arn = AWS_ARN[1]
  338. MSG['AccessKeyId'] = key
  339. MSG['AccessKeySecret'] = secret
  340. MSG['endpoint'] = endpoint
  341. MSG['bucket_name'] = 'asj-log'
  342. MSG['arn'] = arn
  343. return MSG
  344. def rsa_sign(self, Token):
  345. # 私钥签名Token
  346. private_key_file = '''-----BEGIN RSA PRIVATE KEY-----
  347. MIIEpQIBAAKCAQEA5iJzEDPqtGmFMggekVro6C0lrjuC2BjunGkrFNJWpDYzxCzE
  348. X5jf4/Fq7hcIaQd5sqHugDxPVollSLPe9zNilbrd0sZfU+Ed8gRVuKW9KwfE9XFr
  349. L0pt6bKRQ0IIRfiZ9TuR0tsQysvcO1GZSXcYfPue3tGM1zOnWFThWDqZ06+sOxzt
  350. RMRl4yNfbpCG4MfxG3itNXOfrjZv2OMLSXrxmzubSvRpUYSvQPs4fm9302SAnySY
  351. 0MKzx6H6528ZQm/IDDSZy6EmNBIyTRDfxC56vnYcXvqedAQh7jJnjdvt6Q4MhASH
  352. eIYi1FBSdu2NT6wgpnrqXzx5pq9kR/lnsLID0wIDAQABAoIBAQCiF4GT1/1oNSpr
  353. ouxk1PNXFPWFUsVGD8mAwVJmx//eiY7MjfuCmdqYYmI+cFqsH2fIOeYSzGfVO9Dq
  354. 9EYHN1oovAWhf7eFDPpajFMUSyiCNmazub8VAAeKowtNpCTPo9pMsDh1m3aoYA4u
  355. ebrN0+Sbo16y8kWRDgDAZoiR7DSMs8lczk16hwfv5mw8XpNDbaL3Coi4Koe2S1Yh
  356. 2SX3vWFlpd7qF1ZYXuZIp+b8JPrV7n9eUKoFgzj0gqgwQK80CoexIjiOrNMPvkQa
  357. q+8kCvFjAzKxOK7e8gjM8lMRiGodb61kmYZkkJzFwWO4EaGbl34lfVECd1Ixp3tF
  358. be0OWAGBAoGBAPSteXDzzToD8ovM7LL11x0jWwI6HOiHu89kZtW566rIezjWBuA2
  359. TxrcYKM3h9jQRXS3CsMdoIv6XGk5lqM8ADtjn23FBWe/THYLh8bm8JOgh5RRWQDg
  360. SvkLfi9Ih2mM4NJfmuuDOh3Nze2efLM7+kOZWUQwF2Zx9mL5jvRBk351AoGBAPDI
  361. sYmT2Li+i5+0vykA2m5uPF8ZOW8BGtAfCZv0suW7BNzSgin78g9WapRd/4p0NNiL
  362. /nVMqPPCpd1akCUpV+GDWQt0hV+HZjxANE0KWhciQRyo2qvo51j8SWILJSgh0tXC
  363. aTF8qt6oGw3VN3m57vKhbrlDaz0J/NDJFci6msAnAoGBAOuG6bXPGijUj+//DYKf
  364. n7jOxdZ49kboEePrtAncdHzri6IEdI3z+WXT6bpzw/LzWUimwldb96WHFNm9s8Hi
  365. Ch8hIODbnP5naUTgiIzw1XhmONyPCewL/F+LrqX5XVA/alNX8JrwsUrrR2WLAGLQ
  366. Q3I69XDsEjptTU2tCO0bCs3ZAoGBAJ2lCHfm0JHET230zONvp5N9oREyVqQSuRdh
  367. +syc3TQDyh85w/bw+X6JOaaCFHj1tFPC9Iqf8k4GNspCLPXnp54CfR4+38O3xnvU
  368. HWoDSRC0YKT++IxtJGriYrlKSr2Hx54kdvLriIPW1D+uRW/xCDza7L9nIKMKEvgv
  369. b4/IfOEpAoGAeKM9Te7T1VzlAkS0CJOwanzwYV/zrex84WuXxlsGgPQ871lTs5AP
  370. H1QLfLfFXH+UVrCEC2yv4eml/cqFkpB3gE5i4MQ8GPVIOSs5tsIyl8YUA03vdNdB
  371. GCqvlyw5dfxNA+EtxNE2wCW/LW7ENJlACgcfgPlBZtpLheWoZB/maw4=
  372. -----END RSA PRIVATE KEY-----'''
  373. # 使用密钥文件方式
  374. # private_key_file_path = os.path.join(BASE_DIR, 'static/iotCore/private.pem')#.replace('\\', '/')
  375. # private_key_file = open(private_key_file_path, 'r')
  376. private_key = ct.load_privatekey(ct.FILETYPE_PEM, private_key_file)
  377. signature = ct.sign(private_key, Token.encode('utf8'), 'sha256')
  378. signature = encodebytes(signature).decode('utf8').replace('\n', '')
  379. # print('signature:', signature)
  380. return signature