IotCoreController.py 18 KB

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