src/checkin/views.py
from dataclasses import dataclassfrom http import HTTPStatus from django.contrib import messagesfrom django.contrib.auth.mixins import PermissionRequiredMixinfrom django.http import HttpResponse, HttpResponseRedirectfrom django.shortcuts import get_object_or_404from django.urls import reversefrom django.utils import timezonefrom django.utils.html import escapefrom django.utils.translation import gettext_lazy as _from django.views import Viewfrom django.views.generic import TemplateView from card.views import RFIDViewfrom util.view_utils import PreventGetRequestsMixinfrom .models import Profile, RegisterProfile, Skill, SuggestSkill, UserSkill class AdminCheckInView(RFIDView): def card_number_valid(self, card_number): profiles = Profile.objects.filter(user__card_number=card_number) if not profiles.exists(): return HttpResponse(f"{escape(card_number)} is not registered", status=HTTPStatus.UNAUTHORIZED) if profiles.first().on_make: profiles.update(on_make=False) return HttpResponse('check out'.encode(), status=HTTPStatus.OK) else: profiles.update(on_make=True, last_checkin=timezone.now()) return HttpResponse('check in'.encode(), status=HTTPStatus.OK) class UserSkillListView(TemplateView): template_name = 'checkin/user_skill_list.html' expiry_time = (60 * 60) * 3 def is_checkin_expired(self, profile): return (timezone.now() - profile.last_checkin).seconds >= self.expiry_time def check_out_expired(self, profile): if self.is_checkin_expired(profile): profile.on_make = False profile.save() Cyclomatic complexity is too high in method get_context_data. (6)
Function `get_context_data` has a Cognitive Complexity of 8 (exceeds 5 allowed). Consider refactoring. def get_context_data(self, **kwargs): """ Creates dict with skill titles as keys and the highest corresponding skill level as its pair value (quick fix) to show on website. """ # skill_dict = UserSkill.objects.filter(profile__on_make=True).order_by("-skill_level") skill_dict = {} for profile in Profile.objects.filter(on_make=True): self.check_out_expired(profile) for user_skill in profile.user_skills.all(): skill = user_skill.skill if ((skill not in skill_dict or skill.skill_level > skill_dict[skill][0]) and not self.is_checkin_expired(profile)): skill_dict[skill] = (user_skill.skill_level, profile.last_checkin) context = super().get_context_data(**kwargs) context.update({ 'skill_dict': sorted(skill_dict.items(), key=lambda item: item[1][1], reverse=True), }) return context @dataclass(kw_only=True)# `[...]DataClass` might have been a better name, but `[...]Struct` is shorterclass CompletedCourseMessageStruct: completed: bool message: str usage_hint: str = None Cyclomatic complexity is too high in class ProfileDetailView. (9)class ProfileDetailView(TemplateView): template_name = 'checkin/profile_detail.html' Cyclomatic complexity is too high in method post. (6)
Function `post` has a Cognitive Complexity of 9 (exceeds 5 allowed). Consider refactoring. def post(self, request): try: rating = int(request.POST.get('rating')) skill_id = int(request.POST.get('skill')) except ValueError: return HttpResponseRedirect(reverse('profile_detail')) profile = request.user.profile skill = get_object_or_404(Skill, id=skill_id) if 0 <= rating <= 3: if UserSkill.objects.filter(skill=skill, profile=profile).exists(): if rating == 0: UserSkill.objects.filter(skill=skill, profile=profile).delete() else: UserSkill.objects.filter(skill=skill, profile=profile).update(skill_level=rating) elif rating != 0: UserSkill.objects.create(skill=skill, profile=profile, skill_level=rating) return HttpResponseRedirect(reverse('profile_detail')) Cyclomatic complexity is too high in method get_context_data. (9)
Function `get_context_data` has a Cognitive Complexity of 6 (exceeds 5 allowed). Consider refactoring. def get_context_data(self, **kwargs): user = self.request.user profile, _created = Profile.objects.get_or_create(user=user) completed_3d_printer = hasattr(user, 'printer_3d_course') completed_raise3d = completed_3d_printer and user.printer_3d_course.raise3d_course completed_sla = completed_3d_printer and user.printer_3d_course.sla_course completed_course_message_structs = [ CompletedCourseMessageStruct( completed=completed_3d_printer, message=(_("You have completed the 3D printer course") if completed_3d_printer else _("You have not taken the 3D printer course")), usage_hint=_( "To use a 3D printer, make a reservation in the calendar of one of the 3D printers on the “Reservations” page." ) if completed_3d_printer else None, ), CompletedCourseMessageStruct( completed=completed_raise3d, message=(_("You have completed the Raise3D printer course") if completed_raise3d else _("You have not taken the Raise3D printer course")), usage_hint=_( "To use a Raise3D printer, make a reservation in the calendar of one of the Raise3D printers on the “Reservations” page." ) if completed_raise3d else None, ), CompletedCourseMessageStruct( completed=completed_sla, message=(_("You have completed the SLA 3D printer course") if completed_sla else _("You have not taken the SLA 3D printer course")), usage_hint=_( "To use an SLA 3D printer, make a reservation in the calendar of one of the SLA 3D printers on the “Reservations” page." ) if completed_sla else None, ), ] """ Commented out because it's currently not in use; see the template code in `profile_detail_internal.html` user_skills = profile.user_skills.all() skill_dict = {} for user_skill in user_skills: skill = user_skill.skill if skill not in skill_dict or skill.skill_level > skill_dict[skill][0]: skill_dict[skill] = user_skill.skill_level """ context = super().get_context_data(**kwargs) context.update({ 'profile': profile, 'completed_course_message_structs': completed_course_message_structs, # Commented out for the same reason as above # 'userskill': user_skills, # 'skill_dict': skill_dict, # 'all_skills': Skill.objects.all(), }) return context Cyclomatic complexity is too high in class AdminSuggestSkillView. (14)class AdminSuggestSkillView(PermissionRequiredMixin, TemplateView): permission_required = ('checkin.add_suggestskill',) template_name = 'checkin/admin_suggest_skill.html' extra_context = { 'suggestions': SuggestSkill.objects.all(), } Cyclomatic complexity is too high in method post. (13)
Function `post` has a Cognitive Complexity of 15 (exceeds 5 allowed). Consider refactoring. def post(self, request): suggestion = request.POST.get('suggested-skill') suggestion_english = request.POST.get('suggested-skill-english') profile = request.user.profile image = request.FILES.get('image') if suggestion.strip() and not suggestion_english.strip(): messages.error(request, _("Enter both norwegian and english skill name")) return HttpResponseRedirect(reverse('admin_suggest_skill')) elif not suggestion.strip() and suggestion_english.strip(): messages.error(request, _("Enter both norwegian and english skill name")) return HttpResponseRedirect(reverse('admin_suggest_skill')) elif not suggestion.strip() and not suggestion_english.strip(): return HttpResponseRedirect(reverse('admin_suggest_skill')) if Skill.objects.filter(title=suggestion).exists() or Skill.objects.filter( title_en=suggestion_english).exists(): messages.error(request, _("Skill already exists!")) return HttpResponseRedirect(reverse('admin_suggest_skill')) else: if SuggestSkill.objects.filter(title=suggestion).exists(): s = SuggestSkill.objects.get(title=suggestion) s.voters.add(profile) if s.creator == profile or not s.image: s.image = image s.save() else: # does not work for some reason sug = SuggestSkill.objects.create(creator=profile, title=suggestion, title_en=suggestion_english, image=image) sug.voters.add(profile) if SuggestSkill.objects.get(title=suggestion).voters.count() >= 5: Skill.objects.create(title=suggestion, image=image) SuggestSkill.objects.get(title=suggestion).delete() messages.success(request, _("Skill added!")) Avoid too many `return` statements within this function. return HttpResponseRedirect(reverse('admin_suggest_skill')) class AdminRegisterCardView(RFIDView): def card_number_valid(self, card_number): if Profile.objects.filter(user__card__number=card_number).exists(): return HttpResponse(f"{escape(card_number)} is already registered", status=HTTPStatus.CONFLICT) else: RegisterProfile.objects.all().delete() RegisterProfile.objects.create(card_id=card_number, last_scan=timezone.now()) return HttpResponse("Card scanned", status=HTTPStatus.OK) class AdminProfilePictureUpdateView(PreventGetRequestsMixin, View): def post(self, request, *args, **kwargs): image = request.FILES.get('image') profile = request.user.profile profile.image = image profile.save() return HttpResponseRedirect(reverse('profile_detail'))