diff --git a/requirements.txt b/requirements.txt
index fb33ee6b7111cec03803cd7fc6c823d7613070b4..d63534e9e13c0d2846f3470378ae705607cccdae 100644
--- a/requirements.txt
+++ b/requirements.txt
@@ -1,3 +1,4 @@
+pycurl==7.43.0.2
 asn1crypto==0.24.0
 Babel==2.6.0
 backports.functools-lru-cache==1.5
@@ -36,7 +37,6 @@ phonenumberslite==8.9.14
 Pillow==3.4.2
 psycopg2==2.7.6.1
 pycparser==2.19
-pycurl==7.43.0.2
 pyexcel==0.5.3
 pyexcel-io==0.5.9.1
 pyexcel-webio==0.1.4
diff --git a/smash/web/forms/appointment_form.py b/smash/web/forms/appointment_form.py
index a07294741c65d29d1482b27e4f1cce1d4f2a72e9..3966c13ea7a11f97bab74defb7f38c20a5b29082 100644
--- a/smash/web/forms/appointment_form.py
+++ b/smash/web/forms/appointment_form.py
@@ -2,11 +2,12 @@ import logging
 from collections import OrderedDict
 
 from django import forms
+from django.db.models.query import QuerySet
 from django.forms import ModelForm
 
-from web.models.worker_study_role import WORKER_STAFF
 from web.forms.forms import DATETIMEPICKER_DATE_ATTRS, APPOINTMENT_TYPES_FIELD_POSITION
-from web.models import Appointment, Worker, AppointmentTypeLink, AppointmentType
+from web.models import Appointment, Worker, AppointmentTypeLink, AppointmentType, Provenance
+from web.models.worker_study_role import WORKER_STAFF
 from web.views.notifications import get_filter_locations
 
 logger = logging.getLogger(__name__)
@@ -22,6 +23,47 @@ class AppointmentForm(ModelForm):
         self.fields['worker_assigned'].queryset = Worker.get_workers_by_worker_type(WORKER_STAFF).filter(
             locations__in=get_filter_locations(self.user)).distinct().order_by('first_name', 'last_name')
 
+    def save_changes(self):
+        for change in self.changes:
+            if change.modified_table_id is None:
+                change.modified_table_id = self.instance.id
+            change.save()
+
+    def register_changes(self):
+        self.changes = []
+        for field in self.changed_data:
+            new_value = self.cleaned_data[field] or self.data[field]
+            if isinstance(new_value, QuerySet):
+                new_human_values = '; '.join([str(element) for element in new_value]) 
+                new_value = ','.join([str(element.id) for element in new_value]) #overwrite variable
+                #old value
+                if self.instance.id: #update instance
+                    list_of_values = getattr(self.instance, field).all()
+                    old_human_values = '; '.join([str(element) for element in list_of_values]) 
+                    previous_value = ','.join([str(element.id) for element in list_of_values]) 
+                else: #new instance
+                    old_human_values = ''
+                    previous_value = ''
+                # description
+                description = '{} changed from "{}" to "{}"'.format(field, old_human_values, new_human_values)
+            else:
+                if self.instance.id:  # update instance
+                    previous_value = str(getattr(self.instance, field))
+                else:
+                    previous_value = ''
+                new_value = str(self.cleaned_data[field])
+                description = '{} changed from "{}" to "{}"'.format(field, previous_value, new_value)
+
+            p = Provenance(modified_table=Appointment._meta.db_table,
+                           modified_table_id=self.instance.id,
+                           modification_author=self.user,
+                           previous_value=previous_value,
+                           new_value=new_value,
+                           modification_description=description,
+                           modified_field=field,
+                           )
+            self.changes.append(p)
+
 
 class AppointmentDetailForm(AppointmentForm):
     class Meta:
@@ -64,10 +106,16 @@ class AppointmentEditForm(AppointmentForm):
         location = self.cleaned_data['location']
         if self.user.locations.filter(id=location.id).count() == 0:
             self.add_error('location', "You cannot create appointment for this location")
+            raise forms.ValidationError("You cannot create appointment for this location")
         else:
             return location
 
