Объект Django 'When' не является итерируемым
Я использую фильтр django-filter, чтобы показать часы работы сотрудников.
models.py:
...
class Attendance(models.Model):
user = models.ForeignKey(User, on_delete=models.CASCADE, related_name="Attendances")
day = models.DateField()
start_time = models.TimeField()
end_time = models.TimeField()
is_confirmed = models.BooleanField()
task = models.ForeignKey(Tasks, on_delete=models.CASCADE)
class Task(models.Model):
title = models.CharField(max_length=255)
project = models.ForeignKey(Project, on_delete=models.CASCADE, related_name="Tasks")
assignee = models.ForeignKey(User, on_delete=models.CASCADE)
...
filter.py:
class WorkhourFilter(django_filters.FilterSet):
user = django_filters.ModelChoiceFilter(queryset=User.objects.filter(is_superuser=False),field_name="user")
project = django_filters.ModelChoiceFilter(queryset=Project.objects.all(), field_name="task__project")
year = django_filters.ChoiceFilter(choices=tuple((x, x) for x in settings.YEARS_LIST),method='year_search')
month = django_filters.ChoiceFilter(choices=tuple((i, x) for i, x in enumerate(settings.MONTH_LIST)),method='month_search' )
class Meta:
model = Attendance
fields = ['user']
def __init__(self, *args, **kwargs):
super(WorkhourFilter, self).__init__(*args, **kwargs)
start_date = ...
end_date = ...
self.queryset = self.Meta.model.objects.filter(user=self.request.user,
day__gte=start_date, day__lte=end_date).values('task__project__title',
'task__project').annotate(
raw_work_hour=Sum(F('end_time') - F('start_time'), ),
confirmed_work_hour=Sum(Case(When(is_verified=1,
then=F('end_time') -
F('start_time'))))).order_by('task__project')
Я получаю 'When' object is not iterable
ошибку.
confirmed_work_hour
имеет проблему, но сам по себе raw_work_hour
не является проблемой.
Этот запрос хорошо работает в get_queryset()
соответствующем представлении, но когда я пытаюсь применить фильтры, он ломается.
trace :
Django Version: 4.2.5
Python Version: 3.9.13
Installed Applications:
['django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
'django_extensions',
'crispy_forms',
'crispy_bootstrap5',
'django_tables2',
'django_htmx',
'django_filters',
'ckeditor',
'ckeditor_uploader',
'guardian',
'django_cascading_dropdown_widget',
'dashboard.apps.DashboardsConfig',
'auth.apps.AuthConfig']
Installed Middleware:
['django.middleware.security.SecurityMiddleware',
'django.contrib.sessions.middleware.SessionMiddleware',
'django.middleware.common.CommonMiddleware',
'django.middleware.csrf.CsrfViewMiddleware',
'django.contrib.auth.middleware.AuthenticationMiddleware',
'django.contrib.messages.middleware.MessageMiddleware',
'django.middleware.clickjacking.XFrameOptionsMiddleware',
'django_htmx.middleware.HtmxMiddleware']
Traceback (most recent call last):
File "D:\Python\DjangoTest\venv\lib\site-packages\django\core\handlers\exception.py", line 55, in inner
response = get_response(request)
File "D:\Python\DjangoTest\venv\lib\site-packages\django\core\handlers\base.py", line 197, in _get_response
response = wrapped_callback(request, *callback_args, **callback_kwargs)
File "D:\Python\DjangoTest\venv\lib\site-packages\django\views\generic\base.py", line 104, in view
return self.dispatch(request, *args, **kwargs)
File "D:\Python\DjangoTest\venv\lib\site-packages\django\contrib\auth\mixins.py", line 73, in dispatch
return super().dispatch(request, *args, **kwargs)
File "D:\Python\DjangoTest\venv\lib\site-packages\braces\views\_access.py", line 438, in dispatch
return super(StaffuserRequiredMixin, self).dispatch(
File "D:\Python\DjangoTest\venv\lib\site-packages\braces\views\_access.py", line 375, in dispatch
return super(GroupRequiredMixin, self).dispatch(
File "D:\Python\DjangoTest\venv\lib\site-packages\django\views\generic\base.py", line 143, in dispatch
return handler(request, *args, **kwargs)
File "D:\Python\DjangoTest\venv\lib\site-packages\django_filters\views.py", line 74, in get
self.filterset = self.get_filterset(filterset_class)
File "D:\Python\DjangoTest\venv\lib\site-packages\django_filters\views.py", line 38, in get_filterset
return filterset_class(**kwargs)
File "D:\Python\DjangoTest\dashboard\filters\workhour_project.py", line 71, in __init__
confirmed_work_hour=Sum(Case(When(is_verified=1, then=F('end_time') - F('start_time')))),
File "D:\Python\DjangoTest\venv\lib\site-packages\sqlparse\sql.py", line 160, in __init__
[setattr(token, 'parent', self) for token in self.tokens]
Exception Type: TypeError at /report/workhour/by_project
Exception Value: 'When' object is not iterable
проблема не в When
, проблема в том, что Case(When
завернуты в Count
.
В вашем случае решением будет разделить Case(When и Count на две annottate(
декларации:
manager = self.Meta.model.objects
queryset = manager.filter(user=self.request.user, day__gte=start_date, day__lte=end_date)
queryset = queryset.values('task__project__title', 'task__project').order_by('task__project')
queryset = queryset.annotate(raw_work_hour=Sum(F('end_time') - F('start_time'), ), work_hour=Case(When(is_verified=1, then=F('end_time') - F('start_time')), default=0))
Но я не понимаю, что ты хочешь Sum
.
queryset = queryset.annotate(confirmed_work_hour = Sum('work_hour')) # Probably it is not needed.
Спасибо всем друзьям, которые нашли время, чтобы решить мою проблему.
Посмотрев еще раз на error trace
в последних строках, я заметил неправильный импорт. Конечно, последний комментарий @willeM_VanOnsem заставил меня снова посмотреть на импорт, и я благодарен ему.
Команда Case
была ошибочно импортирована из sqlparse.sql
, когда ее следовало импортировать из django.db.models
.
После применения вышеуказанного изменения код заработал без проблем.
Нужная строка в трассировке, указывающая на пакет sqlparse
:
File "D:\Python\DjangoTest\venv\lib\site-packages\sqlparse\sql.py", line 160, in __init__
[setattr(token, 'parent', self) for token in self.tokens]