From d4bfbce6c63f7b24a8d2b9f6edb51504e573d55e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Valentin=20Grou=C3=A8s?= <valentin.groues@uni.lu> Date: Tue, 28 Mar 2017 10:59:06 +0200 Subject: [PATCH] add contact attempts feature - #34 --- .gitignore | 4 +- .gitlab-ci.yml | 2 +- smash/smash/settings.py | 2 - smash/web/forms.py | 34 ++++- smash/web/models/__init__.py | 3 +- smash/web/models/availability.py | 1 - smash/web/models/constants.py | 14 ++ smash/web/models/contact_attempt.py | 28 ++++ smash/web/templates/appointments/add.html | 2 +- smash/web/templates/contact_attempt/add.html | 81 ++++++++++++ smash/web/templates/subjects/edit.html | 130 +++++++++++++------ smash/web/tests/__init__.py | 21 +++ smash/web/tests/test_api.py | 21 +-- smash/web/tests/test_model_appointment.py | 5 +- smash/web/tests/test_model_subject.py | 4 +- smash/web/tests/test_view_appointments.py | 14 +- smash/web/tests/test_view_contact_attempt.py | 53 ++++++++ smash/web/tests/test_view_kit_request.py | 27 +--- smash/web/tests/test_view_login.py | 18 +++ smash/web/tests/test_view_notifications.py | 16 +-- smash/web/tests/test_view_statistics.py | 15 +-- smash/web/tests/test_view_visit.py | 25 +--- smash/web/urls.py | 44 +++++++ smash/web/views/__init__.py | 1 + smash/web/views/contact_attempt.py | 20 +++ smash/web/views/subject.py | 4 +- 26 files changed, 438 insertions(+), 151 deletions(-) create mode 100644 smash/web/models/contact_attempt.py create mode 100644 smash/web/templates/contact_attempt/add.html create mode 100644 smash/web/tests/test_view_contact_attempt.py create mode 100644 smash/web/views/contact_attempt.py diff --git a/.gitignore b/.gitignore index 381cefd9..51fe0734 100644 --- a/.gitignore +++ b/.gitignore @@ -19,8 +19,8 @@ appointment-import/tmp.sql *.iml out .idea +<<<<<<< HEAD -#coverage tool .coverage -smash/htmlcov/* +smash/htmlcov/ diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index c4967b07..fd2748ce 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -17,6 +17,6 @@ test: script: - cp "local_settings_ci.py" "smash/smash/local_settings.py" - cd smash - - python manage.py makemigrations && python manage.py migrate + - python manage.py makemigrations web && python manage.py migrate - coverage run --source web manage.py test - coverage report -m diff --git a/smash/smash/settings.py b/smash/smash/settings.py index e8276ccc..cd99bb7e 100644 --- a/smash/smash/settings.py +++ b/smash/smash/settings.py @@ -15,7 +15,6 @@ import os # Build paths inside the project like this: os.path.join(BASE_DIR, ...) BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) - # Quick-start development settings - unsuitable for production # See https://docs.djangoproject.com/en/1.10/howto/deployment/checklist/ @@ -100,7 +99,6 @@ USE_L10N = True USE_TZ = True - # Static files (CSS, JavaScript, Images) # https://docs.djangoproject.com/en/1.10/howto/static-files/ diff --git a/smash/web/forms.py b/smash/web/forms.py index 80e00359..b5d6da32 100644 --- a/smash/web/forms.py +++ b/smash/web/forms.py @@ -1,17 +1,17 @@ -from datetime import datetime +import datetime from django import forms from django.forms import ModelForm, Form from django.utils.dates import MONTHS -from models import Subject, Worker, Appointment, Visit, AppointmentType +from models import Subject, Worker, Appointment, Visit, AppointmentType, ContactAttempt from models.constants import SUBJECT_TYPE_CHOICES """ Possible redundancy, but if need arises, contents of forms can be easily customized """ -CURRENT_YEAR = datetime.now().year +CURRENT_YEAR = datetime.datetime.now().year YEAR_CHOICES = tuple(range(CURRENT_YEAR, CURRENT_YEAR - 120, -1)) FUTURE_YEAR_CHOICES = tuple(range(CURRENT_YEAR, CURRENT_YEAR + 5, 1)) DATEPICKER_DATE_ATTRS = { @@ -79,6 +79,7 @@ class SubjectAddForm(ModelForm): validate_subject_nd_number(self) + def get_new_screening_number(screening_number_prefix): result_number = 0 subjects = Subject.objects.filter(screening_number__contains=screening_number_prefix) @@ -95,12 +96,14 @@ def get_new_screening_number(screening_number_prefix): return screening_number_prefix + str(result_number + 1).zfill(3) + def get_prefix_screening_number(user): prefix_screening_number = '' if (user is not None) and (user.screening_number_prefix is not None) and (user.screening_number_prefix != ""): prefix_screening_number = user.screening_number_prefix + "-" return prefix_screening_number + class SubjectDetailForm(ModelForm): class Meta: model = Subject @@ -268,6 +271,29 @@ class VisitAddForm(ModelForm): self.add_error('datetime_end', "End date must be after start date") +class ContactAttemptForm(ModelForm): + datetime_when = forms.DateTimeField(label='Contact on (YYYY-MM-DD HH:MM)', + widget=forms.DateTimeInput(DATETIMEPICKER_DATE_ATTRS) + ) + + class Meta: + model = ContactAttempt + fields = '__all__' + + def __init__(self, *args, **kwargs): + user = kwargs.pop('user', None) + if user is None: + raise TypeError("User not defined") + self.user = Worker.get_by_user(user) + if self.user is None: + raise TypeError("Worker not defined for: " + user.username) + subject = kwargs.pop('subject', None) + super(ContactAttemptForm, self).__init__(*args, **kwargs) + self.fields['subject'].initial = subject.id + self.fields['subject'].disabled = True + self.fields['worker'].initial = self.user + + class KitRequestForm(Form): start_date = forms.DateField(label="From date", widget=forms.DateInput(DATEPICKER_DATE_ATTRS, "%Y-%m-%d"), @@ -286,7 +312,7 @@ class StatisticsForm(Form): visit_choices = kwargs['visit_choices'] month = kwargs['month'] year = kwargs['year'] - now = datetime.now() + now = datetime.datetime.now() year_now = now.year number_of_years_for_statistics = year_now - START_YEAR_STATISTICS + 2 diff --git a/smash/web/models/__init__.py b/smash/web/models/__init__.py index 285641e8..0f14c5c3 100644 --- a/smash/web/models/__init__.py +++ b/smash/web/models/__init__.py @@ -17,6 +17,7 @@ from holiday import Holiday from item import Item from language import Language from subject import Subject +from contact_attempt import ContactAttempt def get_current_year(): @@ -24,4 +25,4 @@ def get_current_year(): __all__ = [FlyingTeam, Appointment, AppointmentType, Availability, Holiday, Item, Language, Location, Room, Subject, - Visit, Worker] + Visit, Worker, ContactAttempt] diff --git a/smash/web/models/availability.py b/smash/web/models/availability.py index 5ff9624c..8323a6ab 100644 --- a/smash/web/models/availability.py +++ b/smash/web/models/availability.py @@ -5,7 +5,6 @@ from django.db import models class Availability(models.Model): class Meta: app_label = 'web' - db_table = 'web_avaibility' person = models.ForeignKey("web.Worker", on_delete=models.CASCADE, verbose_name='Worker' diff --git a/smash/web/models/constants.py b/smash/web/models/constants.py index 19d93238..333f0baa 100644 --- a/smash/web/models/constants.py +++ b/smash/web/models/constants.py @@ -13,3 +13,17 @@ SUBJECT_TYPE_CHOICES = { } APPOINTMENT_TYPE_DEFAULT_COLOR = '#cfc600' APPOINTMENT_TYPE_DEFAULT_FONT_COLOR = '#00000' + +CONTACT_TYPES_EMAIL = 'E' +CONTACT_TYPES_PHONE = 'P' +CONTACT_TYPES_SMS = 'S' +CONTACT_TYPES_FAX = 'X' +CONTACT_TYPES_FACE2FACE = 'F' + +CONTACT_TYPES_CHOICES = ( + (CONTACT_TYPES_EMAIL, 'Email'), + (CONTACT_TYPES_FACE2FACE, 'Face to face'), + (CONTACT_TYPES_FAX, 'Fax'), + (CONTACT_TYPES_PHONE, 'Phone'), + (CONTACT_TYPES_SMS, 'SMS'), +) diff --git a/smash/web/models/contact_attempt.py b/smash/web/models/contact_attempt.py new file mode 100644 index 00000000..796b1ce5 --- /dev/null +++ b/smash/web/models/contact_attempt.py @@ -0,0 +1,28 @@ +# coding=utf-8 +from django.db import models + +from constants import CONTACT_TYPES_CHOICES, CONTACT_TYPES_PHONE + +__author__ = 'Valentin Grouès' + + +class ContactAttempt(models.Model): + subject = models.ForeignKey("web.Subject", + verbose_name='Subject' + ) + worker = models.ForeignKey("web.Worker", null=True, + verbose_name='Worker' + ) + type = models.CharField(max_length=2, default=CONTACT_TYPES_PHONE, choices=CONTACT_TYPES_CHOICES) + + datetime_when = models.DateTimeField(verbose_name="Contact on", help_text='When did the contact occurred?') + + success = models.BooleanField(default=False) + + comment = models.TextField(max_length=1024, null=True, blank=True) + + def __str__(self): + return "%s %s" % (self.subject, self.worker) + + def __unicode__(self): + return "%s %s" % (self.subject, self.worker) diff --git a/smash/web/templates/appointments/add.html b/smash/web/templates/appointments/add.html index 63c8d3b1..419e28dc 100644 --- a/smash/web/templates/appointments/add.html +++ b/smash/web/templates/appointments/add.html @@ -18,7 +18,7 @@ {% block page_header %}New appointment{% endblock page_header %} {% block page_description %}{% endblock page_description %} -{% block title %}{{ block.super }} - Add new appoitnment{% endblock %} +{% block title %}{{ block.super }} - Add new appointment{% endblock %} {% block breadcrumb %} {% include "appointments/breadcrumb.html" %} diff --git a/smash/web/templates/contact_attempt/add.html b/smash/web/templates/contact_attempt/add.html new file mode 100644 index 00000000..624407a5 --- /dev/null +++ b/smash/web/templates/contact_attempt/add.html @@ -0,0 +1,81 @@ +{% extends "_base.html" %} +{% load static %} +{% load filters %} + +{% block styles %} + {{ block.super }} + <link rel="stylesheet" href="{% static 'AdminLTE/plugins/awesomplete/awesomplete.css' %}"/> + + {% include "includes/datepicker.css.html" %} +{% endblock styles %} + +{% block ui_active_tab %}'subjects'{% endblock ui_active_tab %} +{% block page_header %}New contact attempt{% endblock page_header %} +{% block page_description %}{% endblock page_description %} + +{% block title %}{{ block.super }} - Add new contact attempt{% endblock %} + +{% block breadcrumb %} + {% include "subjects/breadcrumb.html" %} +{% endblock breadcrumb %} + +{% block maincontent %} + + {% block content %} + <div class="row"> + <div class="col-md-12"> + <div class="box box-success"> + <div class="box-header with-border"> + <h3 class="box-title">Enter contact attempt details</h3> + </div> + + + <form method="post" action="" class="form-horizontal"> + {% csrf_token %} + + <div class="box-body"> + {% for field in form %} + <div class="form-group {% if field.errors %}has-error{% endif %}"> + <label class="col-sm-4 col-lg-offset-1 col-lg-2 control-label"> + {{ field.label }} + </label> + + <div class="col-sm-8 col-lg-4"> + {{ field|add_class:'form-control' }} + </div> + + {% if field.errors %} + <span class="help-block"> + {{ field.errors }} + </span> + {% endif %} + </div> + {% endfor %} + </div><!-- /.box-body --> + <div class="box-footer"> + <div class="col-sm-6"> + <button type="submit" class="btn btn-block btn-success">Add</button> + </div> + <div class="col-sm-6"> + <a href="{% url 'web.views.subject_edit' subject_id %}" + class="btn btn-block btn-default">Cancel</a> + </div> + </div><!-- /.box-footer --> + </form> + </div> + + </div> + </div> + + {% endblock %} + + +{% endblock maincontent %} + +{% block scripts %} + {{ block.super }} + + <script src="{% static 'AdminLTE/plugins/awesomplete/awesomplete.min.js' %}"></script> + + {% include "includes/datetimepicker.js.html" %} +{% endblock scripts %} diff --git a/smash/web/templates/subjects/edit.html b/smash/web/templates/subjects/edit.html index 0c8b2f57..f14b4b28 100644 --- a/smash/web/templates/subjects/edit.html +++ b/smash/web/templates/subjects/edit.html @@ -23,53 +23,103 @@ {% block maincontent %} {% block content %} - <div class="box box-info"> - <div class="box-header with-border"> - <a href="{% url 'web.views.subjects' %}" class="btn btn-block btn-default" onclick="history.back()">Go - back (without changes)</a> + <div class="row"> + <p class="col-lg-3 pull-left"> + <a href="{% url 'web.views.subjects' %}" class="btn btn-block btn-default" + onclick="history.back()">Go + back (discard changes)</a> + </p> + <p class="col-md-2 pull-right"> <a href="{% url 'web.views.subject_visit_details' subject.id %}" type="button" class="btn btn-block btn-default">Subject's visits</a> - </div> - - {% comment %} <div class="box-header with-border"> - <h3 class="box-title">Details of subject</h3> - </div>{% endcomment %} - - <form method="post" action="" class="form-horizontal"> - {% csrf_token %} - - <div class="box-body"> - <div class="col-md-12"> - {% for field in form %} - <div class="col-md-6 form-group {% if field.errors %}has-error{% endif %}"> - <label for="{# TODO #}" class="col-sm-4 control-label"> - {{ field.label }} - </label> - - <div class="col-sm-8"> - {{ field|add_class:'form-control' }} - </div> - - {% if field.errors %} - <span class="help-block"> {{ field.errors }} </span> - {% endif %} - </div> - {% endfor %} + </p> + </div> + <div class="row"> + <div class="col-md-12"> + <div class="box box-success"> + <div class="box-header with-border"> + <h3>Subject details</h3> </div> - </div><!-- /.box-body --> - - <div class="box-footer"> - <div class="col-sm-6"> - <button type="submit" class="btn btn-block btn-success">Save</button> + <div class="box-body"> + <div class="col-md-12"> + <form method="post" action="" class="form-horizontal"> + {% csrf_token %} + + {% for field in form %} + <div class="col-md-6 form-group {% if field.errors %}has-error{% endif %}"> + <label for="{# TODO #}" class="col-sm-4 control-label"> + {{ field.label }} + </label> + + <div class="col-sm-8"> + {{ field|add_class:'form-control' }} + </div> + + {% if field.errors %} + <span class="help-block"> {{ field.errors }} </span> + {% endif %} + </div> + {% endfor %} + + + </form> + </div> + </div><!-- /.box-body --> + + + <div class="box-footer"> + <div class="col-sm-6"> + <button type="submit" class="btn btn-block btn-success">Save</button> + </div> + <div class="col-sm-6"> + <a href="{% url 'web.views.subjects' %}" class="btn btn-block btn-default" + onclick="history.back()">Cancel</a> + </div> + </div><!-- /.box-footer --> + </div><!-- /.box --> + </div><!-- /.col-md-12 --> + </div><!-- /.row --> + <div class="row"> + <div class="col-lg-6"> + <div class="box box-success"> + <div class="box-header with-border"> + <h3>Contact attempts <a title="add a new contact attempt" + id="add-contact-attempt" + href="{% url 'web.views.contact_add' subject.id %}" class="text-primary" + ><i class="fa fa-plus-circle text-success"></i></a></h3> </div> - <div class="col-sm-6"> - <a href="{% url 'web.views.subjects' %}" class="btn btn-block btn-default" - onclick="history.back()">Cancel</a> + <div class="box-body"> + <table class="table table-bordered table-striped"> + <thead> + <tr> + + <th class="text-center">When</th> + <th class="text-center">Who</th> + <th class="text-center">Type</th> + <th class="text-center">Success</th> + <th class="text-center">Comment</th> + </tr> + </thead> + <tbody> + {% for contact_attempt in contact_attempts %} + <tr> + <td>{{ contact_attempt.datetime_when }}</td> + <td>{{ contact_attempt.worker }}</td> + <td class="text-center">{{ contact_attempt.get_type_display }}</td> + <td class="text-center"> + <i class="fa {% if contact_attempt.success %}fa-check text-success{% else %}fa-times text-danger{% endif %}"></i> + </td> + <td>{{ contact_attempt.comment }}</td> + </tr> + {% endfor %} + </tbody> + </table> </div> - </div><!-- /.box-footer --> - </form> + </div> + + </div> </div> <div class="modal modal-danger fade" id="confirm-dead-resigned-mark-dialog" tabindex="-1" role="dialog"> diff --git a/smash/web/tests/__init__.py b/smash/web/tests/__init__.py index e69de29b..72b9f9ad 100644 --- a/smash/web/tests/__init__.py +++ b/smash/web/tests/__init__.py @@ -0,0 +1,21 @@ +from django.contrib.auth.models import User +from django.test import Client +from django.test import TestCase + +from functions import create_worker + + +class LoggedInTestCase(TestCase): + def setUp(self): + self.client = Client() + username = 'piotr' + password = 'top_secret' + self.user = User.objects.create_user( + username=username, email='jacob@bla', password=password) + self.client.login(username=username, password=password) + + +class LoggedInWithWorkerTestCase(LoggedInTestCase): + def setUp(self): + super(LoggedInWithWorkerTestCase, self).setUp() + self.worker = create_worker(self.user) diff --git a/smash/web/tests/test_api.py b/smash/web/tests/test_api.py index 7772b9a2..f7e2b6c0 100644 --- a/smash/web/tests/test_api.py +++ b/smash/web/tests/test_api.py @@ -1,29 +1,18 @@ # coding=utf-8 -import datetime import json -from django.contrib.auth.models import User -from django.test import TestCase -from django.test import Client from django.urls import reverse -from web.models import Visit -from web.api_views import cities -from web.tests.functions import create_subject, create_worker, create_appointment_type +from web.tests.functions import create_subject, create_appointment_type +from . import LoggedInWithWorkerTestCase __author__ = 'Piotr Gawron' -class TestApi(TestCase): +class TestApi(LoggedInWithWorkerTestCase): def setUp(self): + super(TestApi, self).setUp() self.subject = create_subject() - self.client = Client() - username = 'piotr' - password = 'top_secret' - self.user = User.objects.create_user( - username=username, email='jacob@bla', password=password) - self.worker = create_worker(self.user) - self.client.login(username=username, password=password) def test_cities(self): city_name = "some city" @@ -130,7 +119,7 @@ class TestApi(TestCase): found = False for type in appointment_types: - if type['type']==type_name: + if type['type'] == type_name: found = True self.assertTrue(found) diff --git a/smash/web/tests/test_model_appointment.py b/smash/web/tests/test_model_appointment.py index 252c807c..36525f58 100644 --- a/smash/web/tests/test_model_appointment.py +++ b/smash/web/tests/test_model_appointment.py @@ -1,9 +1,6 @@ from django.test import TestCase -from functions import create_subject, create_appointment -from functions import create_visit -from web.models import Appointment -from web.models import Visit +from functions import create_appointment class AppointmentModelTests(TestCase): diff --git a/smash/web/tests/test_model_subject.py b/smash/web/tests/test_model_subject.py index 6401a89e..5397e82b 100644 --- a/smash/web/tests/test_model_subject.py +++ b/smash/web/tests/test_model_subject.py @@ -27,8 +27,8 @@ class SubjectModelTests(TestCase): subject.mark_as_resigned() appointment_status = Appointment.objects.filter(id=appointment.id)[0].status - visit_finsihed = Visit.objects.filter(id=visit.id)[0].is_finished + visit_finished = Visit.objects.filter(id=visit.id)[0].is_finished self.assertTrue(subject.resigned) - self.assertTrue(visit_finsihed) + self.assertTrue(visit_finished) self.assertEquals(Appointment.APPOINTMENT_STATUS_CANCELLED, appointment_status) diff --git a/smash/web/tests/test_view_appointments.py b/smash/web/tests/test_view_appointments.py index 3e7c2d53..52c7e442 100644 --- a/smash/web/tests/test_view_appointments.py +++ b/smash/web/tests/test_view_appointments.py @@ -1,24 +1,14 @@ import datetime -from django.contrib.auth.models import User -from django.test import Client -from django.test import TestCase from django.urls import reverse from functions import create_subject, create_visit, create_appointment, create_worker from web.forms import AppointmentEditForm, SubjectEditForm from web.models import Appointment, Subject +from . import LoggedInTestCase -class AppointmentsViewTests(TestCase): - def setUp(self): - self.client = Client() - username = 'piotr' - password = 'top_secret' - self.user = User.objects.create_user( - username=username, email='jacob@bla', password=password) - self.client.login(username=username, password=password) - +class AppointmentsViewTests(LoggedInTestCase): def test_appointments_list_request(self): response = self.client.get(reverse('web.views.appointments')) self.assertEqual(response.status_code, 200) diff --git a/smash/web/tests/test_view_contact_attempt.py b/smash/web/tests/test_view_contact_attempt.py new file mode 100644 index 00000000..872bfa8f --- /dev/null +++ b/smash/web/tests/test_view_contact_attempt.py @@ -0,0 +1,53 @@ +import datetime + +from django.urls import reverse +from django.utils import timezone + +from functions import create_subject +from web.models import ContactAttempt +from web.models.constants import CONTACT_TYPES_EMAIL +from . import LoggedInWithWorkerTestCase + + +class ContactAttemptViewTests(LoggedInWithWorkerTestCase): + def test_contact_attempt_add_get(self): + subject = create_subject() + response = self.client.get(reverse('web.views.contact_add', kwargs={'subject_id': subject.id})) + self.assertContains(response, 'selected">{}'.format(self.worker), 1) + self.assertContains(response, 'selected">{}'.format(subject), 1) + + def test_contact_attempt_add_post_valid(self): + subject = create_subject() + self.assertEqual(0, ContactAttempt.objects.filter(subject=subject).count()) + now = datetime.datetime.now() + now_aware = timezone.make_aware(now, timezone.get_default_timezone()) + contact_type = CONTACT_TYPES_EMAIL + comment = "this is a comment" + form_data = {'datetime_when': now, 'worker': self.worker.id, 'type': contact_type, 'comment': comment} + response = self.client.post( + reverse('web.views.contact_add', kwargs={'subject_id': subject.id}), data=form_data) + # check correct redirection to suject edit page + self.assertRedirects(response, reverse('web.views.subject_edit', kwargs={'id': subject.id})) + contact_attempts = ContactAttempt.objects.filter(subject=subject).all() + self.assertEqual(1, len(contact_attempts)) + contact_attempt = contact_attempts[0] + self.assertEqual(now_aware, contact_attempt.datetime_when) + self.assertEqual(contact_type, contact_attempt.type) + self.assertEqual(subject, contact_attempt.subject) + self.assertEqual(self.worker, contact_attempt.worker) + self.assertEqual(comment, contact_attempt.comment) + self.assertFalse(contact_attempt.success) + # follow redirect to check if the new contact attempt is correctly listed + response = self.client.get(response.url) + self.assertContains(response, comment, 1) + + def test_contact_attempt_add_post_invalid(self): + subject = create_subject() + self.assertEqual(0, ContactAttempt.objects.filter(subject=subject).count()) + contact_type = CONTACT_TYPES_EMAIL + comment = "this is a comment" + form_data = {'type': contact_type, 'comment': comment} + response = self.client.post( + reverse('web.views.contact_add', kwargs={'subject_id': subject.id}), data=form_data) + self.assertContains(response, "This field is required", 2) + self.assertEqual(0, ContactAttempt.objects.filter(subject=subject).count()) diff --git a/smash/web/tests/test_view_kit_request.py b/smash/web/tests/test_view_kit_request.py index 556308fd..3aa380bc 100644 --- a/smash/web/tests/test_view_kit_request.py +++ b/smash/web/tests/test_view_kit_request.py @@ -1,23 +1,16 @@ import datetime -from django.test import TestCase, RequestFactory from django.urls import reverse -from functions import create_user, create_appointment_type, create_appointment +from functions import create_appointment_type, create_appointment from web.models import Item, Appointment -from web.views.kit import kit_requests from web.views.notifications import get_today_midnight_date +from . import LoggedInTestCase -class ViewFunctionsTests(TestCase): - def setUp(self): - self.factory = RequestFactory() - self.user = create_user() - +class ViewFunctionsTests(LoggedInTestCase): def test_kit_requests(self): - request = self.factory.get(reverse('web.views.kit_requests')) - request.user = self.user - response = kit_requests(request) + response = self.client.get(reverse('web.views.kit_requests')) self.assertEqual(response.status_code, 200) def test_kit_requests_2(self): @@ -32,9 +25,7 @@ class ViewFunctionsTests(TestCase): appointment.appointment_types.add(appointment_type) appointment.save() - request = self.factory.get(reverse('web.views.kit_requests')) - request.user = self.user - response = kit_requests(request) + response = self.client.get(reverse('web.views.kit_requests')) self.assertEqual(response.status_code, 200) self.assertTrue(item_name in response.content) @@ -52,9 +43,7 @@ class ViewFunctionsTests(TestCase): appointment.status = Appointment.APPOINTMENT_STATUS_CANCELLED appointment.save() - request = self.factory.get(reverse('web.views.kit_requests')) - request.user = self.user - response = kit_requests(request) + response = self.client.get(reverse('web.views.kit_requests')) self.assertEqual(response.status_code, 200) self.assertFalse(item_name in response.content) @@ -71,9 +60,7 @@ class ViewFunctionsTests(TestCase): appointment.appointment_types.add(appointment_type) appointment.save() - request = self.factory.get(reverse('web.views.kit_requests')) - request.user = self.user - response = kit_requests(request) + response = self.client.get(reverse('web.views.kit_requests')) self.assertEqual(response.status_code, 200) self.assertTrue(item_name in response.content) diff --git a/smash/web/tests/test_view_login.py b/smash/web/tests/test_view_login.py index 4b5970a9..4e662d9d 100644 --- a/smash/web/tests/test_view_login.py +++ b/smash/web/tests/test_view_login.py @@ -1,4 +1,5 @@ # coding=utf-8 +from django.contrib import auth as django_auth from django.test import Client from django.test import TestCase from django.urls import reverse @@ -14,11 +15,28 @@ class TestLoginView(TestCase): password = 'top_secret' username = user.username login_url = reverse('web.views.login') + self.assertFalse(django_auth.get_user(self.client).is_authenticated()) response = self.client.post(login_url, data={'username': username, 'password': password}, follow=True) self.assertEqual(200, response.status_code) + self.assertTrue(django_auth.get_user(self.client).is_authenticated()) worker = Worker.get_by_user(user) self.assertIsNotNone(worker) worker.last_name = 'Grouès' worker.save() response = self.client.post(login_url, data={'username': username, 'password': password}, follow=True) self.assertEqual(200, response.status_code) + + def test_login_failed(self): + self.client = Client() + user = create_user() + username = user.username + login_url = reverse('web.views.login') + response = self.client.post(login_url, data={'username': username, 'password': 'wrong_password'}, follow=False) + self.assertEqual(302, response.status_code) + self.assertEqual('/login?error=login_failed', response.url) + self.assertFalse(django_auth.get_user(self.client).is_authenticated()) + + def test_logout(self): + self.test_login() + self.client.get(reverse('web.views.logout')) + self.assertFalse(django_auth.get_user(self.client).is_authenticated()) diff --git a/smash/web/tests/test_view_notifications.py b/smash/web/tests/test_view_notifications.py index 817ec07c..e4d4142d 100644 --- a/smash/web/tests/test_view_notifications.py +++ b/smash/web/tests/test_view_notifications.py @@ -1,8 +1,5 @@ import datetime -from django.contrib.auth.models import User -from django.test import TestCase, RequestFactory - from functions import create_appointment, create_location, create_worker, create_appointment_type from functions import create_subject from functions import create_visit @@ -21,16 +18,11 @@ from web.views.notifications import \ get_today_midnight_date, \ get_unfinished_appointments, \ get_unfinished_appointments_count, \ - get_unfinished_visits, get_active_visits_with_missing_appointments - + get_unfinished_visits +from . import LoggedInTestCase -class NotificationViewTests(TestCase): - def setUp(self): - # Every test needs access to the request factory. - self.factory = RequestFactory() - self.user = User.objects.create_user( - username='piotr', email='jacob@bla', password='top_secret') +class NotificationViewTests(LoggedInTestCase): def test_get_exceeded_visit_notifications_count(self): original_notification = get_visits_without_appointments_count(self.user) @@ -85,7 +77,7 @@ class NotificationViewTests(TestCase): appointment = create_appointment(visit) appointment.appointment_types.add(appointment_type) - appointment.status=Appointment.APPOINTMENT_STATUS_FINISHED + appointment.status = Appointment.APPOINTMENT_STATUS_FINISHED appointment.save() notification = get_visits_with_missing_appointments_count(self.user) diff --git a/smash/web/tests/test_view_statistics.py b/smash/web/tests/test_view_statistics.py index b070ca19..eac349b6 100644 --- a/smash/web/tests/test_view_statistics.py +++ b/smash/web/tests/test_view_statistics.py @@ -1,23 +1,14 @@ # coding=utf-8 from datetime import datetime -from django.contrib.auth.models import User -from django.test import Client -from django.test import TestCase from django.urls import reverse -__author__ = 'Valentin Grouès' +from . import LoggedInTestCase +__author__ = 'Valentin Grouès' -class TestStatisticsView(TestCase): - def setUp(self): - self.client = Client() - username = 'piotr' - password = 'top_secret' - self.user = User.objects.create_user( - username=username, email='jacob@bla', password=password) - self.client.login(username=username, password=password) +class TestStatisticsView(LoggedInTestCase): def test_statistics_request(self): url = reverse('web.views.statistics') response = self.client.get(url) diff --git a/smash/web/tests/test_view_visit.py b/smash/web/tests/test_view_visit.py index 21c16290..b8c0a286 100644 --- a/smash/web/tests/test_view_visit.py +++ b/smash/web/tests/test_view_visit.py @@ -1,29 +1,15 @@ import datetime -from django.test import Client -from django.test import TestCase from django.urls import reverse -from functions import \ - create_appointment, \ - create_appointment_type, \ - create_subject, \ - create_visit, \ - create_user +from functions import create_subject, create_visit, create_appointment, create_appointment_type from web.forms import VisitDetailForm, VisitAddForm -from web.models import Subject, Visit +from web.models import Visit from web.views.notifications import get_today_midnight_date +from . import LoggedInTestCase -class VisitViewTests(TestCase): - def setUp(self): - username = 'piotr' - password = 'top_secret' - - self.client = Client() - self.user = create_user(username, password) - self.client.login(username=username, password=password) - +class VisitViewTests(LoggedInTestCase): def test_visit_details_request(self): visit = create_visit() create_appointment(visit) @@ -47,7 +33,7 @@ class VisitViewTests(TestCase): response = self.client.post( reverse('web.views.visit_details', kwargs={'id': visit.id}), data=form_data) self.assertEqual(response.status_code, 200) - self.assertFalse("error" in response.content) + self.assertNotContains(response, "error") def test_render_add_visit(self): subject = create_subject() @@ -133,6 +119,5 @@ class VisitViewTests(TestCase): visit = create_visit() visit.datetime_begin = get_today_midnight_date() + datetime.timedelta(days=-10) visit.save() - response = self.client.get(reverse("web.views.unfinished_visits")) self.assertEqual(response.status_code, 200) diff --git a/smash/web/urls.py b/smash/web/urls.py index 6bfa4151..0a9c9be7 100644 --- a/smash/web/urls.py +++ b/smash/web/urls.py @@ -20,6 +20,11 @@ from django.conf.urls import url from web import views urlpatterns = [ + + #################### + # APPOINTMENTS # + #################### + url(r'^appointments$', views.appointment.appointments, name='web.views.appointments'), url(r'^appointments/unfinished$', views.appointment.unfinished_appointments, name='web.views.unfinished_appointments'), @@ -29,6 +34,10 @@ urlpatterns = [ url(r'^appointments/add/general$', views.appointment.appointment_add, name='web.views.appointment_add_general'), url(r'^appointments/edit/(?P<id>\d+)$', views.appointment.appointment_edit, name='web.views.appointment_edit'), + #################### + # VISITS # + #################### + url(r'^visits$', views.visit.visits, name='web.views.visits'), url(r'^visits/exceeded$', views.visit.exceeded_visits, name='web.views.exceeded_visits'), url(r'^visits/unfinished$', views.visit.unfinished_visits, name='web.views.unfinished_visits'), @@ -43,6 +52,10 @@ urlpatterns = [ url(r'^visits/add/(?P<subject_id>\d+)$', views.visit.visit_add, name='web.views.visit_add'), url(r'^visit/mark/(?P<id>\d+)/(?P<as_what>[A-z]+)$', views.visit.visit_mark, name='web.views.visit_mark'), + #################### + # SUBJECTS # + #################### + url(r'^subjects$', views.subject.subjects, name='web.views.subjects'), url(r'^subjects/no_visit$', views.subject.subject_no_visits, name='web.views.subject_no_visits'), url(r'^subjects/equire_contact$', views.subject.subject_require_contact, name='web.views.subject_require_contact'), @@ -51,6 +64,16 @@ urlpatterns = [ name='web.views.subject_visit_details'), url(r'^subjects/edit/(?P<id>\d+)$', views.subject.subject_edit, name='web.views.subject_edit'), + #################### + # CONTACTS # + #################### + + url(r'^subjects/(?P<subject_id>\d+)/contacts/add$', views.contact_attempt.contact_add, name='web.views.contact_add'), + + #################### + # DOCTORS # + #################### + url(r'^doctors$', views.doctor.doctors, name='web.views.doctors'), url(r'^doctors/add$', views.doctor.doctor_add, name='web.views.doctor_add'), url(r'^doctors/details/(?P<doctor_id>\d+)$', views.doctor.doctor_details, name='web.views.doctor_details'), @@ -61,6 +84,10 @@ urlpatterns = [ views.doctor.doctor_availability_delete, name='web.views.doctor_availability_delete'), + #################### + # EQUIPMENT # + #################### + url(r'^equipment_and_rooms$', views.equipment.equipment_and_rooms, name='web.views.equipment_and_rooms'), url(r'^equipment_and_rooms/eqdef$', views.equipment.equipment_def, name='web.views.equipment_def'), url(r'^equipment_and_rooms/kit_requests$', views.kit.kit_requests, name='web.views.kit_requests'), @@ -69,12 +96,29 @@ urlpatterns = [ url(r'^equipment_and_rooms/kit_requests/(?P<start_date>[\w-]+)/(?P<end_date>[\w-]+)/$', views.kit.kit_requests_send_mail, name='web.views.kit_requests_send_mail'), + #################### + # MAIL # + #################### + url(r'^mail_templates$', views.mails.mail_templates, name='web.views.mail_templates'), + + #################### + # STATISTICS # + #################### + url(r'^statistics$', views.statistics.statistics, name='web.views.statistics'), + #################### + # EXPORT # + #################### + url(r'^export$', views.export.export, name='web.views.export'), url(r'^export/(?P<type>[A-z]+)$', views.export.export_to_csv2, name='web.views.export_to_csv2'), + #################### + # AUTH # + #################### + url(r'^login$', views.auth.login, name='web.views.login'), url(r'^logout$', views.auth.logout, name='web.views.logout'), diff --git a/smash/web/views/__init__.py b/smash/web/views/__init__.py index 1ae7ae1c..49820e91 100644 --- a/smash/web/views/__init__.py +++ b/smash/web/views/__init__.py @@ -60,3 +60,4 @@ import kit import mails import statistics import export +import contact_attempt diff --git a/smash/web/views/contact_attempt.py b/smash/web/views/contact_attempt.py new file mode 100644 index 00000000..229e11f8 --- /dev/null +++ b/smash/web/views/contact_attempt.py @@ -0,0 +1,20 @@ +from django.shortcuts import redirect, get_object_or_404 + +from . import wrap_response +from ..forms import ContactAttemptForm +from ..models import Subject + + +def contact_add(request, subject_id): + subject = get_object_or_404(Subject, id=subject_id) + if request.method == 'POST': + form = ContactAttemptForm(request.POST, user=request.user, subject=subject) + form.instance.subject_id = subject_id + if form.is_valid(): + form.save() + return redirect('web.views.subject_edit', id=subject_id) + else: + form = ContactAttemptForm(user=request.user, subject=subject) + + return wrap_response(request, 'contact_attempt/add.html', + {'form': form, 'subject_id': subject_id}) diff --git a/smash/web/views/subject.py b/smash/web/views/subject.py index 37732d54..325e2eca 100644 --- a/smash/web/views/subject.py +++ b/smash/web/views/subject.py @@ -50,6 +50,7 @@ def subject_require_contact(request): def subject_edit(request, id): the_subject = get_object_or_404(Subject, id=id) + contact_attempts = the_subject.contactattempt_set.order_by('-datetime_when').all() was_dead = the_subject.dead was_resigned = the_subject.resigned if request.method == 'POST': @@ -68,7 +69,8 @@ def subject_edit(request, id): return wrap_response(request, 'subjects/edit.html', { 'form': form, - 'subject': the_subject + 'subject': the_subject, + 'contact_attempts': contact_attempts }) -- GitLab