IotCoreController.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. import hashlib
  4. import logging
  5. import time
  6. import uuid
  7. from collections import OrderedDict
  8. import requests
  9. from django.views import View
  10. from Ansjer.config import AWS_IOT_GETS3_PULL_CHINA_ID, AWS_IOT_GETS3_PULL_CHINA_SECRET, \
  11. AWS_IOT_GETS3_PULL_FOREIGN_ID, AWS_IOT_GETS3_PULL_FOREIGN_SECRET, AWS_ARN, AWS_IOT_SES_ACCESS_CHINA_REGION, \
  12. AWS_IOT_SES_ACCESS_FOREIGN_REGION_ASIA, AWS_IOT_SES_ACCESS_FOREIGN_REGION_EUROPE, \
  13. AWS_IOT_SES_ACCESS_FOREIGN_REGION_AMERICA
  14. from Controller.DeviceConfirmRegion import Device_Region
  15. from Model.models import Device_Info, iotdeviceInfoModel, SerialNumberModel, UidSetModel
  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. class IotCoreView(View):
  21. def get(self, request, *args, **kwargs):
  22. request.encoding = 'utf-8'
  23. request_dict = request.GET
  24. operation = kwargs.get('operation', None)
  25. return self.validate(operation, request_dict, request)
  26. def post(self, request, *args, **kwargs):
  27. request.encoding = 'utf-8'
  28. request_dict = request.POST
  29. operation = kwargs.get('operation', None)
  30. return self.validate(operation, request_dict, request)
  31. def validate(self, operation, request_dict, request):
  32. response = ResponseObject()
  33. lang = request_dict.get('lang', 'en')
  34. response.lang = lang
  35. if operation == 'createKeysAndCertificate':
  36. return self.create_keys_and_certificate(request_dict, response, request)
  37. elif operation == 'requestPublishMessage':
  38. return self.request_publish_message(request_dict, response, request)
  39. elif operation == 'getS3PullKey':
  40. return self.get_s3_pull_key(request_dict, response, request)
  41. elif operation == 'thingRegroup':
  42. return self.thing_regroup(request_dict, response)
  43. elif operation == 'pcGetIotInfo':
  44. return self.pcGetIotInfo(request_dict, response)
  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. elif operation == 'getIotInfo':
  53. return self.getIotInfo(request_dict, response)
  54. else:
  55. return response.json(404)
  56. # 设备注册到aws iot core
  57. @staticmethod
  58. def create_keys_and_certificate(request_dict, response, request):
  59. logger = logging.getLogger('info')
  60. logger.info('设备注册到aws iot core请求参数:{}'.format(request_dict))
  61. token = request_dict.get('token', None)
  62. language = request_dict.get('language', None)
  63. time_stamp = request_dict.get('time_stamp', None)
  64. device_version = request_dict.get('device_version', None).replace('.', '_') # 物品组命名不能包含'.'
  65. if not all([token, language, time_stamp, device_version]):
  66. return response.json(444, {'param': 'token, language, time_stamp, device_version'})
  67. try:
  68. # 时间戳token校验
  69. no_rtc = request_dict.get('no_rtc', None)
  70. if no_rtc:
  71. if not CommonService.check_time_stamp_token_without_distance(token, time_stamp):
  72. return response.json(13)
  73. else:
  74. if not CommonService.check_time_stamp_token(token, time_stamp):
  75. return response.json(13)
  76. uid = request_dict.get('uid', '')
  77. uid_code = request_dict.get('uid_code', None)
  78. company_mark = '11A'
  79. if not uid: # 传序列号
  80. serial_number = request_dict.get('serial_number', None)
  81. serial_number_code = request_dict.get('serial_number_code', None)
  82. if not all([serial_number, serial_number_code]):
  83. return response.json(444, {'param': 'serial_number, serial_number_code'})
  84. # 序列号编码解码校验
  85. serial_number_code = CommonService.decode_data(serial_number_code)
  86. if serial_number != serial_number_code:
  87. return response.json(404)
  88. serial = serial_number[0:6]
  89. company_mark = serial_number[-3:]
  90. try:
  91. SerialNumberModel.objects.get(serial_number=serial)
  92. except:
  93. return response.json(444)
  94. thing_name_suffix = serial_number # 物品名后缀
  95. iot_device_info_qs = iotdeviceInfoModel.objects.filter(serial_number=serial)
  96. else: # 传uid
  97. # uid编码解码校验
  98. uid_code = CommonService.decode_data(uid_code)
  99. if uid != uid_code:
  100. return response.json(404)
  101. serial = '' # iot_deviceInfo表写入serial_number为''
  102. thing_name_suffix = uid # 物品名后缀
  103. iot_device_info_qs = iotdeviceInfoModel.objects.filter(uid=uid)
  104. # 判断设备是否已注册过
  105. if iot_device_info_qs.exists():
  106. iot = iot_device_info_qs[0]
  107. res = {
  108. 'certificateId': iot.certificate_id,
  109. 'certificatePem': iot.certificate_pem,
  110. 'publicKey': iot.public_key,
  111. 'privateKey': iot.private_key,
  112. 'endpoint': iot.endpoint
  113. }
  114. return response.json(0, {'res': res})
  115. else:
  116. # 根据地区id确定使用的boto3 client
  117. region_id = request_dict.get('region_id', None)
  118. if region_id:
  119. region_id = int(region_id)
  120. else:
  121. ip = CommonService.get_ip_address(request)
  122. region_id = Device_Region().get_device_region(ip)
  123. iotClient = IOTClient(region_id)
  124. # 拼接物品名
  125. thingName = CommonService.get_thing_name(company_mark, thing_name_suffix)
  126. thingGroup = device_version + '_' + language
  127. res = iotClient.register_to_iot_core(thingName, thingGroup, response)
  128. token_iot_number = hashlib.md5((str(uuid.uuid1()) + str(int(time.time()))).encode('utf-8')).hexdigest()
  129. iotdeviceInfoModel.objects.create(uid=uid,
  130. serial_number=serial,
  131. endpoint=res[0]['endpoint'],
  132. certificate_id=res[0]['certificateId'],
  133. certificate_pem=res[0]['certificatePem'],
  134. public_key=res[0]['publicKey'],
  135. private_key=res[0]['privateKey'],
  136. thing_name=res[1]['ThingName'],
  137. thing_groups=res[1]['thingGroupName'],
  138. token_iot_number=token_iot_number
  139. )
  140. res = {
  141. 'certificateId': res[0]['certificateId'],
  142. 'certificatePem': res[0]['certificatePem'],
  143. 'publicKey': res[0]['publicKey'],
  144. 'privateKey': res[0]['privateKey'],
  145. 'endpoint': res[0]['endpoint']
  146. }
  147. return response.json(0, {'res': res})
  148. except Exception as e:
  149. print(e)
  150. return response.json(500, repr(e))
  151. @staticmethod
  152. def thing_regroup(request_dict, response):
  153. # 物品重新分组
  154. uid = request_dict.get('uid', '')
  155. token = request_dict.get('token', None)
  156. language = request_dict.get('language', None)
  157. region_id = request_dict.get('region_id', None)
  158. time_stamp = request_dict.get('time_stamp', None)
  159. device_version = request_dict.get('device_version', None)
  160. if not all([token, language, region_id, time_stamp, device_version]):
  161. return response.json(444)
  162. # 时间戳token校验
  163. if not CommonService.check_time_stamp_token(token, time_stamp):
  164. return response.json(13)
  165. company_mark = '11A'
  166. thing_name_suffix = uid
  167. if not uid:
  168. # 使用序列号
  169. serial_number = request_dict.get('serial_number', None)
  170. if not serial_number:
  171. return response.json(444)
  172. company_mark = serial_number[-3:]
  173. thing_name_suffix = serial_number
  174. uid = CommonService.query_uid_with_serial(serial_number)
  175. uid_set_qs = UidSetModel.objects.filter(uid=uid)
  176. thingName = CommonService.get_thing_name(company_mark, thing_name_suffix)
  177. new_thingGroupName = (device_version + '_' + language).replace('.', '_') # 物品组命名不能包含'.'
  178. try:
  179. iotClient = IOTClient(int(region_id))
  180. # 获取旧物品组
  181. list_groups_res = iotClient.client.list_thing_groups_for_thing(thingName=thingName, maxResults=1)
  182. old_thingGroupName = list_groups_res['thingGroups'][0]['groupName']
  183. # 没有新物品组则创建
  184. list_thing_groups_res = iotClient.client.list_thing_groups(namePrefixFilter=new_thingGroupName
  185. , maxResults=1, recursive=False)
  186. if not list_thing_groups_res['thingGroups']:
  187. attributes = {
  188. "update_time": "0"
  189. }
  190. thingGroupProperties = {
  191. "thingGroupDescription": "OTA",
  192. "attributePayload": {
  193. "attributes": attributes,
  194. "merge": False # 更新时覆盖掉而不是合并
  195. }
  196. }
  197. iotClient.client.create_thing_group(thingGroupName=new_thingGroupName
  198. , thingGroupProperties=thingGroupProperties)
  199. iotClient.client.update_thing_groups_for_thing(thingName=thingName
  200. , thingGroupsToAdd=[new_thingGroupName]
  201. , thingGroupsToRemove=[old_thingGroupName])
  202. # 更新设备版本信息
  203. uid_set_qs.update(version=device_version)
  204. return response.json(0)
  205. except Exception as e:
  206. print(e)
  207. return response.json(500, repr(e))
  208. def clear_Iot_Cerm(self, request_dict, response):
  209. serial_number = request_dict.get('serial_number', None)
  210. if serial_number:
  211. iot = iotdeviceInfoModel.objects.filter(thing_name="Ansjer_Device_" + serial_number)
  212. if iot.exists():
  213. iot.delete()
  214. return response.json(0)
  215. else:
  216. return response.json(444)
  217. def request_publish_message(self, request_dict, response, request):
  218. # Alexa请求IoT Core下发MQTT消息通知设备开始或停止推流,或唤醒设备
  219. UID = request_dict.get('UID', None)
  220. rtsp = request_dict.get('rtsp', None)
  221. enable = request_dict.get('enable', '1')
  222. if not all([UID, rtsp]):
  223. return response.json(444)
  224. try:
  225. # 获取设备的物品名后缀
  226. device_info_qs = Device_Info.objects.filter(UID=UID).values('UID', 'serial_number')
  227. if not device_info_qs.exists():
  228. return response.json(10043)
  229. uid = device_info_qs[0]['UID']
  230. serial_number = device_info_qs[0]['serial_number']
  231. # 如果device_info表的serial_number不为空,物品名为'Ansjer_Device_序列号'
  232. thing_name_suffix = serial_number if serial_number != '' else uid
  233. # 获取数据组织将要请求的url
  234. iot = iotdeviceInfoModel.objects.filter(thing_name__contains=thing_name_suffix).values('thing_name',
  235. 'endpoint',
  236. 'token_iot_number')
  237. if not iot.exists():
  238. return response.json(10043)
  239. thing_name = iot[0]['thing_name'][14:] # IoT core上的物品名: Ansjer_Device_ + 序列号+企业编码/uid
  240. endpoint = iot[0]['endpoint']
  241. Token = iot[0]['token_iot_number']
  242. # Token = '297a601b3925e04daab5a60280650e09'
  243. topic_name = 'ansjer/generic/' + thing_name # MQTT主题名
  244. # api doc: https://docs.aws.amazon.com/zh_cn/iot/latest/developerguide/http.html
  245. # url: https://IoT_data_endpoint/topics/url_encoded_topic_name?qos=1
  246. # post请求url来发布MQTT消息
  247. url = 'https://{}/topics/{}'.format(endpoint, topic_name)
  248. authorizer_name = 'Ansjer_Iot_Auth'
  249. signature = CommonService.rsa_sign(Token) # Token签名
  250. headers = {'x-amz-customauthorizer-name': authorizer_name, 'Token': Token,
  251. 'x-amz-customauthorizer-signature': signature}
  252. params = OrderedDict(
  253. [
  254. ('alexaRtspCommand', rtsp),
  255. ('enable', enable),
  256. ]
  257. )
  258. r = requests.post(url=url, headers=headers, json=params, timeout=2)
  259. if r.status_code == 200:
  260. res = r.json()
  261. if res['message'] == 'OK':
  262. return response.json(0)
  263. return response.json(10044)
  264. else:
  265. # print('发布失败')
  266. return response.json(10044)
  267. except Exception as e:
  268. # print(e)
  269. return response.json(500, repr(e))
  270. def get_s3_pull_key(self, request_dict, response, request):
  271. # 通用发布主题通知
  272. UID = request_dict.get('UID', None)
  273. if not all([UID]):
  274. return response.json(444)
  275. try:
  276. # 获取检查uid的序列号,如果没有序列号,不使用MQTT下发消息
  277. device_info_qs = Device_Info.objects.filter(UID=UID).values('UID', 'serial_number')
  278. if not device_info_qs.exists():
  279. return response.json(10043)
  280. uid = device_info_qs[0]['UID']
  281. serial_number = device_info_qs[0]['serial_number']
  282. # 如果device_info表的serial_number不为空,物品名为'Ansjer_Device_序列号'
  283. thing_name_suffix = serial_number if serial_number != '' else uid
  284. # 获取数据组织将要请求的url
  285. iot = iotdeviceInfoModel.objects.filter(thing_name__contains=thing_name_suffix).values('thing_name',
  286. 'endpoint',
  287. 'token_iot_number')
  288. if not iot.exists():
  289. return response.json(10043)
  290. endpoint = iot[0]['endpoint']
  291. MSG = self.get_s3_key_return_msg(endpoint)
  292. return response.json(0, MSG)
  293. except Exception as e:
  294. # print(e)
  295. return response.json(500, repr(e))
  296. def get_s3_key_return_msg(self, endpoint):
  297. MSG = {}
  298. if 'cn-northwest-1' in endpoint:
  299. key = AWS_IOT_GETS3_PULL_CHINA_ID
  300. secret = AWS_IOT_GETS3_PULL_CHINA_SECRET
  301. arn = AWS_ARN[0]
  302. region_name = AWS_IOT_SES_ACCESS_CHINA_REGION
  303. else:
  304. key = AWS_IOT_GETS3_PULL_FOREIGN_ID
  305. secret = AWS_IOT_GETS3_PULL_FOREIGN_SECRET
  306. arn = AWS_ARN[1]
  307. if 'ap-southeast-1' in endpoint:
  308. region_name = AWS_IOT_SES_ACCESS_FOREIGN_REGION_ASIA
  309. if 'eu-west-1' in endpoint:
  310. region_name = AWS_IOT_SES_ACCESS_FOREIGN_REGION_EUROPE
  311. if 'us-east-1' in endpoint:
  312. region_name = AWS_IOT_SES_ACCESS_FOREIGN_REGION_AMERICA
  313. MSG['AccessKeyId'] = key
  314. MSG['AccessKeySecret'] = secret
  315. MSG['bucket_name'] = 'asj-log'
  316. MSG['arn'] = arn
  317. MSG['region_name'] = region_name
  318. return MSG
  319. def getIotInfo(self, request_dict, response):
  320. # 获取IoT数据
  321. serial_number = request_dict.get('serial_number', None)
  322. uid = request_dict.get('uid', None)
  323. if not uid and not serial_number:
  324. return response.json(444)
  325. try:
  326. if serial_number:
  327. serial_number = serial_number[0:6]
  328. iot_info_qs = iotdeviceInfoModel.objects.filter(serial_number=serial_number). \
  329. values('endpoint', 'token_iot_number')
  330. else:
  331. iot_info_qs = iotdeviceInfoModel.objects.filter(uid=uid). \
  332. values('endpoint', 'token_iot_number')
  333. if not iot_info_qs.exists():
  334. return response.json(173)
  335. endpoint = iot_info_qs[0]['endpoint']
  336. token_iot_number = iot_info_qs[0]['token_iot_number']
  337. res = {'endpoint': endpoint, 'token_iot_number': token_iot_number}
  338. return response.json(0, res)
  339. except Exception as e:
  340. return response.json(500, repr(e))
  341. def pcGetIotInfo(self, request_dict, response):
  342. # PC工具获取IoT数据
  343. serial_number = request_dict.get('serial_number', None)
  344. uid = request_dict.get('uid', None)
  345. if not uid and not serial_number:
  346. return response.json(444)
  347. try:
  348. if serial_number:
  349. serial_number = serial_number[0:6]
  350. iot_info_qs = iotdeviceInfoModel.objects.filter(serial_number=serial_number). \
  351. values('endpoint', 'token_iot_number')
  352. else:
  353. iot_info_qs = iotdeviceInfoModel.objects.filter(uid=uid). \
  354. values('endpoint', 'token_iot_number')
  355. if not iot_info_qs.exists():
  356. return response.json(173)
  357. endpoint = iot_info_qs[0]['endpoint']
  358. token_iot_number = iot_info_qs[0]['token_iot_number']
  359. res = {'endpoint': endpoint, 'token_iot_number': token_iot_number}
  360. return response.json(0, res)
  361. except Exception as e:
  362. return response.json(500, repr(e))