SerialNumberController.py 24 KB

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