IotCoreController.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357
  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, AWS_IOT_SES_ACCESS_CHINA_REGION, \
  13. AWS_IOT_SES_ACCESS_FOREIGN_REGION_ASIA, AWS_IOT_SES_ACCESS_FOREIGN_REGION_EUROPE, \
  14. AWS_IOT_SES_ACCESS_FOREIGN_REGION_AMERICA, SERVER_TYPE
  15. from base64 import b64encode, encodebytes
  16. from Controller.DeviceConfirmRegion import Device_Region
  17. from Model.models import Device_User, Device_Info, iotdeviceInfoModel, UIDCompanySerialModel, \
  18. SerialNumberModel
  19. from Object.IOTCore.IotObject import IOTClient
  20. from Object.ResponseObject import ResponseObject
  21. from Object.TokenObject import TokenObject
  22. from Service.CommonService import CommonService
  23. import OpenSSL.crypto as ct
  24. class IotCoreView(View):
  25. def get(self, request, *args, **kwargs):
  26. request.encoding = 'utf-8'
  27. request_dict = request.GET
  28. operation = kwargs.get('operation', None)
  29. return self.validate(operation, request_dict, request)
  30. def post(self, request, *args, **kwargs):
  31. request.encoding = 'utf-8'
  32. request_dict = request.POST
  33. operation = kwargs.get('operation', None)
  34. return self.validate(operation, request_dict, request)
  35. def validate(self, operation, request_dict, request):
  36. response = ResponseObject()
  37. lang = request_dict.get('lang', 'en')
  38. response.lang = lang
  39. if operation == 'createKeysAndCertificate':
  40. return self.create_keys_and_certificate(request_dict, response, request)
  41. elif operation == 'requestPublishMessage':
  42. return self.request_publish_message(request_dict, response, request)
  43. elif operation == 'getS3PullKey':
  44. return self.get_s3_pull_key(request_dict, response, request)
  45. elif operation == 'thingRegroup':
  46. return self.thing_regroup(request_dict, response, request)
  47. else:
  48. token = TokenObject(request_dict.get('token', None))
  49. if token.code != 0:
  50. return response.json(token.code)
  51. response.lang = token.lang
  52. if operation == 'clearIotCerm':
  53. return self.clear_Iot_Cerm(request_dict, response)
  54. elif operation == 'getIotInfo':
  55. return self.getIotInfo(request_dict, response)
  56. else:
  57. return response.json(404)
  58. # CVM注册 :正使用
  59. def create_keys_and_certificate(self, request_dict, response, request):
  60. uid = request_dict.get('uid', '')
  61. token = request_dict.get('token', None)
  62. uid_code = request_dict.get('uid_code', None)
  63. language = request_dict.get('language', None)
  64. time_stamp = request_dict.get('time_stamp', None)
  65. device_version = request_dict.get('device_version', None).replace('.', '_') # 物品组命名不能包含'.'
  66. if not all([token, time_stamp, device_version, language]):
  67. return response.json(444, {'param': 'token, uid_code, time_stamp, device_version, language'})
  68. # 时间戳token校验
  69. if not CommonService.check_time_stamp_token(token, time_stamp):
  70. return response.json(13)
  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. # 设备模拟国外环境测试
  101. # if SERVER_TYPE == 'Ansjer.us_config.formal_settings': # 国外正式配置使用固定ip进行测试
  102. # ip = '67.220.90.13'
  103. # else:
  104. # ip = CommonService.get_ip_address(request)
  105. ip = CommonService.get_ip_address(request)
  106. region_id = Device_Region().get_device_region(ip)
  107. iotClient = IOTClient(region_id)
  108. res = iotClient.create_keys_and_certificate(ThingNameSuffix, thingGroup, response)
  109. token_iot_number = hashlib.md5((str(uuid.uuid1()) + str(int(time.time()))).encode('utf-8')).hexdigest()
  110. iotdeviceInfoModel.objects.create(uid=uid,
  111. serial_number=serial,
  112. endpoint=res[0]['endpoint'],
  113. certificate_id=res[0]['certificateId'],
  114. certificate_pem=res[0]['certificatePem'],
  115. public_key=res[0]['publicKey'],
  116. private_key=res[0]['privateKey'],
  117. thing_name=res[1]['ThingName'],
  118. thing_groups=res[1]['thingGroupName'],
  119. token_iot_number=token_iot_number
  120. )
  121. res = {
  122. 'certificateId': res[0]['certificateId'],
  123. 'certificatePem': res[0]['certificatePem'],
  124. 'publicKey': res[0]['publicKey'],
  125. 'privateKey': res[0]['privateKey'],
  126. 'endpoint': res[0]['endpoint']
  127. }
  128. return response.json(0, {'res': res})
  129. else:
  130. iot = iotdeviceInfo_qs[0]
  131. res = {
  132. 'certificateId': iot.certificate_id,
  133. 'certificatePem': iot.certificate_pem,
  134. 'publicKey': iot.public_key,
  135. 'privateKey': iot.private_key,
  136. 'endpoint': iot.endpoint
  137. }
  138. # print('此设备已注册证书')
  139. return response.json(0, {'res': res})
  140. def thing_regroup(self, request_dict, response, request):
  141. # 物品重新分组
  142. uid = request_dict.get('uid', '')
  143. token = request_dict.get('token', None)
  144. language = request_dict.get('language', None)
  145. time_stamp = request_dict.get('time_stamp', None)
  146. device_version = request_dict.get('device_version', None)
  147. if not all([token, language, time_stamp, device_version]):
  148. return response.json(444, {'param: token, language, time_stamp, device_version'})
  149. # 时间戳token校验
  150. if not CommonService.check_time_stamp_token(token, time_stamp):
  151. return response.json(13)
  152. ip = CommonService.get_ip_address(request)
  153. region_id = Device_Region().get_device_region(ip)
  154. iotClient = IOTClient(region_id)
  155. ThingNameSuffix = uid
  156. if not uid:
  157. # 使用序列号
  158. serial_number = request_dict.get('serial_number', None)
  159. if not serial_number:
  160. return response.json(444)
  161. ThingNameSuffix = serial_number
  162. thingName = 'Ansjer_Device_' + ThingNameSuffix
  163. new_thingGroupName = (device_version + '_' + language).replace('.', '_') # 物品组命名不能包含'.'
  164. try:
  165. # 获取旧物品组
  166. list_groups_res = iotClient.client.list_thing_groups_for_thing(thingName=thingName, maxResults=1)
  167. old_thingGroupName = list_groups_res['thingGroups'][0]['groupName']
  168. # 没有新物品组则创建
  169. list_thing_groups_res = iotClient.client.list_thing_groups(namePrefixFilter=new_thingGroupName
  170. , maxResults=1, recursive=False)
  171. if not list_thing_groups_res['thingGroups']:
  172. attributes = {
  173. "update_time": "0"
  174. }
  175. thingGroupProperties = {
  176. "thingGroupDescription": "OTA",
  177. "attributePayload": {
  178. "attributes": attributes,
  179. "merge": False # 更新时覆盖掉而不是合并
  180. }
  181. }
  182. iotClient.client.create_thing_group(thingGroupName=new_thingGroupName
  183. , thingGroupProperties=thingGroupProperties)
  184. iotClient.client.update_thing_groups_for_thing(thingName=thingName
  185. , thingGroupsToAdd=[new_thingGroupName]
  186. , thingGroupsToRemove=[old_thingGroupName])
  187. # 更新设备版本信息
  188. if uid:
  189. Device_Info.objects.filter(UID=uid).update(version=device_version)
  190. else:
  191. Device_Info.objects.filter(serial_number=serial_number).update(version=device_version)
  192. return response.json(0)
  193. except Exception as e:
  194. print(e)
  195. return response.json(500, repr(e))
  196. def clear_Iot_Cerm(self, userID, request_dict, response):
  197. serial_number = request_dict.get('serial_number', None)
  198. if serial_number:
  199. iot = iotdeviceInfoModel.objects.filter(thing_name="Ansjer_Device_" + serial_number)
  200. if iot.exists():
  201. iot.delete()
  202. return response.json(0)
  203. else:
  204. return response.json(444)
  205. def request_publish_message(self, request_dict, response, request):
  206. # Alexa请求IoT Core下发MQTT消息通知设备开始或停止推流,或唤醒设备
  207. UID = request_dict.get('UID', None)
  208. MSG = request_dict.get('MSG', None)
  209. if not all([UID, MSG]):
  210. return response.json(444)
  211. try:
  212. # 获取设备的物品名后缀
  213. device_info_qs = Device_Info.objects.filter(UID=UID).values('UID', 'serial_number')
  214. if not device_info_qs.exists():
  215. return response.json(10043)
  216. uid = device_info_qs[0]['UID']
  217. serial_number = device_info_qs[0]['serial_number']
  218. # 如果device_info表的serial_number不为空,物品名为'Ansjer_Device_序列号'
  219. thing_name_suffix = serial_number if serial_number != '' else uid
  220. # 获取数据组织将要请求的url
  221. iot = iotdeviceInfoModel.objects.filter(thing_name__contains=thing_name_suffix).values('thing_name',
  222. 'endpoint',
  223. 'token_iot_number')
  224. if not iot.exists():
  225. return response.json(10043)
  226. thing_name = iot[0]['thing_name'][14:] # IoT core上的物品名: Ansjer_Device_ + 序列号+企业编码/uid
  227. endpoint = iot[0]['endpoint']
  228. Token = iot[0]['token_iot_number']
  229. # Token = '297a601b3925e04daab5a60280650e09'
  230. topic_suffix = '_power_topic' if 'Turn' in MSG else '_rtsp_topic'
  231. topic_name = thing_name + topic_suffix # MQTT主题
  232. # api doc: https://docs.aws.amazon.com/zh_cn/iot/latest/developerguide/http.html
  233. # url: https://IoT_data_endpoint/topics/url_encoded_topic_name?qos=1
  234. # post请求url来发布MQTT消息
  235. url = 'https://{}/topics/{}'.format(endpoint, topic_name)
  236. authorizer_name = 'Ansjer_Iot_Auth'
  237. signature = CommonService.rsa_sign(Token) # Token签名
  238. headers = {'x-amz-customauthorizer-name': authorizer_name, 'Token': Token,
  239. 'x-amz-customauthorizer-signature': signature}
  240. params = {'command': MSG}
  241. r = requests.post(url=url, headers=headers, json=params, timeout=2)
  242. if r.status_code == 200:
  243. res = r.json()
  244. if res['message'] == 'OK':
  245. return response.json(0)
  246. return response.json(10044)
  247. else:
  248. # print('发布失败')
  249. return response.json(10044)
  250. except Exception as e:
  251. # print(e)
  252. return response.json(500, repr(e))
  253. def get_s3_pull_key(self, request_dict, response, request):
  254. # 通用发布主题通知
  255. UID = request_dict.get('UID', None)
  256. if not all([UID]):
  257. return response.json(444)
  258. try:
  259. # 获取检查uid的序列号,如果没有序列号,不使用MQTT下发消息
  260. device_info_qs = Device_Info.objects.filter(UID=UID).values('UID', 'serial_number')
  261. if not device_info_qs.exists():
  262. return response.json(10043)
  263. uid = device_info_qs[0]['UID']
  264. serial_number = device_info_qs[0]['serial_number']
  265. # 如果device_info表的serial_number不为空,物品名为'Ansjer_Device_序列号'
  266. thing_name_suffix = serial_number if serial_number != '' else uid
  267. # 获取数据组织将要请求的url
  268. iot = iotdeviceInfoModel.objects.filter(thing_name__contains=thing_name_suffix).values('thing_name',
  269. 'endpoint',
  270. 'token_iot_number')
  271. if not iot.exists():
  272. return response.json(10043)
  273. endpoint = iot[0]['endpoint']
  274. MSG = self.get_s3_key_return_msg(endpoint)
  275. return response.json(0,MSG)
  276. except Exception as e:
  277. # print(e)
  278. return response.json(500, repr(e))
  279. def get_s3_key_return_msg(self,endpoint):
  280. MSG = {}
  281. if 'cn-northwest-1' in endpoint :
  282. key = AWS_IOT_GETS3_PULL_CHINA_ID
  283. secret = AWS_IOT_GETS3_PULL_CHINA_SECRET
  284. arn = AWS_ARN[0]
  285. region_name = AWS_IOT_SES_ACCESS_CHINA_REGION
  286. else:
  287. key = AWS_IOT_GETS3_PULL_FOREIGN_ID
  288. secret = AWS_IOT_GETS3_PULL_FOREIGN_SECRET
  289. arn = AWS_ARN[1]
  290. if 'ap-southeast-1' in endpoint :
  291. region_name = AWS_IOT_SES_ACCESS_FOREIGN_REGION_ASIA
  292. if 'eu-west-1' in endpoint :
  293. region_name = AWS_IOT_SES_ACCESS_FOREIGN_REGION_EUROPE
  294. if 'us-east-1' in endpoint :
  295. region_name = AWS_IOT_SES_ACCESS_FOREIGN_REGION_AMERICA
  296. MSG['AccessKeyId'] = key
  297. MSG['AccessKeySecret'] = secret
  298. MSG['bucket_name'] = 'asj-log'
  299. MSG['arn'] = arn
  300. MSG['region_name'] = region_name
  301. return MSG
  302. def getIotInfo(self, request_dict, response):
  303. # 获取IoT数据
  304. serial_number = request_dict.get('serial_number', None)
  305. if not serial_number:
  306. return response.json(444)
  307. try:
  308. serial_number = serial_number[0:6]
  309. iot_info_qs = iotdeviceInfoModel.objects.filter(serial_number=serial_number).values('endpoint', 'token_iot_number')
  310. if not iot_info_qs.exists():
  311. return response.json(173)
  312. endpoint = iot_info_qs[0]['endpoint']
  313. token_iot_number = iot_info_qs[0]['token_iot_number']
  314. res = {'endpoint': endpoint, 'token_iot_number': token_iot_number}
  315. return response.json(0, res)
  316. except Exception as e:
  317. return response.json(500, repr(e))