DeviceReportController.py 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. # -*- encoding: utf-8 -*-
  2. """
  3. @File : DeviceReportController.py
  4. @Time : 2025/4/3 16:20
  5. @Author : stephen
  6. @Email : zhangdongming@asj6.wecom.work
  7. @Software: PyCharm
  8. """
  9. import time
  10. from datetime import timedelta, datetime
  11. import pytz
  12. from django.db.models import Sum, Value
  13. from django.db.models.functions import Coalesce
  14. from django.http import QueryDict
  15. from django.views import View
  16. from Ansjer.config import LOGGER
  17. from Model.models import DeviceDailyReport
  18. from Object.ResponseObject import ResponseObject
  19. from Object.TokenObject import TokenObject
  20. class DeviceReportView(View):
  21. def get(self, request, *args, **kwargs):
  22. request.encoding = 'utf-8'
  23. operation = kwargs.get('operation')
  24. return self.validation(request.GET, request, operation)
  25. def post(self, request, *args, **kwargs):
  26. request.encoding = 'utf-8'
  27. operation = kwargs.get('operation')
  28. return self.validation(request.POST, request, operation)
  29. def delete(self, request, *args, **kwargs):
  30. request.encoding = 'utf-8'
  31. operation = kwargs.get('operation')
  32. delete = QueryDict(request.body)
  33. if not delete:
  34. delete = request.GET
  35. return self.validation(delete, request, operation)
  36. def put(self, request, *args, **kwargs):
  37. request.encoding = 'utf-8'
  38. operation = kwargs.get('operation')
  39. put = QueryDict(request.body)
  40. return self.validation(put, request, operation)
  41. def validation(self, request_dict, request, operation):
  42. response = ResponseObject('cn')
  43. tko = TokenObject(request.META.get('HTTP_AUTHORIZATION'))
  44. if tko.code != 0:
  45. return response.json(tko.code)
  46. response.lang = tko.lang
  47. userID = tko.userID
  48. if operation == 'getBatteryCapacityList': # 获取电池电量列表
  49. return self.get_battery_capacity_list(userID, request, request_dict, response)
  50. elif operation == 'getWakeSleepData':
  51. return self.get_wake_sleep_data(userID, request_dict, response)
  52. @classmethod
  53. def get_wake_sleep_data(cls, userID, request_dict, response):
  54. """
  55. 获取设备睡眠唤醒统计数据
  56. :param userID: 响应对象
  57. :param request_dict: 参数对象
  58. :param response: 响应对象
  59. :return: 包含7天汇总统计和30天明细数据的JSON响应
  60. """
  61. try:
  62. # region 时间参数计算
  63. uid = request_dict.get('device_id', None)
  64. if not uid:
  65. return response.json(444)
  66. current_time = int(time.time())
  67. SECONDS_PER_DAY = 24 * 60 * 60 # 单日秒数常量
  68. # 计算时间范围(秒级时间戳)
  69. seven_days_ago = current_time - 7 * SECONDS_PER_DAY # 7天前
  70. thirty_days_ago = current_time - 30 * SECONDS_PER_DAY # 30天前
  71. # endregion
  72. # region 7天汇总统计 (使用Coalesce避免空值)
  73. seven_days_stats = DeviceDailyReport.objects.filter(
  74. device_id=uid,
  75. report_time__range=(seven_days_ago, current_time) # 时间范围查询
  76. ).aggregate(
  77. total_human_detection=Coalesce(Sum('human_detection'), Value(0)),
  78. total_working_hours=Coalesce(Sum('working_hours'), Value(0)),
  79. total_wake_sleep=Coalesce(Sum('wake_sleep'), Value(0))
  80. )
  81. # endregion
  82. # region 30天明细数据(按时间倒序)
  83. thirty_days_details = DeviceDailyReport.objects.filter(
  84. device_id=uid,
  85. report_time__gte=thirty_days_ago,
  86. report_time__lte=current_time
  87. ).order_by('-report_time').values_list(
  88. 'human_detection',
  89. 'working_hours',
  90. 'wake_sleep',
  91. 'battery_level',
  92. 'report_time',
  93. named=True # 使用命名元组方便访问
  94. )
  95. # endregion
  96. # 组合返回数据结构
  97. result_data = {
  98. **seven_days_stats,
  99. 'detail_report': list(thirty_days_details) # 转换查询集为列表
  100. }
  101. return response.json(0, result_data)
  102. except Exception as e:
  103. error_msg = f"查询异常: {str(e)},userID: {userID}"
  104. error_line = e.__traceback__.tb_lineno
  105. LOGGER.error(f"{error_msg} 行号: {error_line}")
  106. # 返回安全空数据
  107. return response.json(500, {
  108. 'total_human_detection': 0,
  109. 'total_working_hours': 0,
  110. 'total_wake_sleep': 0,
  111. 'detail_report': []})
  112. @classmethod
  113. def get_battery_capacity_list(cls, userID, request, request_dict, response):
  114. try:
  115. # 参数校验
  116. device_id = request_dict.get('device_id')
  117. if not device_id:
  118. return response.json(444, "设备ID不能为空")
  119. tz_offset = float(request_dict.get('tz', 0)) # 允许小数时区如5.5
  120. if not (-12 <= tz_offset <= 14):
  121. raise ValueError
  122. tz = pytz.FixedOffset(int(tz_offset * 60)) # 转换为分钟偏移
  123. # 计算日期范围(关键修改:end_time取昨日23:59:59)
  124. query_days = int(request_dict.get('days', 7))
  125. client_now = datetime.now(tz)
  126. end_time = client_now.replace(
  127. hour=23, minute=59, second=59
  128. ) - timedelta(days=1) # 截止到客户端昨日
  129. start_time = end_time - timedelta(days=query_days - 1)
  130. start_time = start_time.replace(hour=0, minute=0, second=0)
  131. # 转换时间范围到UTC时间戳
  132. start_utc_timestamp = int(start_time.astimezone(pytz.utc).timestamp())
  133. end_utc_timestamp = int(end_time.astimezone(pytz.utc).timestamp())
  134. # 查询数据库
  135. records = DeviceDailyReport.objects.filter(
  136. device_id=device_id,
  137. type=1,
  138. report_time__gte=start_utc_timestamp,
  139. report_time__lte=end_utc_timestamp
  140. ).order_by('report_time')
  141. # 构建日期-电量映射(按客户端时区)
  142. date_battery_map = {}
  143. for record in records:
  144. local_date = datetime.fromtimestamp(record.report_time, tz).date()
  145. date_str = local_date.strftime("%Y-%m-%d")
  146. date_battery_map[date_str] = record.battery_level
  147. # 生成完整日期序列
  148. report_data = []
  149. for day_offset in range(query_days):
  150. current_date = (end_time.date() - timedelta(days=query_days - 1 - day_offset))
  151. report_data.append({
  152. "index": current_date.day - 1, # 日期下标从0开始
  153. "battery": date_battery_map.get(current_date.strftime("%Y-%m-%d"), 0),
  154. "time": current_date.strftime("%Y-%m-%d")
  155. })
  156. return response.json(0, report_data)
  157. except Exception as e:
  158. LOGGER.error('查询设备电量上报列表异常error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  159. return response.json(0, {})