SerialNumberController.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. import json
  4. import logging
  5. import time
  6. import requests
  7. from bulk_update.helper import bulk_update
  8. from django.db import transaction
  9. from django.db.models import Count
  10. from django.views import View
  11. from Model.models import SerialNumberModel, UserModel, UserSerialNumberModel, CompanySerialModel, \
  12. MacModel, LogModel, CompanyModel
  13. from Object.RedisObject import RedisObject
  14. from Object.ResponseObject import ResponseObject
  15. from Object.TokenObject import TokenObject
  16. from Service.AlgorithmService import AlgorithmBaseOn35
  17. from Service.CommonService import CommonService
  18. class SerialNumberView(View):
  19. def get(self, request, *args, **kwargs):
  20. request.encoding = 'utf-8'
  21. request_dict = request.GET
  22. operation = kwargs.get('operation')
  23. return self.validate(request_dict, operation, request)
  24. def post(self, request, *args, **kwargs):
  25. request.encoding = 'utf-8'
  26. request_dict = request.POST
  27. operation = kwargs.get('operation')
  28. return self.validate(request_dict, operation, request)
  29. def validate(self, request_dict, operation, request):
  30. response = ResponseObject()
  31. if operation == 'create':
  32. return self.do_create(request_dict, response)
  33. elif operation == 'getSerial':
  34. return self.do_get_serial_number(request_dict, request, response)
  35. token = request_dict.get('token', None)
  36. token = TokenObject(token)
  37. if token.code != 0:
  38. return response.json(token.code)
  39. if operation == 'quantity': # 查询当前可用的UID的数量
  40. return self.do_quantity(token.userID, response)
  41. elif operation == 'allot':
  42. return self.do_allot(request_dict, response)
  43. elif operation == 'createSerial': # 生成序列号
  44. return self.create_serial(request_dict, response, request, token.userID)
  45. elif operation == 'serialCompany': # 将序列号分匹配到指定企业
  46. return self.do_serial_company(request_dict, response, request, token.userID)
  47. elif operation == 'revise/state': # 修改序列号状态
  48. return self.revise_state(request_dict, response, request)
  49. else:
  50. return response.json(309)
  51. def revise_state(self, request_dict, response, request):
  52. """
  53. 修改序列号状态
  54. @param request_dict:请求参数
  55. @param response: 响应对象
  56. """
  57. use_status = request_dict.get('useStatus', None) # 序列号表的状态
  58. status = request_dict.get('status', None) # 关联企业序列号表的状态
  59. if not all([use_status, status]):
  60. return response.json(444)
  61. serial_number_qs = SerialNumberModel.objects.filter(use_status=use_status).values('serial_number')
  62. if not serial_number_qs.exists():
  63. return response.json(173)
  64. serial_list = [item[key] for item in serial_number_qs for key in item]
  65. try:
  66. country_serial_qs = CompanySerialModel.objects.filter(serial_number__in=serial_list).values('status')
  67. country_serial_qs.filter(status=status).update(status=2)
  68. return response.json(0)
  69. except Exception as e:
  70. print(e)
  71. return response.json(500)
  72. def do_serial_company(self, request_dict, response, request, user_id):
  73. """
  74. 将序列号分匹配到指定企业
  75. :param request_dict: 请求参数
  76. :param response: 响应对象
  77. :param request: 请求
  78. :param user_id: 用户id
  79. :return:
  80. """
  81. id = request_dict.get('id', None)
  82. quantity = request_dict.get('quantity', None)
  83. if not all([id, quantity]):
  84. return response.json(444)
  85. company_qs = CompanyModel.objects.filter(id=id)
  86. if not company_qs.exists():
  87. return response.json(444)
  88. sum_Serial = SerialNumberModel.objects.filter().count()
  89. sum_Serial_company = CompanySerialModel.objects.filter().count()
  90. sum_bind = sum_Serial - sum_Serial_company # 剩余可绑定的序列号
  91. if int(quantity) > int(sum_bind):
  92. return response.json(10041)
  93. try:
  94. company = company_qs[0]
  95. start_1 = sum_Serial_company
  96. end_1 = int(start_1) + int(quantity)
  97. serial_qs = SerialNumberModel.objects.filter()[start_1:end_1]
  98. if not serial_qs.exists():
  99. return response.json(173)
  100. company_serial_bulk = []
  101. now_time = int(time.time())
  102. for item in serial_qs: # 更新状态为已分配但未使用
  103. company_serial_bulk.append(CompanySerialModel(
  104. status=3,
  105. add_time=now_time,
  106. update_time=now_time,
  107. company_id=company.id,
  108. serial_number=item.serial_number,
  109. ))
  110. SerialNumberModel.objects.filter(serial_number=item.serial_number).update(
  111. use_status=3,
  112. add_time=now_time
  113. )
  114. # 记录操作日志
  115. ip = CommonService.get_ip_address(request)
  116. content = json.loads(json.dumps(request_dict))
  117. log = {
  118. 'ip': ip,
  119. 'user_id': user_id,
  120. 'status': 200,
  121. 'time': now_time,
  122. 'url': 'serialNumber/serialCompany',
  123. 'content': json.dumps(content),
  124. 'operation': '{}生成{}个序列号{}: {}'.format(company.name, quantity, '成功', '同步更新成功'),
  125. }
  126. with transaction.atomic():
  127. CompanySerialModel.objects.bulk_create(company_serial_bulk)
  128. company.quantity = CompanySerialModel.objects.filter(company_id=id).count()
  129. company.save()
  130. Log = LogModel.objects.create(**log)
  131. # 同步更新业务服务器和uid管理系统的企业序列号表
  132. url1 = 'http://test.zositechc.cn/company/createSerial'
  133. url2 = 'https://www.zositechc.cn/company/createSerial'
  134. url3 = 'http://www.dvema.com/company/createSerial'
  135. # url4 = 'http://www.zositeche.com/company/createSerial'
  136. requests_data = {'id': id, 'quantity': quantity}
  137. res1 = requests.post(url=url1, data=requests_data, timeout=2 * 60)
  138. if res1.status_code != 200:
  139. fail_reason = '请求测试服务器生成序列号响应状态码异常'
  140. return self.failResponse(company.name, quantity, fail_reason, Log, response)
  141. res1 = res1.json()
  142. if res1['result_code'] != 0:
  143. fail_reason = '测试服务器生成序列号发生异常'
  144. return self.failResponse(company.name, quantity, fail_reason, Log, response)
  145. res2 = requests.post(url=url2, data=requests_data, timeout=2 * 60)
  146. if res2.status_code != 200:
  147. fail_reason = '请求国内服务器生成序列号响应状态码异常'
  148. return self.failResponse(company.name, quantity, fail_reason, Log, response)
  149. res2 = res2.json()
  150. if res2['result_code'] != 0:
  151. fail_reason = '国内服务器生成序列号发生异常'
  152. return self.failResponse(company.name, quantity, fail_reason, Log, response)
  153. res3 = requests.post(url=url3, data=requests_data, timeout=2 * 60)
  154. if res3.status_code != 200:
  155. fail_reason = '请求美国服务器生成序列号响应状态码异常'
  156. return self.failResponse(company.name, quantity, fail_reason, Log, response)
  157. res3 = res3.json()
  158. if res3['result_code'] != 0:
  159. fail_reason = '美国服务器生成序列号发生异常'
  160. return self.failResponse(company.name, quantity, fail_reason, Log, response)
  161. # res4 = requests.post(url=url4, data=requests_data, timeout=2 * 60)
  162. # if res4.status_code != 200:
  163. # fail_reason = '请求欧洲服务器生成序列号响应状态码异常'
  164. # return self.failResponse(company.name, quantity, fail_reason, Log, response)
  165. # res4 = res4.json()
  166. # if res4['code'] != 0:
  167. # fail_reason = '欧洲服务器生成序列号发生异常'
  168. # return self.failResponse(company.name, quantity, fail_reason, Log, response)
  169. return response.json(0)
  170. except Exception as e:
  171. djangoLogger = logging.getLogger('django')
  172. djangoLogger.exception(repr(e))
  173. return response.json(176)
  174. def failResponse(self, company_name, quantity, fail_reason, Log, response):
  175. operation = '{}生成{}个序列号{}: {}'.format(company_name, quantity, '失败', fail_reason)
  176. Log.operation = operation
  177. Log.save()
  178. return response.json(177)
  179. def create_serial(self, request_dict, response, request, user_id):
  180. """
  181. 生成序列号
  182. :param request_dict: 请求参数
  183. :param response: 响应对象
  184. :param request: 请求
  185. :param user_id: 用户id
  186. :return:
  187. """
  188. quantity = int(request_dict.get('quantity', 0))
  189. if not quantity:
  190. return response.json(444)
  191. try:
  192. try:
  193. sum = SerialNumberModel.objects.last().id
  194. except:
  195. sum = 0
  196. serial_number_bulk = []
  197. now_time = int(time.time())
  198. algorithm = AlgorithmBaseOn35()
  199. for i in range(quantity):
  200. serial_number = algorithm.getLetter(sum)
  201. sum += 1 # sum每次递增1
  202. # 前面补0至六位
  203. serial_number = (6 - len(serial_number)) * '0' + serial_number
  204. serial_number_bulk.append(SerialNumberModel(serial_number=serial_number, add_time=now_time))
  205. # 记录操作日志
  206. ip = CommonService.get_ip_address(request)
  207. content = json.loads(json.dumps(request_dict))
  208. log = {
  209. 'ip': ip,
  210. 'user_id': user_id,
  211. 'status': 200,
  212. 'time': now_time,
  213. 'url': 'serialNumber/createSerial',
  214. 'content': json.dumps(content),
  215. 'operation': '生成{}个序列号{}: {}'.format(quantity, '成功', '同步更新成功'),
  216. }
  217. # 开启事务写入
  218. with transaction.atomic():
  219. SerialNumberModel.objects.bulk_create(serial_number_bulk)
  220. Log = LogModel.objects.create(**log)
  221. # 同步更新业务服务器和uid管理系统的序列号表
  222. url1 = 'http://test.zositechc.cn/serialNumber/create'
  223. url2 = 'https://www.zositechc.cn/serialNumber/create'
  224. url3 = 'http://www.dvema.com/serialNumber/create'
  225. # url4 = 'http://www.zositeche.com/company/createSerial'
  226. requests_data = {'quantity': quantity}
  227. res1 = requests.post(url=url1, data=requests_data, timeout=2 * 60)
  228. if res1.status_code != 200:
  229. fail_reason = '请求测试服务器生成序列号响应状态码异常'
  230. return self.generateFail(quantity, fail_reason, Log, response)
  231. res1 = res1.json()
  232. if res1['result_code'] != 0:
  233. fail_reason = '测试服务器生成序列号发生异常'
  234. return self.generateFail(quantity, fail_reason, Log, response)
  235. res2 = requests.post(url=url2, data=requests_data, timeout=2 * 60)
  236. if res2.status_code != 200:
  237. fail_reason = '请求国内服务器生成序列号响应状态码异常'
  238. return self.generateFail(quantity, fail_reason, Log, response)
  239. res2 = res2.json()
  240. if res2['result_code'] != 0:
  241. fail_reason = '国内服务器生成序列号发生异常'
  242. return self.generateFail(quantity, fail_reason, Log, response)
  243. res3 = requests.post(url=url3, data=requests_data, timeout=2 * 60)
  244. if res3.status_code != 200:
  245. fail_reason = '请求美国服务器生成序列号响应状态码异常'
  246. return self.generateFail(quantity, fail_reason, Log, response)
  247. res3 = res3.json()
  248. if res3['result_code'] != 0:
  249. fail_reason = '美国服务器生成序列号发生异常'
  250. return self.generateFail(quantity, fail_reason, Log, response)
  251. # res4 = requests.post(url=url4, data=requests_data, timeout=2 * 60)
  252. # if res4.status_code != 200:
  253. # fail_reason = '请求欧洲服务器生成序列号响应状态码异常'
  254. # return self.generateFail(quantity, fail_reason, Log, response)
  255. # res4 = res4.json()
  256. # if res4['code'] != 0:
  257. # fail_reason = '欧洲服务器生成序列号发生异常'
  258. # return self.generateFail(quantity, fail_reason, Log, response)
  259. return response.json(0)
  260. except Exception as e:
  261. djangoLogger = logging.getLogger('django')
  262. djangoLogger.exception(repr(e))
  263. return response.json(176)
  264. def generateFail(self, quantity, fail_reason, Log, response):
  265. operation = '生成{}个序列号{}: {}'.format(quantity, '失败', fail_reason)
  266. Log.operation = operation
  267. Log.save()
  268. return response.json(177)
  269. def do_quantity(self, userID, response):
  270. """
  271. 查询当前可用的UID的数量
  272. :param userID: 响应对象
  273. :param response: 请求
  274. :return:
  275. """
  276. user_qs = UserModel.objects.filter(id=userID)
  277. if not user_qs.exists():
  278. return response.json(9)
  279. unused_serial_number_count = SerialNumberModel.objects.filter(use_status=0).count()
  280. remain_qs = CompanySerialModel.objects.filter(status=1).count()
  281. company_qs = CompanyModel.objects.values('id', 'name')
  282. try:
  283. company_serial_list = [] # 剩余已分配未使用的序列号数量
  284. company_not_used_list = [] # 已使用的序列号数量
  285. for company in company_qs:
  286. id = company['id']
  287. name = company['name']
  288. not_used_qs = CompanySerialModel.objects.filter(status=3, company_id=id).count()
  289. res = {
  290. 'name': name,
  291. 'number': not_used_qs
  292. }
  293. company_not_used_list.append(res)
  294. for company in company_qs:
  295. id = company['id']
  296. name = company['name']
  297. count = CompanySerialModel.objects.filter(status=1, company_id=id).count()
  298. # 获取最后一个序列号
  299. if count == 0:
  300. last_serial = ''
  301. else:
  302. last_serial = CompanySerialModel.objects.filter(status=1, company_id=id).order_by('-id').values(
  303. 'serial_number')[:1][0]['serial_number']
  304. res = {
  305. 'name': name,
  306. 'numbers': count,
  307. 'lastSerial': last_serial
  308. }
  309. company_serial_list.append(res)
  310. res_data = {'code': 0, 'companyRemainCount': company_not_used_list, 'companyRemain': company_serial_list,
  311. 'unused_serial_number_count': unused_serial_number_count,
  312. 'unused_all_count': remain_qs}
  313. return response.json(0, {'data': res_data})
  314. except Exception as e:
  315. print(e)
  316. return response.json(500)
  317. def do_create(self, request_dict, response):
  318. quantity = int(request_dict.get('quantity', 0))
  319. if not quantity:
  320. return response.json(444)
  321. try:
  322. try:
  323. sum = SerialNumberModel.objects.last().id
  324. except:
  325. sum = 0
  326. serial_number_bulk = []
  327. now_time = int(time.time())
  328. algorithm = AlgorithmBaseOn35()
  329. for i in range(quantity):
  330. serial_number = algorithm.getLetter(sum)
  331. sum += 1 # sum每次递增1
  332. # 前面补0至六位
  333. serial_number = (6 - len(serial_number)) * '0' + serial_number
  334. serial_number_bulk.append(SerialNumberModel(serial_number=serial_number, add_time=now_time))
  335. # 开启事务写入
  336. with transaction.atomic():
  337. SerialNumberModel.objects.bulk_create(serial_number_bulk)
  338. return response.json(0)
  339. except Exception as e:
  340. print(e)
  341. return response.json(500, repr(e))
  342. # 分配序列号
  343. @transaction.atomic
  344. def do_allot(self, request_dict, response):
  345. username = request_dict.get('username', None)
  346. quantity = int(request_dict.get('quantity', None))
  347. token = request_dict.get('token', None)
  348. token = TokenObject(token)
  349. if token.code != 0:
  350. return response.json(token.code)
  351. user = UserModel.objects.get(id=token.userID)
  352. if not user or '0' not in user.permission:
  353. return response.json(404)
  354. # 要分配的对象
  355. allot_user_qs = UserModel.objects.filter(username=username)
  356. if not allot_user_qs.exists():
  357. return response.json(444, 'username')
  358. # 取出对应区域可用的UID分配给allot_user
  359. sn_qs = SerialNumberModel.objects.filter(use_status=3)[0:quantity]
  360. sns = []
  361. for sn in sn_qs:
  362. sns.append(sn.serial_number)
  363. cs_qs = CompanySerialModel.objects.filter(serial_number__in=sns, status=3)
  364. sns = []
  365. for cs in cs_qs:
  366. sns.append(cs.serial_number)
  367. sn_qs = SerialNumberModel.objects.filter(serial_number__in=sns)
  368. count = sn_qs.count()
  369. if count < quantity:
  370. return response.json(444, '序列号不足')
  371. try:
  372. updates = []
  373. datas = []
  374. count = 0
  375. if sn_qs.exists():
  376. sn_qs = sn_qs[0:quantity]
  377. now_time = int(time.time())
  378. for i in range(len(sn_qs)):
  379. item = sn_qs[i]
  380. serialNumberModel = SerialNumberModel(
  381. id=item.id,
  382. serial_number=item.serial_number,
  383. status=item.status,
  384. use_status=1,
  385. add_time=item.add_time
  386. )
  387. CompanySerialModel.objects.filter(serial_number=item.serial_number).update(status=1)
  388. user_serial_number = UserSerialNumberModel()
  389. user_serial_number.serial_number = serialNumberModel
  390. user_serial_number.user = allot_user_qs[0]
  391. user_serial_number.add_time = now_time
  392. user_serial_number.update_time = now_time
  393. datas.append(user_serial_number)
  394. updates.append(serialNumberModel)
  395. if len(updates) % 5000 == 0:
  396. bulk_update(updates)
  397. UserSerialNumberModel.objects.bulk_create(datas)
  398. if len(updates) > 0:
  399. bulk_update(updates)
  400. UserSerialNumberModel.objects.bulk_create(datas)
  401. del datas
  402. del updates
  403. return response.json(0)
  404. else:
  405. return response.json(444)
  406. except Exception as e:
  407. print(e)
  408. return response.json(500)
  409. # 提供给pc端获取序列号
  410. def do_get_serial_number(self, request_dict, request, response):
  411. quantity = 1 # 只能取一个
  412. company_id = request_dict.get('company_id', None)
  413. token = request_dict.get('token', None)
  414. time_stamp = request_dict.get('time_stamp', None)
  415. mac = request_dict.get('mac', None)
  416. if not all([token, time_stamp, company_id]):
  417. return response.json(444)
  418. token = int(CommonService.decode_data(token))
  419. time_stamp = int(time_stamp)
  420. now_time = int(time.time())
  421. distance = now_time - time_stamp
  422. if token != time_stamp or distance > 60000 or distance < -60000: # 为了全球化时间控制在一天内
  423. return response.json(404)
  424. redisObj = RedisObject()
  425. # redis加锁,防止同时进行其他操作
  426. serial_operate_lock_key = 'serial_operate_lock'
  427. isLock = redisObj.CONN.setnx(serial_operate_lock_key, 1)
  428. if not isLock:
  429. return response.json(5)
  430. redisObj.CONN.expire(serial_operate_lock_key, 60)
  431. user_qs = UserModel.objects.filter(id=3)
  432. if not user_qs.exists():
  433. redisObj.del_data(key=serial_operate_lock_key) # redis解锁
  434. return response.json(9)
  435. user = user_qs[0]
  436. result = UserSerialNumberModel.objects.filter(user__id=user.id, serial_number__use_status=1). \
  437. aggregate(num=Count('serial_number__use_status'))
  438. us_qs = UserSerialNumberModel.objects.filter(user__id=user.id, serial_number__use_status=1). \
  439. values('serial_number__serial_number')
  440. sns = []
  441. for us in us_qs:
  442. sns.append(us['serial_number__serial_number'])
  443. cs_qs = CompanySerialModel.objects.filter(serial_number__in=sns, status=1, company__secret=company_id).values(
  444. 'serial_number', 'company__mark')
  445. sns = []
  446. mark = ""
  447. if cs_qs.exists():
  448. for cs in cs_qs:
  449. sns.append(cs['serial_number'])
  450. mark = cs_qs[0]['company__mark']
  451. # 判断是否还有可用的序列号
  452. count = result['num']
  453. sn_qs = SerialNumberModel.objects.filter(serial_number__in=sns)
  454. if not sn_qs.exists() or count < quantity:
  455. redisObj.del_data(key=serial_operate_lock_key) # redis解锁
  456. return response.json(444, '序列号不足')
  457. try:
  458. with transaction.atomic(): # 开启事务
  459. sn_qs = sn_qs[0: quantity]
  460. serial_number = sn_qs[0].serial_number
  461. # 防止重复获取序列号
  462. is_lock = redisObj.CONN.setnx(serial_number + 'serial_number_lock', 1)
  463. redisObj.CONN.expire(serial_number + 'serial_number_lock', 60)
  464. if not is_lock:
  465. return response.json(5)
  466. if not mac: # 不传入则分配mac
  467. mac_qs = MacModel.objects.filter().values('id', 'value', 'is_active')[0]
  468. if not mac_qs['is_active']:
  469. return response.json(175)
  470. mac = mac_qs['value']
  471. # 绑定mac地址成功后更新mac表
  472. next_mac = CommonService.updateMac(mac) # mac地址值+1;后3个字节为FF时返回None
  473. if next_mac:
  474. MacModel.objects.filter().update(value=next_mac, update_time=now_time) # 更新mac表的mac地址值
  475. else:
  476. MacModel.objects.filter().update(is_active=False, update_time=now_time)
  477. # 更新序列号表数据
  478. serialNumberModel = SerialNumberModel(
  479. id=sn_qs[0].id,
  480. serial_number=serial_number,
  481. status=sn_qs[0].status,
  482. use_status=2,
  483. add_time=now_time
  484. )
  485. updates = []
  486. updates.append(serialNumberModel)
  487. bulk_update(updates)
  488. # 更新企业关联序列号状态
  489. CompanySerialModel.objects.filter(serial_number=serial_number).update(status=2, update_time=now_time)
  490. # 记录操作日志
  491. ip = CommonService.get_ip_address(request)
  492. content = json.loads(json.dumps(request.POST))
  493. operation = '获取序列号:{}'.format(serial_number)
  494. log = {
  495. 'status': 200,
  496. 'content': json.dumps(content),
  497. 'ip': ip,
  498. 'time': now_time,
  499. 'url': 'serialNumber/getSerial',
  500. 'operation': operation,
  501. 'user': user_qs[0]
  502. }
  503. LogModel.objects.create(**log)
  504. redisObj.del_data(key=serial_operate_lock_key) # redis解锁
  505. return response.json(0, {'serial_number': serial_number + mark, 'mac': mac})
  506. except Exception as e:
  507. redisObj.del_data(key=serial_operate_lock_key) # redis解锁
  508. return response.json(500, repr(e))