+    def clean(self):
+        if len(self.errors) == 0:
+            self.register_changes()  # right before instance is changed
+
     def save(self, commit=True):
+
         appointment = super(AppointmentEditForm, self).save(commit)
         # if appointment date change, remove appointment_type links
         if 'datetime_when' in self.changed_data:
@@ -89,6 +137,8 @@ class AppointmentEditForm(AppointmentForm):
                 appointment_type_link = AppointmentTypeLink(appointment=appointment,
                                                             appointment_type=appointment_type)
                 appointment_type_link.save()
+
+        self.save_changes()
         return appointment
 
 
@@ -117,12 +167,16 @@ class AppointmentAddForm(AppointmentForm):
                                                                              )
         fields['worker_assigned'].widget.attrs = {'class': 'search_worker_availability'}
         fields['datetime_when'].widget.attrs = {'class': 'start_date', 'placeholder': 'yyyy-mm-dd HH:MM',
-        'pattern': '[0-9]{4}-[0-9]{2}-[0-9]{2} [0-9]{2}:[0-9]{2}'}
+                                                'pattern': '[0-9]{4}-[0-9]{2}-[0-9]{2} [0-9]{2}:[0-9]{2}'}
         fields['length'].widget.attrs = {'class': 'appointment_duration'}
 
         self.fields = fields
         self.fields['location'].queryset = get_filter_locations(self.user)
 
+    def clean(self):
+        if len(self.errors) == 0:
+            self.register_changes()  # right before instance is changed
+
     def clean_location(self):
         location = self.cleaned_data['location']
         if self.user.locations.filter(id=location.id).count() == 0:
@@ -136,4 +190,5 @@ class AppointmentAddForm(AppointmentForm):
         for appointment_type in appointment_types:
             appointment_type_link = AppointmentTypeLink(appointment=appointment, appointment_type=appointment_type)
             appointment_type_link.save()
+        self.save_changes()
         return appointment
