DeviceReportController.py 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  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))
  120. if not (-12 <= tz_offset <= 14):
  121. raise ValueError("时区偏移超出范围")
  122. tz = pytz.FixedOffset(int(tz_offset * 60))
  123. # 计算日期范围
  124. query_days = int(request_dict.get('days', 7))
  125. client_now = datetime.now(tz)
  126. # 数据日期范围 = [今天 - query_days, 昨天]
  127. end_date = (client_now - timedelta(days=1)).date() # 昨天
  128. start_date = end_date - timedelta(days=query_days - 1) # 起始日期
  129. # 转换查询范围到UTC时间戳
  130. start_utc = int(tz.localize(
  131. datetime.combine(start_date, datetime.min.time())
  132. ).astimezone(pytz.utc).timestamp())
  133. end_utc = int(tz.localize(
  134. datetime.combine(end_date + timedelta(days=1), datetime.min.time())
  135. ).astimezone(pytz.utc).timestamp())
  136. # 查询数据库
  137. records = DeviceDailyReport.objects.filter(
  138. device_id=device_id,
  139. type=1,
  140. report_time__gte=start_utc,
  141. report_time__lt=end_utc
  142. ).order_by('report_time')
  143. # 构建日期-电量映射(直接使用上报日期)
  144. date_battery_map = {}
  145. for record in records:
  146. # 关键变更:直接使用上报日期作为数据日期
  147. record_date = datetime.fromtimestamp(record.report_time, tz).date()
  148. # 只记录查询范围内的数据
  149. if start_date <= record_date <= end_date:
  150. date_battery_map[record_date.isoformat()] = record.battery_level
  151. # 生成完整日期序列
  152. report_data = []
  153. for day_offset in range(query_days):
  154. current_date = start_date + timedelta(days=day_offset)
  155. report_data.append({
  156. "index": current_date.day - 1, # 日期下标从0开始
  157. "battery": date_battery_map.get(current_date.isoformat(), 0),
  158. "time": current_date.strftime("%Y-%m-%d")
  159. })
  160. return response.json(0, report_data)
  161. except Exception as e:
  162. LOGGER.error(f'查询设备电量上报列表异常: {e}')
  163. return response.json(0, {})