SerialNumberController.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429
  1. import json
  2. import logging
  3. import time
  4. from django.db import transaction
  5. from django.views import View
  6. from Ansjer.config import CRCKey, CONFIG_INFO, CONFIG_TEST, CONFIG_US, \
  7. CONFIG_CN, USED_SERIAL_REDIS_LIST, UNUSED_SERIAL_REDIS_LIST
  8. from Model.models import SerialNumberModel, CompanySerialModel, UIDCompanySerialModel, UIDModel, Device_Info, \
  9. iotdeviceInfoModel, LogModel, UidSetModel, UID_Bucket, \
  10. Unused_Uid_Meal, Order_Model, StsCrdModel, VodHlsModel, ExperienceContextModel, UidUserModel, ExperienceAiModel, \
  11. AiService, DeviceDomainRegionModel, RegionModel
  12. from Object.RedisObject import RedisObject
  13. from Object.uidManageResponseObject import uidManageResponseObject
  14. from Service.AlgorithmService import AlgorithmBaseOn35
  15. from Service.CommonService import CommonService
  16. from Service.EquipmentInfoService import EquipmentInfoService
  17. class SerialNumberView(View):
  18. def get(self, request, *args, **kwargs):
  19. request.encoding = 'utf-8'
  20. operation = kwargs.get('operation', None)
  21. request_dict = request.GET
  22. return self.validate(request_dict, operation, request)
  23. def post(self, request, *args, **kwargs):
  24. request.encoding = 'utf-8'
  25. operation = kwargs.get('operation', None)
  26. request_dict = request.POST
  27. return self.validate(request_dict, operation, request)
  28. def validate(self, request_dict, operation, request):
  29. response = uidManageResponseObject()
  30. if operation == 'attachUID': # 绑定uid
  31. return self.do_attach_uid(request_dict, response, request)
  32. elif operation == 'detachUID': # 解绑uid
  33. return self.do_detach_uid(request, request_dict, response)
  34. elif operation == 'create': # 创建序列号
  35. return self.do_create(request_dict, response)
  36. elif operation == 'getUID': # 根据序列号获取uid
  37. return self.do_get_uid(request_dict, response)
  38. elif operation == 'getRegionInfo': # 根据序列号状态确认uid地区(PC工具使用)
  39. return self.get_region_info(request_dict, response)
  40. elif operation == 'saveRegion': # 保存序列号地区信息(app使用)
  41. return self.save_region(request, request_dict, response)
  42. elif operation == 'getDomain': # 获取域名(设备使用)
  43. return self.get_domain(request_dict, response)
  44. else:
  45. return response.json(414)
  46. @staticmethod
  47. def do_create(request_dict, response):
  48. quantity = int(request_dict.get('quantity', 0))
  49. if not quantity:
  50. return response.json(444)
  51. try:
  52. try:
  53. number = SerialNumberModel.objects.last().id
  54. except:
  55. number = 0
  56. serial_number_bulk = []
  57. now_time = int(time.time())
  58. algorithm = AlgorithmBaseOn35()
  59. for i in range(quantity):
  60. serial_number = algorithm.getLetter(number)
  61. number += 1 # sum每次递增1
  62. # 前面补0至六位
  63. serial_number = (6 - len(serial_number)) * '0' + serial_number
  64. serial_number_bulk.append(SerialNumberModel(serial_number=serial_number, add_time=now_time))
  65. # 开启事务写入
  66. with transaction.atomic():
  67. SerialNumberModel.objects.bulk_create(serial_number_bulk)
  68. return response.json(0)
  69. except Exception as e:
  70. print(e)
  71. return response.json(500, repr(e))
  72. def do_attach_uid(self, request_dict, response, request):
  73. serial_number = request_dict.get('serial_number', None)
  74. token = request_dict.get('token', None)
  75. time_stamp = request_dict.get('time_stamp', None)
  76. if not all([serial_number, token, time_stamp]):
  77. return response.json(444)
  78. # 时间戳token校验
  79. if not CommonService.check_time_stamp_token(token, time_stamp):
  80. return response.json(13)
  81. now_time = int(time.time())
  82. serial = serial_number[0:6]
  83. company_identity = serial_number[6:9]
  84. full_serial = serial_number[0:9]
  85. # 根据企业标识确认企业秘钥
  86. company_secret = ''
  87. if company_identity == '11A':
  88. company_secret = 'MTEyMTNB'
  89. elif company_identity == '11L':
  90. company_secret = 'VmXEWnBR'
  91. elif company_identity == '11Z':
  92. company_secret = 'ZsKWcxdD'
  93. try:
  94. # 判断序列号是否已和企业关联
  95. company_serial_qs = CompanySerialModel.objects.filter(company__secret=company_secret, serial_number=serial)
  96. if not company_serial_qs.exists():
  97. return response.json(173)
  98. company_serial = company_serial_qs[0]
  99. if company_serial.status == 0: # 该序列号未绑定企业
  100. return response.json(173)
  101. elif company_serial.status == 1: # 绑定uid
  102. # redis加锁,防止同一个序列号重复绑定
  103. key = serial + 'do_attach_uid'
  104. redisObj = RedisObject()
  105. isLock = redisObj.CONN.setnx(key, 1)
  106. redisObj.CONN.expire(key, 60)
  107. if not isLock:
  108. return response.json(5)
  109. # 获取并判断region_id
  110. region_id = CommonService.confirm_region_id()
  111. if region_id not in [1, 2, 3, 4]:
  112. return response.json(444, {'invalid region_id': region_id})
  113. p2p_type = request_dict.get('p2ptype', 1)
  114. if serial_number[9:10]:
  115. p2p_type = serial_number[9:10]
  116. p2p_type = int(p2p_type)
  117. with transaction.atomic():
  118. count = 0
  119. while count < 3:
  120. # 查询是否存在未绑定序列号的uid
  121. uid_qs = UIDModel.objects.filter(vpg__company_id=company_serial.company.id,
  122. vpg__region_id=region_id, status=0, p2p_type=p2p_type). \
  123. order_by('id')
  124. if not uid_qs.exists():
  125. return response.json(375)
  126. uid = uid_qs[0]
  127. # 判断uid是否已绑定过序列号
  128. uid_company_serial_qs = UIDCompanySerialModel.objects.filter(uid_id=uid.id)
  129. if uid_company_serial_qs.exists():
  130. return response.json(377)
  131. result = UIDModel.objects.filter(id=uid.id, status=0).update(status=2, update_time=now_time)
  132. if int(result) <= 0: # 更新失败
  133. count += 1
  134. continue
  135. # UID关联【企业关联序列号】表创建数据
  136. UIDCompanySerialModel.objects.create(uid_id=uid.id, company_serial_id=company_serial.id,
  137. add_time=now_time, update_time=now_time)
  138. company_serial.status = 2
  139. company_serial.save()
  140. dev = Device_Info.objects.filter(UID=uid.uid)
  141. if dev.exists():
  142. dev.update(serial_number=full_serial)
  143. full_uid_code = uid.full_uid_code
  144. if uid.platform in CRCKey.keys():
  145. full_uid_code += ':' + CRCKey[uid.platform]
  146. res = {
  147. 'full_uid_code': CommonService.encode_data(full_uid_code),
  148. 'uid': CommonService.encode_data(uid.uid),
  149. 'extra': uid.uid_extra,
  150. 'platform': uid.platform,
  151. 'initString': uid.init_string,
  152. 'initStringApp': uid.init_string_app,
  153. }
  154. # 记录操作日志
  155. ip = CommonService.get_ip_address(request)
  156. content = json.loads(json.dumps(request_dict))
  157. log = {
  158. 'ip': ip,
  159. 'user_id': 1,
  160. 'status': 200,
  161. 'time': now_time,
  162. 'content': json.dumps(content),
  163. 'url': 'serialNumber/attachUID',
  164. 'operation': '序列号{}绑定uid: {}'.format(serial, uid.uid),
  165. }
  166. LogModel.objects.create(**log)
  167. if CONFIG_INFO != CONFIG_TEST: # 不为测试服,则序列号写入redis列表
  168. redisObj.rpush(USED_SERIAL_REDIS_LIST, serial)
  169. redisObj.del_data(key=key)
  170. return response.json(0, res)
  171. return response.json(5)
  172. elif company_serial.status == 2: # 返回uid
  173. uid_qs = UIDCompanySerialModel.objects.filter(company_serial_id=company_serial.id)
  174. if not uid_qs.exists():
  175. return response.json(173)
  176. uid = uid_qs.values('uid__uid', 'uid__mac', 'uid__uid_extra', 'uid__full_uid_code',
  177. 'uid__platform', 'uid__init_string', 'uid__init_string_app')[0]
  178. full_uid_code = uid['uid__full_uid_code']
  179. if uid['uid__platform'] in CRCKey.keys():
  180. full_uid_code += ':' + CRCKey[uid['uid__platform']]
  181. res = {
  182. 'full_uid_code': CommonService.encode_data(full_uid_code),
  183. 'uid': CommonService.encode_data(uid['uid__uid']),
  184. 'extra': uid['uid__uid_extra'],
  185. 'platform': uid['uid__platform'],
  186. 'initString': uid['uid__init_string'],
  187. 'initStringApp': uid['uid__init_string_app'],
  188. }
  189. return response.json(0, res)
  190. elif company_serial.status == 3:
  191. return response.json(10042)
  192. except Exception as e:
  193. djangoLogger = logging.getLogger('django')
  194. djangoLogger.exception(repr(e))
  195. return response.json(176, str(e))
  196. def do_get_uid(self, request_dict, response):
  197. serial_number = request_dict.get('serial_number', None)
  198. token = request_dict.get('token', None)
  199. time_stamp = request_dict.get('time_stamp', None)
  200. if token and time_stamp and serial_number:
  201. # 时间戳token校验
  202. if not CommonService.check_time_stamp_token(token, time_stamp):
  203. return response.json(13)
  204. mark = serial_number[6:9]
  205. serial = serial_number[0:6]
  206. uid_company_serial_qs = UIDCompanySerialModel.objects.filter(company_serial__company__mark=mark,
  207. company_serial__serial_number__serial_number=serial)
  208. if uid_company_serial_qs.exists():
  209. uid = uid_company_serial_qs.values('uid__uid', 'uid__mac', 'uid__uid_extra')[0]
  210. res = {
  211. 'uid': CommonService.encode_data(uid['uid__uid']),
  212. 'mac': CommonService.encode_data(uid['uid__mac']),
  213. 'extra': uid['uid__uid_extra']
  214. }
  215. return response.json(0, res)
  216. else:
  217. return response.json(173)
  218. else:
  219. return response.json(444)
  220. def do_detach_uid(self, request, request_dict, response):
  221. token = request_dict.get('token', None)
  222. time_stamp = request_dict.get('time_stamp', None)
  223. serial_number = request_dict.get('serial_number', None)
  224. if not all([token, time_stamp, serial_number]):
  225. return response.json(444)
  226. # 时间戳token校验
  227. if not CommonService.check_time_stamp_token(token, time_stamp):
  228. return response.json(13)
  229. now_time = int(time.time())
  230. serial = serial_number[0:6]
  231. company_serial_qs = CompanySerialModel.objects.filter(serial_number=serial).values('status')
  232. if not company_serial_qs.exists():
  233. return response.json(379)
  234. status = company_serial_qs[0]['status']
  235. if status == 1:
  236. return response.json(0, {'success': '序列号未绑定uid'})
  237. elif status == 3:
  238. return response.json(10043)
  239. uid_serial_qs = UIDCompanySerialModel.objects.filter(company_serial__serial_number=serial)
  240. if not uid_serial_qs.exists():
  241. return response.json(173)
  242. uid_serial = uid_serial_qs[0]
  243. redisObj = RedisObject()
  244. try:
  245. with transaction.atomic():
  246. uid = uid_serial.uid.uid
  247. company_serial_qs.update(status=1) # 更新序列号状态
  248. # 删除设备相关数据,参考后台的设备重置删除的数据
  249. Device_Info.objects.filter(UID=uid).delete()
  250. UidSetModel.objects.filter(uid=uid).delete()
  251. UidUserModel.objects.filter(UID=uid).delete()
  252. iotdeviceInfoModel.objects.filter(serial_number=serial).delete()
  253. # 删除推送消息
  254. for val in range(1, 8):
  255. EquipmentInfoService.get_equipment_info_model('', val).filter(device_uid=uid).delete()
  256. # 重置设备云存
  257. UID_Bucket.objects.filter(uid=uid).delete()
  258. Unused_Uid_Meal.objects.filter(uid=uid).delete()
  259. Order_Model.objects.filter(UID=uid).delete()
  260. StsCrdModel.objects.filter(uid=uid).delete()
  261. VodHlsModel.objects.filter(uid=uid).delete()
  262. ExperienceContextModel.objects.filter(uid=uid).delete()
  263. # 重置AI
  264. ExperienceAiModel.objects.filter(uid=uid).delete()
  265. AiService.objects.filter(uid=uid).delete()
  266. if CONFIG_INFO != CONFIG_TEST: # 不为测试服,则序列号写入redis列表
  267. redisObj.rpush(UNUSED_SERIAL_REDIS_LIST, serial)
  268. UIDModel.objects.filter(uid=uid).update(status=0, mac='') # 重置uid的使用状态为未使用
  269. uid_serial.delete()
  270. # 记录操作日志
  271. ip = CommonService.get_ip_address(request)
  272. content = json.loads(json.dumps(request_dict))
  273. log = {
  274. 'ip': ip,
  275. 'user_id': 1,
  276. 'status': 200,
  277. 'time': now_time,
  278. 'content': json.dumps(content),
  279. 'url': 'serialNumber/detachUID',
  280. 'operation': '序列号{}解绑uid: {}'.format(serial, uid),
  281. }
  282. LogModel.objects.create(**log)
  283. return response.json(0)
  284. except Exception as e:
  285. djangoLogger = logging.getLogger('django')
  286. djangoLogger.exception(repr(e))
  287. return response.json(176, str(e))
  288. @staticmethod
  289. def get_region_info(request_dict, response):
  290. """
  291. 根据序列号状态确认uid地区
  292. @param request_dict: 请求参数
  293. @param response: 响应对象
  294. @request_dict serial_number: 序列号
  295. @return: response
  296. """
  297. serial_number = request_dict.get('serial_number', None)
  298. if not serial_number:
  299. return response(444)
  300. company_serial_qs = CompanySerialModel.objects.filter(serial_number=serial_number[:6]).values('status')
  301. if not company_serial_qs.exists():
  302. return response.json(173)
  303. status = company_serial_qs[0]['status']
  304. if status == 2:
  305. if CONFIG_INFO == CONFIG_CN:
  306. return response.json(0, {'region': 1})
  307. elif CONFIG_INFO == CONFIG_US:
  308. return response.json(0, {'region': 3})
  309. elif status == 3:
  310. if CONFIG_INFO == CONFIG_CN:
  311. return response.json(0, {'region': 3})
  312. elif CONFIG_INFO == CONFIG_US:
  313. return response.json(0, {'region': 1})
  314. return response.json(0)
  315. @staticmethod
  316. def save_region(request, request_dict, response):
  317. """
  318. 保存序列号地区信息
  319. @param request: 请求体
  320. @param request_dict: 请求参数
  321. @param response: 响应对象
  322. @request_dict serial_number: 序列号
  323. @return: response
  324. """
  325. serial_number = request_dict.get('serial_number', None)
  326. time_stamp_token = request_dict.get('time_stamp_token', None)
  327. time_stamp = request_dict.get('time_stamp', None)
  328. if not all([serial_number, time_stamp_token, time_stamp]):
  329. return response(444)
  330. # 时间戳token校验
  331. if not CommonService.check_time_stamp_token(time_stamp_token, time_stamp):
  332. return response.json(13)
  333. try:
  334. serial_number = serial_number[:9]
  335. region_data = {
  336. 'ip': CommonService.get_ip_address(request),
  337. 'region_id': CommonService.confirm_region_id()
  338. }
  339. device_domain_region_qs = DeviceDomainRegionModel.objects.filter(serial_number=serial_number)
  340. if device_domain_region_qs.exists():
  341. device_domain_region_qs.update(region_data)
  342. else:
  343. region_data['serial_number'] = serial_number
  344. DeviceDomainRegionModel.objects.create(serial_number=serial_number)
  345. return response.json(0)
  346. except Exception as e:
  347. return response.json(500, repr(e))
  348. @staticmethod
  349. def get_domain(request_dict, response):
  350. """
  351. 获取域名
  352. @param request_dict: 请求参数
  353. @param response: 响应对象
  354. @request_dict serial_number: 序列号
  355. @return: response
  356. """
  357. serial_number = request_dict.get('serial_number', None)
  358. time_stamp_token = request_dict.get('time_stamp_token', None)
  359. time_stamp = request_dict.get('time_stamp', None)
  360. if not all([serial_number, time_stamp_token, time_stamp]):
  361. return response(444)
  362. # 时间戳token校验
  363. if not CommonService.check_time_stamp_token(time_stamp_token, time_stamp):
  364. return response.json(13)
  365. try:
  366. serial_number = serial_number[:9]
  367. device_domain_region_qs = DeviceDomainRegionModel.objects.filter(serial_number=serial_number).values(
  368. 'region_id')
  369. if not device_domain_region_qs.exists():
  370. return response.json(173)
  371. region_id = device_domain_region_qs[0]['region_id']
  372. region_qs = RegionModel.objects.filter(id=region_id).values('api', 'push_api')
  373. res = {
  374. 'api': region_qs[0]['api'],
  375. 'push_api': region_qs[0]['push_api']
  376. }
  377. return response.json(0, res)
  378. except Exception as e:
  379. return response.json(500, repr(e))