diff --git a/smash/web/migrations/0141_auto_20200319_1040.py b/smash/web/migrations/0141_auto_20200319_1040.py
new file mode 100644
index 0000000000000000000000000000000000000000..a47dfea4765b67f9455d92a7234844ad40ae73ed
--- /dev/null
+++ b/smash/web/migrations/0141_auto_20200319_1040.py
@@ -0,0 +1,20 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.11.5 on 2020-03-19 10:40
+from __future__ import unicode_literals
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('web', '0140_auto_20190528_0953'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='appointmenttype',
+            name='calendar_font_color',
+            field=models.CharField(default=b'#00000', max_length=2000, verbose_name=b'Calendar font color'),
+        ),
+    ]
diff --git a/smash/web/migrations/0142_provenance.py b/smash/web/migrations/0142_provenance.py
new file mode 100644
index 0000000000000000000000000000000000000000..34e1568b71a6a267beeef1f3793c2f1252255be5
--- /dev/null
+++ b/smash/web/migrations/0142_provenance.py
@@ -0,0 +1,29 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.11.5 on 2020-03-19 10:50
+from __future__ import unicode_literals
+
+from django.db import migrations, models
+import django.db.models.deletion
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('web', '0141_auto_20200319_1040'),
+    ]
+
+    operations = [
+        migrations.CreateModel(
+            name='Provenance',
+            fields=[
+                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('modified_table', models.CharField(max_length=1024, verbose_name=b'Modified table')),
+                ('modified_table_id', models.CharField(max_length=1024, verbose_name=b'Modified table row')),
+                ('modification_date', models.DateTimeField(verbose_name=b'Modified on')),
+                ('previous_value', models.CharField(blank=True, max_length=2048, null=True, verbose_name=b'Previous Value')),
+                ('new_value', models.CharField(blank=True, max_length=2048, null=True, verbose_name=b'New Value')),
+                ('modification_description', models.CharField(max_length=2048, verbose_name=b'Description')),
+                ('modification_author', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='web.Worker', verbose_name=b'Worker who modified the row')),
+            ],
+        ),
+    ]
diff --git a/smash/web/migrations/0143_auto_20200319_1121.py b/smash/web/migrations/0143_auto_20200319_1121.py
new file mode 100644
index 0000000000000000000000000000000000000000..57b5f92d46ca0a2346d91164f552407fa46a4ed5
--- /dev/null
+++ b/smash/web/migrations/0143_auto_20200319_1121.py
@@ -0,0 +1,20 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.11.5 on 2020-03-19 11:21
+from __future__ import unicode_literals
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('web', '0142_provenance'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='provenance',
+            name='modification_date',
+            field=models.DateTimeField(auto_now_add=True, verbose_name=b'Modified on'),
+        ),
+    ]
diff --git a/smash/web/migrations/0144_auto_20200319_1221.py b/smash/web/migrations/0144_auto_20200319_1221.py
new file mode 100644
index 0000000000000000000000000000000000000000..a232453bb6bb96bd20d1f7dbe5510454a9fa85cf
--- /dev/null
+++ b/smash/web/migrations/0144_auto_20200319_1221.py
@@ -0,0 +1,26 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.11.5 on 2020-03-19 12:21
+from __future__ import unicode_literals
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('web', '0143_auto_20200319_1121'),
+    ]
+
+    operations = [
+        migrations.AddField(
+            model_name='provenance',
+            name='modified_field',
+            field=models.CharField(default='', max_length=1024, verbose_name=b'Modified field'),
+            preserve_default=False,
+        ),
+        migrations.AlterField(
+            model_name='provenance',
+            name='modification_description',
+            field=models.CharField(max_length=20480, verbose_name=b'Description'),
+        ),
+    ]
diff --git a/smash/web/migrations/0145_auto_20200319_1404.py b/smash/web/migrations/0145_auto_20200319_1404.py
new file mode 100644
index 0000000000000000000000000000000000000000..fda7e5189461e4375b4b95e177a553b9f0955bd0
--- /dev/null
+++ b/smash/web/migrations/0145_auto_20200319_1404.py
@@ -0,0 +1,20 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.11.5 on 2020-03-19 14:04
+from __future__ import unicode_literals
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('web', '0144_auto_20200319_1221'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='provenance',
+            name='modified_field',
+            field=models.CharField(blank=b'', max_length=1024, verbose_name=b'Modified field'),
+        ),
+    ]
diff --git a/smash/web/migrations/0146_auto_20200319_1446.py b/smash/web/migrations/0146_auto_20200319_1446.py
new file mode 100644
index 0000000000000000000000000000000000000000..8cd04b4486246d6dd67b724fe62121db273d4c41
--- /dev/null
+++ b/smash/web/migrations/0146_auto_20200319_1446.py
@@ -0,0 +1,21 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.11.5 on 2020-03-19 14:46
+from __future__ import unicode_literals
+
+from django.db import migrations, models
+import django.db.models.deletion
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('web', '0145_auto_20200319_1404'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='provenance',
+            name='modification_author',
+            field=models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to='web.Worker', verbose_name=b'Worker who modified the row'),
+        ),
+    ]
diff --git a/smash/web/migrations/0147_auto_20200320_0931.py b/smash/web/migrations/0147_auto_20200320_0931.py
new file mode 100644
index 0000000000000000000000000000000000000000..58a3dc901515fa768382f239352409bb6a7a9ed8
--- /dev/null
+++ b/smash/web/migrations/0147_auto_20200320_0931.py
@@ -0,0 +1,20 @@
+# -*- coding: utf-8 -*-
+# Generated by Django 1.11.5 on 2020-03-20 09:31
+from __future__ import unicode_literals
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('web', '0146_auto_20200319_1446'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='provenance',
+            name='modified_table_id',
+            field=models.IntegerField(default=0, verbose_name=b'Modified table row'),
+        ),
+    ]
diff --git a/smash/web/migrations/0141_auto_20200319_1301.py b/smash/web/migrations/0148_auto_20200319_1301.py
similarity index 93%
rename from smash/web/migrations/0141_auto_20200319_1301.py
rename to smash/web/migrations/0148_auto_20200319_1301.py
index 82d45afe3c2cb59017c44e55a03078f898c82b0c..1b2abecbe955819c77d988944336b22ad28bd016 100644
--- a/smash/web/migrations/0141_auto_20200319_1301.py
+++ b/smash/web/migrations/0148_auto_20200319_1301.py
@@ -7,7 +7,7 @@ from django.db import migrations, models
 
 class Migration(migrations.Migration):
     dependencies = [
-        ('web', '0140_auto_20190528_0953'),
+        ('web', '0147_auto_20200320_0931'),
     ]
 
     operations = [
diff --git a/smash/web/migrations/0142_auto_20200319_1415.py b/smash/web/migrations/0149_auto_20200319_1415.py
similarity index 90%
rename from smash/web/migrations/0142_auto_20200319_1415.py
rename to smash/web/migrations/0149_auto_20200319_1415.py
index ea5d462085fccf023fc04503eefdf26fda15e101..c948e9780d3d5da309c62035ff5afb27f0d835fc 100644
--- a/smash/web/migrations/0142_auto_20200319_1415.py
+++ b/smash/web/migrations/0149_auto_20200319_1415.py
@@ -8,7 +8,7 @@ from django.db import migrations
 class Migration(migrations.Migration):
 
     dependencies = [
-        ('web', '0141_auto_20200319_1301'),
+        ('web', '0148_auto_20200319_1301'),
     ]
 
     operations = [
diff --git a/smash/web/migrations/0143_auto_20200319_1446.py b/smash/web/migrations/0150_auto_20200319_1446.py
similarity index 91%
rename from smash/web/migrations/0143_auto_20200319_1446.py
rename to smash/web/migrations/0150_auto_20200319_1446.py
index d6c74b0cb78b6fc342a2140b38ac4bec2bd968cb..0fcdea7e40cf414c94558b82d7befe42e7dbc275 100644
--- a/smash/web/migrations/0143_auto_20200319_1446.py
+++ b/smash/web/migrations/0150_auto_20200319_1446.py
@@ -8,7 +8,7 @@ from django.db import migrations
 class Migration(migrations.Migration):
 
     dependencies = [
-        ('web', '0142_auto_20200319_1415'),
+        ('web', '0149_auto_20200319_1415'),
     ]
 
     operations = [
diff --git a/smash/web/migrations/0144_auto_20200319_1518.py b/smash/web/migrations/0151_auto_20200319_1518.py
similarity index 91%
rename from smash/web/migrations/0144_auto_20200319_1518.py
rename to smash/web/migrations/0151_auto_20200319_1518.py
index bdb5b3eebb7a1b9a2aed55790bd8a9fb7dae6dc1..a034d7b50283e5a6011cbcf3e9bf24a465c0cfcb 100644
--- a/smash/web/migrations/0144_auto_20200319_1518.py
+++ b/smash/web/migrations/0151_auto_20200319_1518.py
@@ -8,7 +8,7 @@ from django.db import migrations
 class Migration(migrations.Migration):
 
     dependencies = [
-        ('web', '0143_auto_20200319_1446'),
+        ('web', '0150_auto_20200319_1446'),
     ]
 
     operations = [
diff --git a/smash/web/migrations/0145_add_permissions_to_existing_workers.py b/smash/web/migrations/0152_add_permissions_to_existing_workers.py
similarity index 96%
rename from smash/web/migrations/0145_add_permissions_to_existing_workers.py
rename to smash/web/migrations/0152_add_permissions_to_existing_workers.py
index 877a37547c0ff7b9d87e409235effc33c5d7d8f7..67752e4690b59643da8f7ae2fe375db39a43f2a2 100644
--- a/smash/web/migrations/0145_add_permissions_to_existing_workers.py
+++ b/smash/web/migrations/0152_add_permissions_to_existing_workers.py
@@ -7,7 +7,7 @@ from django.db import migrations
 
 class Migration(migrations.Migration):
     dependencies = [
-        ('web', '0144_auto_20200319_1518'),
+        ('web', '0151_auto_20200319_1518'),
     ]
 
     operations = [
diff --git a/smash/web/migrations/0146_auto_20200320_0932.py b/smash/web/migrations/0153_auto_20200320_0932.py
similarity index 97%
rename from smash/web/migrations/0146_auto_20200320_0932.py
rename to smash/web/migrations/0153_auto_20200320_0932.py
index 93b2487ba4a8bcf2f002110c262a83cc370d33b8..06a7de106240cf856fceb9720477be0b9aa87526 100644
--- a/smash/web/migrations/0146_auto_20200320_0932.py
+++ b/smash/web/migrations/0153_auto_20200320_0932.py
@@ -8,7 +8,7 @@ from django.db import migrations, models
 class Migration(migrations.Migration):
 
     dependencies = [
-        ('web', '0145_add_permissions_to_existing_workers'),
+        ('web', '0152_add_permissions_to_existing_workers'),
     ]
 
     operations = [
diff --git a/smash/web/models/__init__.py b/smash/web/models/__init__.py
index 3dc346ec8d6b719b90273225b1626aa765571ac8..ee5fdb79d7681d2cd9a10e0fdde4d862e136ed39 100644
--- a/smash/web/models/__init__.py
+++ b/smash/web/models/__init__.py
@@ -3,6 +3,7 @@ from __future__ import unicode_literals
 
 from django.contrib.auth.models import User
 
+from provenance import Provenance
 from configuration_item import ConfigurationItem
 from flying_team import FlyingTeam
 from location import Location
diff --git a/smash/web/models/provenance.py b/smash/web/models/provenance.py
new file mode 100644
index 0000000000000000000000000000000000000000..0d9aeb7a7e3f2020d36c61254dd681dd02322522
--- /dev/null
+++ b/smash/web/models/provenance.py
@@ -0,0 +1,44 @@
+# coding=utf-8
+from django.db import models
+
+
+class Provenance(models.Model):
+    class Meta:
+        app_label = 'web'
+
+
+    modified_table = models.CharField(max_length=1024,
+                                     verbose_name='Modified table',
+                                     blank=False, null=False
+                                    )
+
+    modified_table_id = models.IntegerField(default=0, verbose_name='Modified table row', blank=False, null=False)    
+
+    modification_date = models.DateTimeField(
+        verbose_name='Modified on',
+        null=False, blank=False,
+        auto_now_add=True
+    )
+
+    modification_author = models.ForeignKey("web.Worker",
+                            verbose_name='Worker who modified the row',
+                            null=True, blank=False
+                        )
+
+    modified_field = models.CharField(max_length=1024,
+                                     verbose_name='Modified field',
+                                     blank='', null=False
+                                    )
+
+    previous_value = models.CharField(max_length=2048,
+                                      verbose_name='Previous Value',
+                                      blank=True, null=True)
+
+    new_value = models.CharField(max_length=2048,
+                                      verbose_name='New Value',
+                                      blank=True, null=True)    
+
+    modification_description = models.CharField(max_length=20480,
+                                     verbose_name='Description',
+                                     blank=False, null=False
+                                    )
\ No newline at end of file
diff --git a/smash/web/models/subject.py b/smash/web/models/subject.py
index fa9f067db93781e5e4fb90a70a51f5eee41a339e..bc70691410f4d7459a0ea7b08336abd8a506d7a0 100644
--- a/smash/web/models/subject.py
+++ b/smash/web/models/subject.py
@@ -6,7 +6,7 @@ from django.db.models.signals import post_save
 from django.dispatch import receiver
 
 from constants import SEX_CHOICES, COUNTRY_OTHER_ID
-from web.models import Country, Visit, Appointment
+from web.models import Country, Visit, Appointment, Provenance
 from . import Language
 
 logger = logging.getLogger(__name__)
@@ -154,4 +154,11 @@ class Subject(models.Model):
 @receiver(post_save, sender=Subject)
 def set_as_deceased(sender, instance, **kwargs):
     if instance.dead:
+        p = Provenance(modified_table = Subject._meta.db_table,
+                    modified_table_id = instance.id, modification_author = None,
+                    previous_value = instance.dead, new_value = True,
+                    modification_description = 'Subject "{}" marked as dead'.format(instance),
+                    modified_field = 'dead',
+                    )
         instance.mark_as_dead()
+        p.save()
\ No newline at end of file
diff --git a/smash/web/templates/provenance/breadcrumb.html b/smash/web/templates/provenance/breadcrumb.html
new file mode 100644
index 0000000000000000000000000000000000000000..1d4019a699c2cb1dee0b005b58ed1e2a4da53099
--- /dev/null
+++ b/smash/web/templates/provenance/breadcrumb.html
@@ -0,0 +1,2 @@
+<li><a href="{% url 'web.views.provenance' %}"><i class="fa fa-dashboard"></i> Dashboard</a></li>
+<li class="active"><a href="{% url 'web.views.provenance' %}">Provenance</a></li>
\ No newline at end of file
diff --git a/smash/web/templates/provenance/list.html b/smash/web/templates/provenance/list.html
new file mode 100644
index 0000000000000000000000000000000000000000..42840c32a6b7b5be70bf0ea9cb2149e784efccda
--- /dev/null
+++ b/smash/web/templates/provenance/list.html
@@ -0,0 +1,70 @@
+{% extends "_base.html" %}
+{% load static %}
+
+{% block styles %}
+    {{ block.super }}
+    <!-- DataTables -->
+    <link rel="stylesheet" href="{% static 'AdminLTE/plugins/datatables/dataTables.bootstrap.css' %}">
+{% endblock styles %}
+
+{% block ui_active_tab %}'provenance'{% endblock ui_active_tab %}
+{% block page_header %}Provenance{% endblock page_header %}
+{% block page_description %}{% endblock page_description %}
+
+{% block breadcrumb %}
+    {% include "languages/breadcrumb.html" %}
+{% endblock breadcrumb %}
+
+{% block maincontent %}
+
+    <div class="box-body">
+        <table id="table" class="table table-bordered table-striped">
+            <thead>
+            <tr>
+                <th>Table</th>
+                <th>Row</th>
+                <th>Date</th>
+                <th>Author</th>
+                <th>Field</th>
+                <th>Previous Value</th>
+                <th>New Value</th>
+                <th>Description</th>
+            </tr>
+            </thead>
+            <tbody>
+            {% for provenance in provenances %}
+                <tr>
+                    <td>{{ provenance.modified_table }}</td>
+                    <td>{{ provenance.modified_table_id }}</td>
+                    <td>{{ provenance.modification_date }}</td>
+                    <td>{{ provenance.modification_author }}</td>
+                    <td>{{ provenance.modified_field }}</td>
+                    <td>{{ provenance.previous_value }}</td>
+                    <td>{{ provenance.new_value }}</td>
+                    <td>{{ provenance.modification_description }}</td>
+                </tr>
+            {% endfor %}
+            </tbody>
+        </table>
+    </div>
+{% endblock maincontent %}
+
+{% block scripts %}
+    {{ block.super }}
+
+    <script src="{% static 'AdminLTE/plugins/datatables/jquery.dataTables.min.js' %}"></script>
+    <script src="{% static 'AdminLTE/plugins/datatables/dataTables.bootstrap.min.js' %}"></script>
+
+    <script>
+        $(function () {
+            $('#table').DataTable({
+                "paging": true,
+                "lengthChange": false,
+                "searching": true,
+                "ordering": true,
+                "info": true,
+                "autoWidth": false
+            });
+        });
+    </script>
+{% endblock scripts %}
diff --git a/smash/web/urls.py b/smash/web/urls.py
index bbbdfbcde4fee7032359ec9a7030ab98e83e96ff..e8b61fa0aaf83e4b04cf9c3c1dc6dc908b829b10 100644
--- a/smash/web/urls.py
+++ b/smash/web/urls.py
@@ -233,6 +233,8 @@ urlpatterns = [
 
     url(r'^statistics$', views.statistics.statistics, name='web.views.statistics'),
 
+    url(r'^provenance$', views.provenance.ProvenanceListView.as_view(), name='web.views.provenance'),
+
     ####################
     #    STUDY         #
     ####################
diff --git a/smash/web/views/__init__.py b/smash/web/views/__init__.py
index f662f7e759ca56715f910877c68e7d51a4e4197c..66c3b6339e4afecdf3332e273a6f03d4e1c43705 100644
--- a/smash/web/views/__init__.py
+++ b/smash/web/views/__init__.py
@@ -104,4 +104,5 @@ import rooms
 import uploaded_files
 import study
 import password
-import appointment_type
\ No newline at end of file
+import appointment_type
+import provenance
\ No newline at end of file
diff --git a/smash/web/views/appointment.py b/smash/web/views/appointment.py
index d2d1f0ff92fa75353f7e8a08845270c45c67908e..84b12b2493f97f22c0e1073588c785815f2b515d 100644
--- a/smash/web/views/appointment.py
+++ b/smash/web/views/appointment.py
@@ -12,7 +12,7 @@ from web.models.appointment_list import APPOINTMENT_LIST_APPROACHING, APPOINTMEN
 from . import wrap_response
 from web.forms import AppointmentDetailForm, AppointmentEditForm, AppointmentAddForm, SubjectEditForm, \
     StudySubjectEditForm
-from ..models import Appointment, StudySubject, MailTemplate, Visit, Study
+from ..models import Appointment, StudySubject, MailTemplate, Visit, Study, Provenance, Worker
 from django.views.generic import DeleteView
 from . import WrappedView
 from django.urls import reverse_lazy
@@ -184,6 +184,16 @@ class AppointmentDeleteView(DeleteView, WrappedView):
             return redirect('web.views.appointment_edit', id=appointment.id)
         else:
             messages.success(request, "Appointment deleted")
+            worker = Worker.get_by_user(request.user)
+            p = Provenance(modified_table = Appointment._meta.db_table,
+                            modified_table_id = appointment.id,
+                            modification_author = worker,
+                            previous_value = '',
+                            new_value = '',
+                            modification_description = 'Appointment deleted',
+                            modified_field = '',
+                            )
+            p.save()
             return super(AppointmentDeleteView, self).delete(request, *args, **kwargs)
 
     @PermissionDecorator('delete_appointment', 'configuration')
diff --git a/smash/web/views/provenance.py b/smash/web/views/provenance.py
new file mode 100644
index 0000000000000000000000000000000000000000..dc2e85293892712dac29c5f00f679f394d388b9b
--- /dev/null
+++ b/smash/web/views/provenance.py
@@ -0,0 +1,15 @@
+# coding=utf-8
+from django.views.generic import ListView
+
+from . import WrappedView
+from ..models import Provenance
+from web.decorators import PermissionDecorator
+
+class ProvenanceListView(ListView, WrappedView):
+    model = Provenance
+    context_object_name = "provenances"
+    template_name = 'provenance/list.html'
+
+    @PermissionDecorator('view_provenance', 'configuration')
+    def dispatch(self, *args, **kwargs):
+        return super(ProvenanceListView, self).dispatch(*args, **kwargs)
\ No newline at end of file
diff --git a/smash/web/views/subject.py b/smash/web/views/subject.py
index a11a77f73cb0eea85db236cf96c4117377c629c3..cb8cf69d285f57f1496dd66e092e08bf9365cc1d 100644
--- a/smash/web/views/subject.py
+++ b/smash/web/views/subject.py
@@ -6,8 +6,8 @@ from django.shortcuts import redirect, get_object_or_404
 
 from . import wrap_response
 from ..forms import VisitDetailForm, SubjectAddForm, SubjectEditForm, StudySubjectAddForm, StudySubjectEditForm
-from ..models import StudySubject, MailTemplate, Worker, Study
-from ..models.constants import GLOBAL_STUDY_ID
+from ..models import StudySubject, MailTemplate, Worker, Study, Provenance, Subject
+from ..models.constants import GLOBAL_STUDY_ID, SUBJECT_TYPE_CHOICES
 from ..models.study_subject_list import SUBJECT_LIST_GENERIC, SUBJECT_LIST_NO_VISIT, SUBJECT_LIST_REQUIRE_CONTACT, \
     SUBJECT_LIST_VOUCHER_EXPIRY, SUBJECT_LIST_CHOICES
 
@@ -68,6 +68,7 @@ def subject_edit(request, id):
     contact_attempts = study_subject.contactattempt_set.order_by('-datetime_when').all()
     was_dead = study_subject.subject.dead
     was_resigned = study_subject.resigned
+    old_type = study_subject.type
     endpoint_was_reached = study_subject.endpoint_reached
     if request.method == 'POST':
         study_subject_form = StudySubjectEditForm(request.POST, request.FILES, instance=study_subject,
@@ -80,8 +81,44 @@ def subject_edit(request, id):
             study_subject_form.save()
             subject_form.save()
             # check if subject was marked as dead or resigned
+            if 'type' in study_subject_form.changed_data and old_type != study_subject_form.cleaned_data['type']:
+                worker = Worker.get_by_user(request.user)
+                old_value = SUBJECT_TYPE_CHOICES.get(old_type, old_type)
+                new_value = SUBJECT_TYPE_CHOICES.get(study_subject_form.cleaned_data['type'], study_subject_form.cleaned_data['type'])
+                p = Provenance(modified_table = StudySubject._meta.db_table,
+                                modified_table_id = study_subject.id,
+                                modification_author = worker,
+                                previous_value = old_type,
+                                new_value = study_subject_form.cleaned_data['type'],
+                                modification_description = 'Worker "{}" changed study subject "{}" from "{}" to "{}"'.format(worker, 
+                                                study_subject.subject, old_value, new_value),
+                                modified_field = 'type',
+                                )
+                p.save()
             if subject_form.cleaned_data['dead'] and not was_dead:
+                worker = Worker.get_by_user(request.user)
+                p = Provenance(modified_table = Subject._meta.db_table,
+                                modified_table_id = study_subject.subject.id,
+                                modification_author = worker,
+                                previous_value = was_dead,
+                                new_value = True,
+                                modification_description = 'Worker "{}" marks subject "{}" as dead'.format(worker, study_subject.subject),
+                                modified_field = 'dead',
+                                )
                 study_subject.subject.mark_as_dead()
+                p.save()
+            if study_subject_form.cleaned_data['resigned'] and not was_resigned:
+                worker = Worker.get_by_user(request.user)
+                p = Provenance(modified_table = StudySubject._meta.db_table,
+                                modified_table_id = study_subject.id,
+                                modification_author = worker,
+                                previous_value = was_resigned,
+                                new_value = True,
+                                modification_description = 'Worker "{}" marks study subject "{}" as resigned from study "{}"'.format(worker, study_subject.nd_number, study_subject.study),
+                                modified_field = 'resigned',
+                                )
+                study_subject.mark_as_resigned()
+                p.save()
             messages.success(request, "Modifications saved")
             if '_continue' in request.POST:
                 return redirect('web.views.subject_edit', id=study_subject.id)
diff --git a/smash/web/views/visit.py b/smash/web/views/visit.py
index 10f2f80cf96e3fb6237ce303131869ba124aaa95..05d0ac44e9022231b52552752d32d64f31917ecd 100644
--- a/smash/web/views/visit.py
+++ b/smash/web/views/visit.py
@@ -9,7 +9,7 @@ from web.models.study_visit_list import VISIT_LIST_GENERIC, VISIT_LIST_MISSING_A
     VISIT_LIST_UNFINISHED, VISIT_LIST_CHOICES
 from . import wrap_response
 from ..forms import VisitDetailForm, VisitAddForm, SubjectDetailForm, StudySubjectDetailForm
-from ..models import Visit, Appointment, StudySubject, MailTemplate
+from ..models import Visit, Appointment, StudySubject, MailTemplate, Worker, Provenance
 
 logger = logging.getLogger(__name__)
 
@@ -93,7 +93,17 @@ def visit_details(request, id):
 def visit_mark(request, id, as_what):
     visit = get_object_or_404(Visit, id=id)
     if as_what == 'finished':
+        worker = Worker.get_by_user(request.user)
+        p = Provenance(modified_table = Visit._meta.db_table,
+                    modified_table_id = id,
+                    modification_author = worker,
+                    previous_value = visit.is_finished,
+                    new_value = True,
+                    modification_description = 'Worker "{}" marked visit from "{}" as finished'.format(worker, visit.subject),
+                    modified_field = 'is_finished',
+            )
         visit.mark_as_finished()
+        p.save()
 
     return redirect('web.views.visit_details', id=id)