diff --git a/CHANGELOG b/CHANGELOG
index dbe23372c4f65f254a3bf043cb4e1328d2818257..5de78527ff78208648d5195df7235c19f91ad0ba 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -5,6 +5,8 @@ smasch (1.0.0~alpha.1-0) unstable; urgency=low
   * small improvement: possibility to unfinish visit (#351)
   * small improvement: all requests to export data to csv/xls are tracked
     (#285)
+  * small improvement: all configuration options that are not obligatory are 
+    moved to configuration panel (#343)
   
  -- Piotr Gawron <piotr.gawron@uni.lu>  Tue, 10 Nov 2020 14:00:00 +0200
 
diff --git a/debian-files/smasch.py b/debian-files/smasch.py
index 5a860c1a694174471b8516fc7b2177c903ade16b..357f4639499f4143ff9a72365c523bb38a4159ec 100644
--- a/debian-files/smasch.py
+++ b/debian-files/smasch.py
@@ -7,8 +7,6 @@ DEBUG = False
 # Should the static/media files be served by Django
 SERVE_STATIC = True
 
-WSGI_APPLICATION = 'smash.wsgi.application'
-
 DATABASES = {
     'default': {
         'ENGINE': 'django.db.backends.sqlite3',
@@ -19,15 +17,8 @@ DATABASES = {
 STATIC_ROOT = '/usr/lib/smasch/data/static'
 MEDIA_ROOT = '/usr/lib/smasch/data/media'
 
-LOGIN_PAGE_BACKGROUND_IMAGE = 'background.jpg'  # Path to a static file containing background image, used in login.html
-
 ALLOWED_HOSTS = ["127.0.0.1", "localhost"]
 
-STATICFILES_STORAGE = 'django.contrib.staticfiles.storage.StaticFilesStorage'
-NEXMO_API_KEY = 'API_KEY'
-NEXMO_API_SECRET = 'API_SECRET'
-NEXMO_DEFAULT_FROM = 'Scheduling'  # the sender of the message (phone number or text)
-
 LOGGING = {
     'version': 1,
     'disable_existing_loggers': False,
@@ -64,3 +55,4 @@ LOGGING = {
     },
 }
 
+TWO_FACTOR_SMS_GATEWAY = "web.nexmo_gateway.Nexmo"
diff --git a/local_settings_ci.py b/local_settings_ci.py
index 498971bd66253cec0e393dde57c6ba2a0828734e..0d3635ecf21577f55f30d9c1512d70c22603bd42 100644
--- a/local_settings_ci.py
+++ b/local_settings_ci.py
@@ -4,8 +4,6 @@ SECRET_KEY = 'Paste long random string here'  # Insert long random string
 # SECURITY WARNING: don't run with debug turned on in production!
 DEBUG = True
 
-WSGI_APPLICATION = 'smash.wsgi.application'
-
 # Database
 # https://docs.djangoproject.com/en/1.10/ref/settings/#databases
 
@@ -29,12 +27,8 @@ DATABASES = {
 
 STATIC_ROOT = '/tmp/static'  # Warning! `/tmp` directory can be flushed in any moment; use a persistent one; e.g. ~/tmp/static
 MEDIA_ROOT = '/tmp/media'  # Warning! `/tmp` directory can be flushed in any moment; use a persistent one, e.g. ~/tmp/media
+UPLOAD_ROOT = '/tmp/upload'
 
-LOGIN_PAGE_BACKGROUND_IMAGE = 'background.jpg'  # Path to a static file containing background image, used in login.html
+STATICFILES_STORAGE = 'django.contrib.staticfiles.storage.StaticFilesStorage'
 
 ALLOWED_HOSTS = ["127.0.0.1", "localhost"]
-
-STATICFILES_STORAGE = 'django.contrib.staticfiles.storage.StaticFilesStorage'
-NEXMO_API_KEY = 'API_KEY'
-NEXMO_API_SECRET = 'API_SECRET'
-NEXMO_DEFAULT_FROM = 'Scheduling'  # the sender of the message (phone number or text)
diff --git a/local_settings_ci_sqlite.py b/local_settings_ci_sqlite.py
index 2e98aec13d400b12228ab10b4a8c6079527ee217..b56d164043bdf58d8f3194a012f68f960eaf2ed3 100644
--- a/local_settings_ci_sqlite.py
+++ b/local_settings_ci_sqlite.py
@@ -4,8 +4,6 @@ SECRET_KEY = 'Paste long random string here'  # Insert long random string
 # SECURITY WARNING: don't run with debug turned on in production!
 DEBUG = True
 
-WSGI_APPLICATION = 'smash.wsgi.application'
-
 # Database
 # https://docs.djangoproject.com/en/1.10/ref/settings/#databases
 
@@ -18,12 +16,8 @@ DATABASES = {
 
 STATIC_ROOT = '/tmp/static'  # Warning! `/tmp` directory can be flushed in any moment; use a persistent one; e.g. ~/tmp/static
 MEDIA_ROOT = '/tmp/media'  # Warning! `/tmp` directory can be flushed in any moment; use a persistent one, e.g. ~/tmp/media
+UPLOAD_ROOT = '/tmp/upload'
 
-LOGIN_PAGE_BACKGROUND_IMAGE = 'background.jpg'  # Path to a static file containing background image, used in login.html
+STATICFILES_STORAGE = 'django.contrib.staticfiles.storage.StaticFilesStorage'
 
 ALLOWED_HOSTS = ["127.0.0.1", "localhost"]
-
-STATICFILES_STORAGE = 'django.contrib.staticfiles.storage.StaticFilesStorage'
-NEXMO_API_KEY = 'API_KEY'
-NEXMO_API_SECRET = 'API_SECRET'
-NEXMO_DEFAULT_FROM = 'Scheduling'  # the sender of the message (phone number or text)
diff --git a/smash/db_scripts/import_file.py b/smash/db_scripts/import_file.py
index 71c8486f8c19e26e4a89516bb217d2366c29bb2c..d8d8d18406cdf77f3120eef893e44fb2cf266907 100644
--- a/smash/db_scripts/import_file.py
+++ b/smash/db_scripts/import_file.py
@@ -1,9 +1,10 @@
 # coding=utf-8
-import os, sys
+import os
+import sys
+
 sys.path.append(sys.path.append(os.path.join(os.path.dirname(__file__), '..'))) #run script as it was on parent folder
 import getpass
 import django
-from django.conf import settings
 os.environ.setdefault("DJANGO_SETTINGS_MODULE", "smash.settings")
 django.setup()
 from smash.local_settings import MEDIA_ROOT
@@ -15,17 +16,16 @@ import datetime
 from dateutil.relativedelta import relativedelta
 import re
 from operator import itemgetter
-from collections import OrderedDict, defaultdict
-import string
+from collections import OrderedDict
 from django.contrib.auth.models import User
-from web.models.constants import VOUCHER_STATUS_IN_USE, SUBJECT_TYPE_CHOICES_PATIENT, GLOBAL_STUDY_ID, SEX_CHOICES, SEX_CHOICES_MALE, SEX_CHOICES_FEMALE
+from web.models.constants import VOUCHER_STATUS_IN_USE, SUBJECT_TYPE_CHOICES_PATIENT, GLOBAL_STUDY_ID, SEX_CHOICES_MALE, SEX_CHOICES_FEMALE
 from web.algorithm import VerhoeffAlgorithm, LuhnAlgorithm
 from web.utils import is_valid_social_security_number
 
-from web.models import VoucherType, Voucher, Country, AppointmentTypeLink, AppointmentType, Study, Worker, Language, Subject, WorkerStudyRole, StudySubject, Location, FlyingTeam, Visit, Appointment, AppointmentType
+from web.models import VoucherType, Voucher, Country, AppointmentTypeLink, Study, Worker, Language, Subject, WorkerStudyRole, StudySubject, Location, FlyingTeam, Visit, Appointment, AppointmentType
 
-from web.models.worker_study_role import ROLE_CHOICES_TECHNICIAN, WORKER_STAFF, ROLE_CHOICES_SECRETARY, ROLE_CHOICES_HEALTH_PARTNER, \
-    WORKER_HEALTH_PARTNER, ROLE_CHOICES_DOCTOR, ROLE_CHOICES_VOUCHER_PARTNER, ROLE_CHOICES, ROLE_CHOICES_NURSE, ROLE_CHOICES_PSYCHOLOGIST, ROLE_CHOICES_PROJECT_MANAGER
+from web.models.worker_study_role import ROLE_CHOICES_TECHNICIAN, ROLE_CHOICES_SECRETARY, ROLE_CHOICES_HEALTH_PARTNER, \
+    ROLE_CHOICES_DOCTOR, ROLE_CHOICES_VOUCHER_PARTNER, ROLE_CHOICES_NURSE, ROLE_CHOICES_PSYCHOLOGIST, ROLE_CHOICES_PROJECT_MANAGER
 
 DEFAULT_LOCATION = 'CHL'
 DEFAULT_LOCATION_PREFIX = 'P'
diff --git a/smash/smash/context_processors.py b/smash/smash/context_processors.py
index fec3d587a5d3859a9775aa33d56221a21fc66f43..f60066ac7aa617647182e6569a8833219eec31be 100644
--- a/smash/smash/context_processors.py
+++ b/smash/smash/context_processors.py
@@ -1,12 +1,12 @@
-from django.conf import settings
+from web.models import ConfigurationItem
+from web.models.constants import LOGIN_PAGE_BACKGROUND_IMAGE
 
 
-BACKGROUND_IMAGE = getattr(settings, "LOGIN_PAGE_BACKGROUND_IMAGE", "background.jpg")
-
+# noinspection PyUnusedLocal
 def login_background(request):
     """
     Context processor - these values will be available to templates once registered in settings.py
     """
     return {
-        'login_page_background': BACKGROUND_IMAGE
+        'login_page_background': ConfigurationItem.objects.get(type=LOGIN_PAGE_BACKGROUND_IMAGE).value
     }
diff --git a/smash/smash/local_settings.py.template b/smash/smash/local_settings.py.template
index dbbf01a34f1d159b043df629b989615a57674971..79c90e5220130d1e6ea9467b312d3edf30beee9b 100644
--- a/smash/smash/local_settings.py.template
+++ b/smash/smash/local_settings.py.template
@@ -4,10 +4,6 @@ SECRET_KEY = 'Paste long random string here'  # Insert long random string
 # SECURITY WARNING: don't run with debug turned on in production!
 DEBUG = True
 
-WSGI_APPLICATION = 'smash.wsgi.application'
-
-IMPORTER_USER = 'admin' #username
-
 EMAIL_USE_TLS = False
 EMAIL_USE_SSL = False
 EMAIL_HOST = 'smtp.uni.lu'
@@ -15,12 +11,8 @@ EMAIL_HOST_USER = ''
 EMAIL_HOST_PASSWORD = ''
 EMAIL_PORT = 25
 EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend'
-DEFAULT_FROM_EMAIL = 'prc-scheduling-admin@uni.lu'
 
-ALLOWED_HOSTS = ['prc.parkinson.lu', 'localhost']
-
-# Database
-# https://docs.djangoproject.com/en/1.10/ref/settings/#databases
+ALLOWED_HOSTS = ['your.domain.com', 'localhost']
 
 DATABASES = {
     'default': {
@@ -37,16 +29,15 @@ DATABASES = {
     }
 }
 
-STATIC_ROOT = '/tmp/static'  # Warning! `/tmp` directory can be flushed in any moment; use a persistent one; e.g. ~/tmp/static
-MEDIA_ROOT = '/tmp/media'  # Warning! `/tmp` directory can be flushed in any moment; use a persistent one, e.g. ~/tmp/media
-
-LOGIN_PAGE_BACKGROUND_IMAGE = 'background.jpg'  # Path to a static file containing background image, used in login.html
-
-STATICFILES_STORAGE = 'django.contrib.staticfiles.storage.StaticFilesStorage'
-
-NEXMO_API_KEY = 'API_KEY'
-NEXMO_API_SECRET = 'API_SECRET'
-NEXMO_DEFAULT_FROM = 'Scheduling'  # the sender of the message (phone number or text)
+# Warning! `/tmp` directory can be flushed in any moment;
+# use a persistent one; e.g. ~/tmp/static
+STATIC_ROOT = '~/tmp/static'
+# Warning! `/tmp` directory can be flushed in any moment;
+# use a persistent one, e.g. ~/tmp/media
+MEDIA_ROOT = '~/tmp/media'
+# Warning! `/tmp` directory can be flushed in any moment;
+# use a persistent one, e.g. ~/tmp/upload
+UPLOAD_ROOT = '~/tmp/upload'
 
 LOGGING = {
     'version': 1,
@@ -83,3 +74,5 @@ LOGGING = {
         },
     },
 }
+
+TWO_FACTOR_SMS_GATEWAY = "web.nexmo_gateway.Nexmo"
diff --git a/smash/smash/settings.py b/smash/smash/settings.py
index edaaa38ada3e74eb94a52f64e755b527497e5f30..15be4ea9c4cd68270d4226892658989016d80985 100644
--- a/smash/smash/settings.py
+++ b/smash/smash/settings.py
@@ -20,6 +20,8 @@ PROJECT_PATH = os.path.abspath(os.path.dirname(__file__))
 
 DEBUG = True
 
+WSGI_APPLICATION = 'smash.wsgi.application'
+
 SERVE_STATIC = False
 
 # Application definition
diff --git a/smash/web/importer/csv_tns_visit_import_reader.py b/smash/web/importer/csv_tns_visit_import_reader.py
index 54a111e48701e84413b09810e05db5a7f22e94ae..2669ad51342b94fbffc5e311248b2f61a4b00314 100644
--- a/smash/web/importer/csv_tns_visit_import_reader.py
+++ b/smash/web/importer/csv_tns_visit_import_reader.py
@@ -7,11 +7,11 @@ import sys
 import traceback
 
 import pytz
-from django.conf import settings
 
+from web.models import StudySubject, Study, Visit, Appointment, AppointmentType, Location, AppointmentTypeLink, Subject, \
+    User, Worker, Provenance, ConfigurationItem
+from web.models.constants import GLOBAL_STUDY_ID, IMPORTER_USER, IMPORT_APPOINTMENT_TYPE
 from .warning_counter import MsgCounterHandler
-from web.models import StudySubject, Study, Visit, Appointment, AppointmentType, Location, AppointmentTypeLink, Subject, User, Worker, Provenance
-from web.models.constants import GLOBAL_STUDY_ID
 
 CSV_DATE_FORMAT = "%d/%m/%Y"
 
@@ -21,13 +21,13 @@ logger = logging.getLogger(__name__)
 class TnsCsvVisitImportReader:
     def __init__(self):
         self.study = Study.objects.filter(id=GLOBAL_STUDY_ID)[0]
-        appointment_code = getattr(settings, "IMPORT_APPOINTMENT_TYPE", "SAMPLES")
+        appointment_code = ConfigurationItem.objects.get(type=IMPORT_APPOINTMENT_TYPE).value
 
         appointment_types = AppointmentType.objects.filter(code=appointment_code)
         if len(appointment_types) > 0:
             self.appointment_type = appointment_types[0]
         else:
-            logger.warn("Appointment type does not exist: " + appointment_code)
+            logger.warning("Appointment type does not exist: " + appointment_code)
             self.appointment_type = None
         self.problematic_count = 0
         self.processed_count = 0
@@ -35,11 +35,11 @@ class TnsCsvVisitImportReader:
 
         self.importer_user = None
 
-        importer_user_name = getattr(settings, "IMPORTER_USER", None)
-        if importer_user_name is not None:
+        importer_user_name = ConfigurationItem.objects.get(type=IMPORTER_USER).value
+        if importer_user_name is not None and importer_user_name != '':
             user = User.objects.filter(username=importer_user_name)
             if user is None:
-                logger.warn("User does not exist: " + importer_user_name)
+                logger.warning("User does not exist: " + importer_user_name)
             else:
                 self.importer_user = Worker.objects.filter(user=user)
 
@@ -100,13 +100,13 @@ class TnsCsvVisitImportReader:
                                 continue
                             description = '{} changed from "{}" to "{}"'.format(field, old_value, new_value)
                             p = Provenance(modified_table=Visit._meta.db_table,
-                                        modified_table_id=visit.id,
-                                        modification_author=self.importer_user,
-                                        previous_value=old_value,
-                                        new_value=new_value,
-                                        modification_description=description,
-                                        modified_field=field,
-                                        )
+                                           modified_table_id=visit.id,
+                                           modification_author=self.importer_user,
+                                           previous_value=old_value,
+                                           new_value=new_value,
+                                           modification_description=description,
+                                           modified_field=field,
+                                           )
                             setattr(visit, field, new_value)
                             p.save()
                         visit.save()
@@ -115,30 +115,30 @@ class TnsCsvVisitImportReader:
                                                      datetime_begin=date,
                                                      datetime_end=date + datetime.timedelta(days=14))
                         visit.save()
-                        #visit does not have id until .save() is done
+                        # visit does not have id until .save() is done
                         for field in Visit._meta.get_fields():
                             if field.get_internal_type() == "CharField" or field.get_internal_type() == "DateField" or field.get_internal_type() is "BooleanField":
                                 new_value = getattr(visit, field.name)
                                 if new_value is not None and new_value != "":
                                     description = '{} changed from "{}" to "{}"'.format(field, '', new_value)
                                     p = Provenance(modified_table=Visit._meta.db_table,
-                                                modified_table_id=visit.id,
-                                                modification_author=self.importer_user,
-                                                previous_value='',
-                                                new_value=new_value,
-                                                modification_description=description,
-                                                modified_field=field,
-                                                )
+                                                   modified_table_id=visit.id,
+                                                   modification_author=self.importer_user,
+                                                   previous_value='',
+                                                   new_value=new_value,
+                                                   modification_description=description,
+                                                   modified_field=field,
+                                                   )
                                     p.save()
-                        
+
                     result.append(visit)
 
                     appointments = Appointment.objects.filter(visit=visit, appointment_types=self.appointment_type)
                     if len(appointments) > 0:
                         logger.debug("Appointment for subject " + nd_number + " already set. Updating")
                         appointment = appointments[0]
-                        
-                        #(field, new_value)
+
+                        # (field, new_value)
                         changes = [('length', 60),
                                    ('datetime_when', date),
                                    ('location', location)]
@@ -149,13 +149,13 @@ class TnsCsvVisitImportReader:
                                 continue
                             description = '{} changed from "{}" to "{}"'.format(field, old_value, new_value)
                             p = Provenance(modified_table=Appointment._meta.db_table,
-                                        modified_table_id=appointment.id,
-                                        modification_author=self.importer_user,
-                                        previous_value=old_value,
-                                        new_value=new_value,
-                                        modification_description=description,
-                                        modified_field=field,
-                                        )
+                                           modified_table_id=appointment.id,
+                                           modification_author=self.importer_user,
+                                           previous_value=old_value,
+                                           new_value=new_value,
+                                           modification_description=description,
+                                           modified_field=field,
+                                           )
                             setattr(appointment, field, new_value)
                             p.save()
                         appointment.save()
@@ -165,22 +165,22 @@ class TnsCsvVisitImportReader:
                         if self.appointment_type is not None:
                             AppointmentTypeLink.objects.create(appointment_id=appointment.id,
                                                                appointment_type=self.appointment_type)
-                        
+
                         appointment.save()
-                        #appointment does not have id until .save() is done
+                        # appointment does not have id until .save() is done
                         for field in Appointment._meta.get_fields():
                             if field.get_internal_type() == "CharField" or field.get_internal_type() == "DateField" or field.get_internal_type() is "BooleanField":
                                 new_value = getattr(appointment, field.name)
                                 if new_value is not None and new_value != "":
                                     description = '{} changed from "{}" to "{}"'.format(field, '', new_value)
                                     p = Provenance(modified_table=Appointment._meta.db_table,
-                                                modified_table_id=appointment.id,
-                                                modification_author=self.importer_user,
-                                                previous_value='',
-                                                new_value=new_value,
-                                                modification_description=description,
-                                                modified_field=field,
-                                                )
+                                                   modified_table_id=appointment.id,
+                                                   modification_author=self.importer_user,
+                                                   previous_value='',
+                                                   new_value=new_value,
+                                                   modification_description=description,
+                                                   modified_field=field,
+                                                   )
                                     p.save()
                     self.processed_count += 1
                 except:
diff --git a/smash/web/importer/exporter_cron_job.py b/smash/web/importer/exporter_cron_job.py
index 4af44d80a508c1b5678d0f578020f2e67fcf85d0..8e4849457d8a9ea72e48ad5ae77de26c01e3dc2c 100644
--- a/smash/web/importer/exporter_cron_job.py
+++ b/smash/web/importer/exporter_cron_job.py
@@ -3,29 +3,33 @@ import logging
 import traceback
 
 import timeout_decorator
-from django.conf import settings
 from django_cron import CronJobBase, Schedule
 
+from web.models import ConfigurationItem
+from web.models.constants import CRON_JOB_TIMEOUT, DEFAULT_FROM_EMAIL, DAILY_SUBJECT_EXPORT_FILE, \
+    DAILY_VISIT_EXPORT_FILE, SUBJECT_EXPORT_RUN_AT, VISIT_EXPORT_RUN_AT
+from web.smash_email import EmailSender
 from .exporter import SubjectExporter, VisitExporter
-from web.models.constants import CRON_JOB_TIMEOUT
-from ..smash_email import EmailSender
 
 logger = logging.getLogger(__name__)
 
 
 class SubjectExporterCronJob(CronJobBase):
-    RUN_AT_TIMES = getattr(settings, "EXPORT_RUN_AT", ['23:55'])
+    item = ConfigurationItem.objects.filter(type=SUBJECT_EXPORT_RUN_AT).first()
+    RUN_AT_TIMES = []
+    if item is not None:
+        RUN_AT_TIMES = item.value.split(';')
     schedule = Schedule(run_at_times=RUN_AT_TIMES)
     code = 'web.export_subject_daily_job'  # a unique code
 
     @timeout_decorator.timeout(CRON_JOB_TIMEOUT)
     def do(self):
         email_title = "Daily subject export"
-        email_recipients = getattr(settings, "DEFAULT_FROM_EMAIL", None)
+        email_recipients = ConfigurationItem.objects.get(type=DEFAULT_FROM_EMAIL).value
 
-        filename = getattr(settings, "DAILY_SUBJECT_EXPORT_FILE", None)
+        filename = ConfigurationItem.objects.get(type=DAILY_SUBJECT_EXPORT_FILE).value
 
-        if filename is None:
+        if filename is None or filename == '':
             logger.info("Exporting subjects skipped. File not defined ")
             return "export file not defined"
         logger.info("Exporting subjects to file: " + filename)
@@ -47,18 +51,21 @@ class SubjectExporterCronJob(CronJobBase):
 
 
 class VisitExporterCronJob(CronJobBase):
-    RUN_AT_TIMES = getattr(settings, "EXPORT_RUN_AT", ['23:55'])
+    item = ConfigurationItem.objects.filter(type=VISIT_EXPORT_RUN_AT).first()
+    RUN_AT_TIMES = []
+    if item is not None:
+        RUN_AT_TIMES = item.value.split(';')
     schedule = Schedule(run_at_times=RUN_AT_TIMES)
     code = 'web.export_visit_daily_job'  # a unique code
 
     @timeout_decorator.timeout(CRON_JOB_TIMEOUT)
     def do(self):
         email_title = "Daily visit export"
-        email_recipients = getattr(settings, "DEFAULT_FROM_EMAIL", None)
+        email_recipients = ConfigurationItem.objects.get(type=DEFAULT_FROM_EMAIL).value
 
-        filename = getattr(settings, "DAILY_VISIT_EXPORT_FILE", None)
+        filename = ConfigurationItem.objects.get(type=DAILY_VISIT_EXPORT_FILE).value
 
-        if filename is None:
+        if filename is None or filename == '':
             logger.info("Exporting visit skipped. File not defined ")
             return "export file not defined"
         logger.info("Exporting visits to file: " + filename)
diff --git a/smash/web/importer/importer.py b/smash/web/importer/importer.py
index 84eece4d198d39747c32fdedddd829abd17abfcc..00d2bf1d7c2c633ee9c36e6a35b37abd794e32ee 100644
--- a/smash/web/importer/importer.py
+++ b/smash/web/importer/importer.py
@@ -3,13 +3,12 @@ import logging
 import sys
 import traceback
 
-from django.conf import settings
 from django.contrib.auth.models import User
 
+from web.models import StudySubject, Subject, Provenance, Worker, ConfigurationItem
+from web.models.constants import GLOBAL_STUDY_ID, IMPORTER_USER
 from .subject_import_reader import SubjectImportReader
 from .warning_counter import MsgCounterHandler
-from web.models import StudySubject, Subject, Provenance, Worker
-from web.models.constants import GLOBAL_STUDY_ID
 
 logger = logging.getLogger(__name__)
 
@@ -27,11 +26,11 @@ class Importer(object):
 
         self.importer_user = None
 
-        importer_user_name = getattr(settings, "IMPORTER_USER", None)
-        if importer_user_name is not None:
+        importer_user_name = ConfigurationItem.objects.get(type=IMPORTER_USER).value
+        if importer_user_name is not None and importer_user_name != '':
             user = User.objects.filter(username=importer_user_name)
             if user is None:
-                logger.warn("User does not exist: " + importer_user_name)
+                logger.warning("User does not exist: " + importer_user_name)
             else:
                 self.importer_user = Worker.objects.filter(user=user)
 
diff --git a/smash/web/importer/importer_cron_job.py b/smash/web/importer/importer_cron_job.py
index db4a2da3788785258bcd2b70c372af309c1d4988..1fe761e1fe19a46e95af643a52aa21d86e5f55d7 100644
--- a/smash/web/importer/importer_cron_job.py
+++ b/smash/web/importer/importer_cron_job.py
@@ -6,31 +6,35 @@ import os.path
 import traceback
 
 import timeout_decorator
-from django.conf import settings
 from django_cron import CronJobBase, Schedule
 
+from web.models import ConfigurationItem
+from web.models.constants import CRON_JOB_TIMEOUT, DEFAULT_FROM_EMAIL, DAILY_SUBJECT_IMPORT_FILE, \
+    DAILY_VISIT_IMPORT_FILE, SUBJECT_IMPORT_RUN_AT, VISIT_IMPORT_RUN_AT
+from web.smash_email import EmailSender
 from .csv_tns_subject_import_reader import TnsCsvSubjectImportReader
 from .csv_tns_visit_import_reader import TnsCsvVisitImportReader
 from .importer import Importer
-from web.models.constants import CRON_JOB_TIMEOUT
-from ..smash_email import EmailSender
 
 logger = logging.getLogger(__name__)
 
 
 class SubjectImporterCronJob(CronJobBase):
-    RUN_AT_TIMES = getattr(settings, "SUBJECT_IMPORT_RUN_AT", ['23:45'])
+    item = ConfigurationItem.objects.filter(type=SUBJECT_IMPORT_RUN_AT).first()
+    RUN_AT_TIMES = []
+    if item is not None:
+        RUN_AT_TIMES = item.value.split(';')
     schedule = Schedule(run_at_times=RUN_AT_TIMES)
     code = 'web.import_subjects_daily_job'  # a unique code
 
     @timeout_decorator.timeout(CRON_JOB_TIMEOUT)
     def do(self):
         email_title = "Subjects daily import"
-        email_recipients = getattr(settings, "DEFAULT_FROM_EMAIL", None)
+        email_recipients = ConfigurationItem.objects.get(type=DEFAULT_FROM_EMAIL).value
 
-        filename = getattr(settings, "DAILY_SUBJECT_IMPORT_FILE", None)
+        filename = ConfigurationItem.objects.get(type=DAILY_SUBJECT_IMPORT_FILE).value
 
-        if filename is None:
+        if filename is None or filename == '':
             logger.info("Importing subjects skipped. File not defined ")
             return "import file not defined"
         logger.info("Importing subjects from file: " + filename)
@@ -40,7 +44,7 @@ class SubjectImporterCronJob(CronJobBase):
                                      email_recipients)
             return "import file not found"
         try:
-            importer = Importer(settings.DAILY_SUBJECT_IMPORT_FILE, TnsCsvSubjectImportReader())
+            importer = Importer(filename, TnsCsvSubjectImportReader())
             importer.execute()
             email_body = importer.get_summary()
             EmailSender().send_email(email_title,
@@ -63,18 +67,21 @@ class SubjectImporterCronJob(CronJobBase):
 
 
 class VisitImporterCronJob(CronJobBase):
-    RUN_AT_TIMES = getattr(settings, "VISIT_IMPORT_RUN_AT", ['23:55'])
+    item = ConfigurationItem.objects.filter(type=VISIT_IMPORT_RUN_AT).first()
+    RUN_AT_TIMES = []
+    if item is not None:
+        RUN_AT_TIMES = item.value.split(';')
     schedule = Schedule(run_at_times=RUN_AT_TIMES)
     code = 'web.import_visits_daily_job'  # a unique code
 
     @timeout_decorator.timeout(CRON_JOB_TIMEOUT)
     def do(self):
         email_title = "Visits daily import"
-        email_recipients = getattr(settings, "DEFAULT_FROM_EMAIL", None)
+        email_recipients = ConfigurationItem.objects.get(type=DEFAULT_FROM_EMAIL).value
 
-        filename = getattr(settings, "DAILY_VISIT_IMPORT_FILE", None)
+        filename = ConfigurationItem.objects.get(type=DAILY_VISIT_IMPORT_FILE).value
 
-        if filename is None:
+        if filename is None or filename == '':
             logger.info("Importing visits skipped. File not defined ")
             return "import file not defined"
         logger.info("Importing visits from file: " + filename)
diff --git a/smash/web/migrations/0176_configurationitem_local_setting_clean.py b/smash/web/migrations/0176_configurationitem_local_setting_clean.py
new file mode 100644
index 0000000000000000000000000000000000000000..3723e5377c976ee1736dba7245b31f0baf8372a4
--- /dev/null
+++ b/smash/web/migrations/0176_configurationitem_local_setting_clean.py
@@ -0,0 +1,85 @@
+# -*- coding: utf-8 -*-
+from django.conf import settings
+from django.db import migrations
+
+from web.models.constants import DEFAULT_FROM_EMAIL, DAILY_SUBJECT_IMPORT_FILE, DAILY_SUBJECT_EXPORT_FILE, \
+    DAILY_VISIT_IMPORT_FILE, DAILY_VISIT_EXPORT_FILE, SUBJECT_IMPORT_RUN_AT, SUBJECT_EXPORT_RUN_AT, \
+    VISIT_EXPORT_RUN_AT, VISIT_IMPORT_RUN_AT, IMPORTER_USER, IMPORT_APPOINTMENT_TYPE, NEXMO_API_KEY, NEXMO_API_SECRET, \
+    NEXMO_DEFAULT_FROM, LOGIN_PAGE_BACKGROUND_IMAGE
+
+
+def create_item(apps, item_type, value, name):
+    # We can't import the ConfigurationItem model directly as it may be a newer
+    # version than this migration expects. We use the historical version.
+
+    # noinspection PyPep8Naming
+    ConfigurationItem = apps.get_model("web", "ConfigurationItem")
+    item = ConfigurationItem.objects.create()
+    item.type = item_type
+    item.value = value
+    item.name = name
+    item.save()
+
+
+# noinspection PyUnusedLocal
+def configuration_items(apps, schema_editor):
+    email_from = getattr(settings, "DEFAULT_FROM_EMAIL", '')
+    create_item(apps, DEFAULT_FROM_EMAIL, email_from,
+                "Default email address used in the from field when sending emails")
+
+    subject_import_file = getattr(settings, "DAILY_SUBJECT_IMPORT_FILE", '')
+    create_item(apps, DAILY_SUBJECT_IMPORT_FILE, subject_import_file, "File used to import subjects automatically")
+
+    subject_export_file = getattr(settings, "DAILY_SUBJECT_EXPORT_FILE", '')
+    create_item(apps, DAILY_SUBJECT_EXPORT_FILE, subject_export_file, "File used to export subjects automatically")
+
+    visit_import_file = getattr(settings, "DAILY_VISIT_IMPORT_FILE", '')
+    create_item(apps, DAILY_VISIT_IMPORT_FILE, visit_import_file, "File used to import visits automatically")
+
+    visit_export_file = getattr(settings, "DAILY_VISIT_EXPORT_FILE", '')
+    create_item(apps, DAILY_VISIT_EXPORT_FILE, visit_export_file, "File used to export visits automatically")
+
+    subject_import_run_at_times = getattr(settings, "SUBJECT_IMPORT_RUN_AT", ['23:45'])
+    create_item(apps, SUBJECT_IMPORT_RUN_AT, ';'.join(subject_import_run_at_times),
+                "At what times should the subject importer run")
+
+    export_run_at_times = getattr(settings, "EXPORT_RUN_AT", ['23:55'])
+    create_item(apps, SUBJECT_EXPORT_RUN_AT, ';'.join(export_run_at_times),
+                "At what times should the subject exporter run")
+    create_item(apps, VISIT_EXPORT_RUN_AT, ';'.join(export_run_at_times),
+                "At what times should the visit exporter run")
+
+    visit_import_run_at_times = getattr(settings, "VISIT_IMPORT_RUN_AT", ['23:55'])
+    create_item(apps, VISIT_IMPORT_RUN_AT, ';'.join(visit_import_run_at_times),
+                "At what times should the visit importer run")
+
+    importer_user_name = getattr(settings, "IMPORTER_USER", '')
+    create_item(apps, IMPORTER_USER, importer_user_name,
+                "User that should be assigned to changes introduced by importer")
+
+    appointment_code = getattr(settings, "IMPORT_APPOINTMENT_TYPE", "SAMPLES")
+    create_item(apps, IMPORT_APPOINTMENT_TYPE, appointment_code,
+                "Type of appointment assigned to imported visits")
+
+    nexmo_api_key = getattr(settings, 'NEXMO_API_KEY', '')
+    create_item(apps, NEXMO_API_KEY, nexmo_api_key, "NEXMO API KEY")
+
+    nexmo_api_secret = getattr(settings, 'NEXMO_API_SECRET', '')
+    create_item(apps, NEXMO_API_SECRET, nexmo_api_secret, "NEXMO API SECRET")
+
+    nexmo_api_from = getattr(settings, 'NEXMO_DEFAULT_FROM', '')
+    create_item(apps, NEXMO_DEFAULT_FROM, nexmo_api_from, "The sender of the message from NEXMO (phone number or text)")
+
+    background_image = getattr(settings, "LOGIN_PAGE_BACKGROUND_IMAGE", "background.jpg")
+    create_item(apps, LOGIN_PAGE_BACKGROUND_IMAGE, background_image,
+                "Path to a static file containing background image, used in login.html")
+
+
+class Migration(migrations.Migration):
+    dependencies = [
+        ('web', '0175_auto_20201109_1404'),
+    ]
+
+    operations = [
+        migrations.RunPython(configuration_items),
+    ]
diff --git a/smash/web/models/constants.py b/smash/web/models/constants.py
index 5b7c92f906c90c11377ac8ebc039a68185971d97..f061f3061fe03099a3355dd489092b1cb96f3fcf 100644
--- a/smash/web/models/constants.py
+++ b/smash/web/models/constants.py
@@ -56,6 +56,21 @@ KIT_EMAIL_DAY_OF_WEEK_CONFIGURATION_TYPE = "KIT_EMAIL_DAY_OF_WEEK_CONFIGURATION_
 KIT_DAILY_EMAIL_DAYS_PERIOD_TYPE = "KIT_DAILY_EMAIL_DAYS_PERIOD_TYPE"
 KIT_DAILY_EMAIL_TIME_FORMAT_TYPE = "KIT_DAILY_EMAIL_TIME_FORMAT_TYPE"
 VIRUS_EMAIL_HOUR_CONFIGURATION_TYPE = "VIRUS_EMAIL_HOUR_CONFIGURATION_TYPE"
+DEFAULT_FROM_EMAIL = "DEFAULT_FROM_EMAIL"
+DAILY_SUBJECT_IMPORT_FILE = "DAILY_SUBJECT_IMPORT_FILE"
+DAILY_SUBJECT_EXPORT_FILE = "DAILY_SUBJECT_EXPORT_FILE"
+DAILY_VISIT_IMPORT_FILE = "DAILY_VISIT_IMPORT_FILE"
+DAILY_VISIT_EXPORT_FILE = "DAILY_VISIT_EXPORT_FILE"
+SUBJECT_IMPORT_RUN_AT = "SUBJECT_IMPORT_RUN_AT"
+SUBJECT_EXPORT_RUN_AT = "SUBJECT_EXPORT_RUN_AT"
+VISIT_EXPORT_RUN_AT = "VISIT_EXPORT_RUN_AT"
+VISIT_IMPORT_RUN_AT = "VISIT_IMPORT_RUN_AT"
+IMPORTER_USER = "IMPORTER_USER"
+IMPORT_APPOINTMENT_TYPE = "IMPORT_APPOINTMENT_TYPE"
+NEXMO_API_KEY = "NEXMO_API_KEY"
+NEXMO_API_SECRET = "NEXMO_API_SECRET"
+NEXMO_DEFAULT_FROM = "NEXMO_DEFAULT_FROM"
+LOGIN_PAGE_BACKGROUND_IMAGE = "LOGIN_PAGE_BACKGROUND_IMAGE"
 
 RED_CAP_LANGUAGE_4_FIELD_TYPE = 'RED_CAP_LANGUAGE_4_FIELD_TYPE'
 RED_CAP_LANGUAGE_3_FIELD_TYPE = 'RED_CAP_LANGUAGE_3_FIELD_TYPE'
@@ -72,7 +87,6 @@ RED_CAP_SAMPLE_DATE_FIELD_TYPE = "RED_CAP_SAMPLE_DATE_FIELD_TYPE"
 RED_CAP_IGA_STATUS_FIELD_TYPE = "RED_CAP_IGA_STATUS_FIELD_TYPE"
 RED_CAP_IGG_STATUS_FIELD_TYPE = "RED_CAP_IGG_STATUS_FIELD_TYPE"
 
-
 MAIL_TEMPLATE_CONTEXT_SUBJECT = 'S'
 MAIL_TEMPLATE_CONTEXT_APPOINTMENT = 'A'
 MAIL_TEMPLATE_CONTEXT_VISIT = 'V'
diff --git a/smash/web/nexmo_gateway.py b/smash/web/nexmo_gateway.py
index 0b7190bab23a30ad0c22c79dd8759960056e7422..4191639277ed298ed9a3675dc3ac75166d1de78c 100644
--- a/smash/web/nexmo_gateway.py
+++ b/smash/web/nexmo_gateway.py
@@ -1,7 +1,9 @@
 import logging
 
 import nexmo
-from django.conf import settings
+
+from web.models import ConfigurationItem
+from web.models.constants import NEXMO_API_KEY, NEXMO_API_SECRET, NEXMO_DEFAULT_FROM
 
 logger = logging.getLogger(__name__)
 
@@ -26,8 +28,10 @@ class Nexmo:
     """
 
     def __init__(self):
-        self.client = nexmo.Client(key=getattr(settings, 'NEXMO_API_KEY'), secret=getattr(settings, 'NEXMO_API_SECRET'))
-        self.default_from = getattr(settings, 'NEXMO_DEFAULT_FROM')
+        api_key = ConfigurationItem.objects.get(type=NEXMO_API_KEY).value
+        api_secret = ConfigurationItem.objects.get(type=NEXMO_API_SECRET).value
+        self.client = nexmo.Client(key=api_key, secret=api_secret)
+        self.default_from = ConfigurationItem.objects.get(type=NEXMO_DEFAULT_FROM).value
 
     def send_sms(self, device, token):
         body = 'Your authentication token is %s' % token
diff --git a/smash/web/redcap_connector.py b/smash/web/redcap_connector.py
index fde5e61eb13e30a2ad7ea0d1b4d2304b63e6616f..ef62c8bfabd6f7dfbc6ccca165285fcade332571 100644
--- a/smash/web/redcap_connector.py
+++ b/smash/web/redcap_connector.py
@@ -7,7 +7,6 @@ import logging
 import certifi
 import pycurl
 import timeout_decorator
-from django.conf import settings
 from django.forms.models import model_to_dict
 from django_cron import CronJobBase, Schedule
 
@@ -20,7 +19,7 @@ from web.models.constants import REDCAP_TOKEN_CONFIGURATION_TYPE, \
     RED_CAP_LANGUAGE_2_FIELD_TYPE, RED_CAP_LANGUAGE_1_FIELD_TYPE, RED_CAP_MPOWER_ID_FIELD_TYPE, RED_CAP_DEAD_FIELD_TYPE, \
     RED_CAP_SEX_FIELD_TYPE, RED_CAP_DATE_BORN_FIELD_TYPE, RED_CAP_ND_NUMBER_FIELD_TYPE, RED_CAP_VIRUS_FIELD_TYPE, \
     GLOBAL_STUDY_ID, RED_CAP_SAMPLE_DATE_FIELD_TYPE, RED_CAP_KIT_ID_FIELD_TYPE, RED_CAP_IGA_STATUS_FIELD_TYPE, \
-    RED_CAP_IGG_STATUS_FIELD_TYPE
+    RED_CAP_IGG_STATUS_FIELD_TYPE, IMPORTER_USER, IMPORT_APPOINTMENT_TYPE
 from web.models.inconsistent_subject import InconsistentField, InconsistentSubject
 from web.models.missing_subject import MissingSubject
 
@@ -114,11 +113,11 @@ class RedcapConnector(object):
 
         self.importer_user = None
 
-        importer_user_name = getattr(settings, "IMPORTER_USER", None)
-        if importer_user_name is not None:
+        importer_user_name = ConfigurationItem.objects.get(type=IMPORTER_USER).value
+        if importer_user_name is not None and importer_user_name != '':
             user = User.objects.filter(username=importer_user_name)
             if user is None:
-                logger.warn("User does not exist: " + importer_user_name)
+                logger.warning("User does not exist: " + importer_user_name)
             else:
                 self.importer_user = Worker.objects.filter(user=user)
 
@@ -196,7 +195,7 @@ class RedcapConnector(object):
         self.add_inconsistent(inconsistent)
 
     def find_inconsistent(self):
-        appointment_type_code_to_finish = getattr(settings, "IMPORT_APPOINTMENT_TYPE", None)
+        appointment_type_code_to_finish = ConfigurationItem.objects.get(type=IMPORT_APPOINTMENT_TYPE).value
         appointment_type_to_finish = None
         if appointment_type_code_to_finish is not None:
             appointment_types = AppointmentType.objects.filter(code=appointment_type_code_to_finish)
diff --git a/smash/web/smash_email.py b/smash/web/smash_email.py
index f6fe326c7d9b8500602a26a1d99a3d6dcbc70d6b..1f06b7b60e6955df4f9045d5811ad4873aec1887 100644
--- a/smash/web/smash_email.py
+++ b/smash/web/smash_email.py
@@ -2,10 +2,11 @@
 
 import logging
 
-from django.conf import settings
-from django.core import mail
 from django.core.mail import EmailMessage
 
+from web.models import ConfigurationItem
+from web.models.constants import DEFAULT_FROM_EMAIL
+
 logger = logging.getLogger(__name__)
 
 
@@ -13,7 +14,7 @@ class EmailSender(object):
     def send_email(self, subject, body, recipients, cc_recipients=None):
         if cc_recipients is None:
             cc_recipients = []
-        email_from = getattr(settings, "DEFAULT_FROM_EMAIL", None)
+        email_from = ConfigurationItem.objects.get(type=DEFAULT_FROM_EMAIL).value
         recipient_list = []
         for recipient in recipients.split(";"):
             recipient_list.append(recipient)
diff --git a/smash/web/tests/importer/test_exporter_cron_job.py b/smash/web/tests/importer/test_exporter_cron_job.py
index 7bb15fc9e5924e97912e7a19d65f9e6c1e7dd67a..e2b657c13727fbf5aeee6c2d381df7e26191121a 100644
--- a/smash/web/tests/importer/test_exporter_cron_job.py
+++ b/smash/web/tests/importer/test_exporter_cron_job.py
@@ -3,24 +3,19 @@
 import logging
 import tempfile
 
-from django.conf import settings
+from django.core import mail
 from django.test import TestCase
+from django_cron.models import CronJobLog
 
 from web.importer import SubjectExporterCronJob
+from web.models import ConfigurationItem
+from web.models.constants import DAILY_SUBJECT_EXPORT_FILE
 
 logger = logging.getLogger(__name__)
-from django.core import mail
-from django_cron.models import CronJobLog
 
 
 class TestCronJobExporter(TestCase):
 
-    def setUp(self):
-        setattr(settings, "DAILY_SUBJECT_EXPORT_FILE", None)
-
-    def tearDown(self):
-        setattr(settings, "DAILY_SUBJECT_EXPORT_FILE", None)
-
     def test_import_without_configuration(self):
         CronJobLog.objects.all().delete()
 
@@ -34,7 +29,10 @@ class TestCronJobExporter(TestCase):
     def test_import(self):
         new_file, tmp = tempfile.mkstemp()
 
-        setattr(settings, "DAILY_SUBJECT_EXPORT_FILE", tmp)
+        item = ConfigurationItem.objects.get(type=DAILY_SUBJECT_EXPORT_FILE)
+        item.value = tmp
+        item.save()
+
         CronJobLog.objects.all().delete()
 
         job = SubjectExporterCronJob()
diff --git a/smash/web/tests/importer/test_importer_cron_job.py b/smash/web/tests/importer/test_importer_cron_job.py
index 685d5159688408cbb9792d65a17fdf1b61cd513b..2fbfe03ffe435e085239594b7475ce95ac335e47 100644
--- a/smash/web/tests/importer/test_importer_cron_job.py
+++ b/smash/web/tests/importer/test_importer_cron_job.py
@@ -4,22 +4,20 @@ import logging
 import tempfile
 from shutil import copyfile
 
-from django.conf import settings
+from django.core import mail
 from django.test import TestCase
+from django_cron.models import CronJobLog
 
 from web.importer import SubjectImporterCronJob
+from web.models import ConfigurationItem
+from web.models.constants import DAILY_SUBJECT_IMPORT_FILE
 from web.tests.functions import get_resource_path
 
 logger = logging.getLogger(__name__)
-from django.core import mail
-from django_cron.models import CronJobLog
 
 
 class TestCronJobImporter(TestCase):
 
-    def tearDown(self):
-        setattr(settings, "DAILY_SUBJECT_IMPORT_FILE", None)
-
     def test_import_without_configuration(self):
         CronJobLog.objects.all().delete()
 
@@ -35,7 +33,9 @@ class TestCronJobImporter(TestCase):
         new_file, tmp = tempfile.mkstemp()
         copyfile(filename, tmp)
 
-        setattr(settings, "DAILY_SUBJECT_IMPORT_FILE", tmp)
+        conf = ConfigurationItem.objects.get(type=DAILY_SUBJECT_IMPORT_FILE)
+        conf.value = tmp
+        conf.save()
         CronJobLog.objects.all().delete()
 
         job = SubjectImporterCronJob()
diff --git a/smash/web/tests/importer/test_tns_csv_visit_import_reader.py b/smash/web/tests/importer/test_tns_csv_visit_import_reader.py
index 475bcf4726a894eff54750c8b88fd452e919d0cb..d15bc9f823833b628c92a0886a06e17fb87cd0dc 100644
--- a/smash/web/tests/importer/test_tns_csv_visit_import_reader.py
+++ b/smash/web/tests/importer/test_tns_csv_visit_import_reader.py
@@ -2,12 +2,13 @@
 
 import logging
 
-from django.conf import settings
+import pytz
 from django.test import TestCase
 from django.utils import timezone
 
 from web.importer import TnsCsvVisitImportReader, MsgCounterHandler
-from web.models import Appointment, Visit, StudySubject, AppointmentTypeLink, AppointmentType
+from web.models import Appointment, Visit, StudySubject, AppointmentTypeLink, AppointmentType, ConfigurationItem
+from web.models.constants import IMPORT_APPOINTMENT_TYPE
 from web.tests.functions import get_resource_path, create_study_subject, create_appointment_type, create_location
 
 logger = logging.getLogger(__name__)
@@ -17,7 +18,10 @@ class TestTnsCsvVisitReader(TestCase):
     def setUp(self):
         self.warning_counter = MsgCounterHandler()
         logging.getLogger('').addHandler(self.warning_counter)
-        setattr(settings, "IMPORT_APPOINTMENT_TYPE", "SAMPLE_2")
+
+        item = ConfigurationItem.objects.get(type=IMPORT_APPOINTMENT_TYPE)
+        item.value = "SAMPLE_2"
+        item.save()
         create_appointment_type(code="SAMPLE_2")
 
         create_study_subject(nd_number='cov-000111')
@@ -29,7 +33,6 @@ class TestTnsCsvVisitReader(TestCase):
         create_location(name="Ketterthill	1-3, rue de la Continentale 	4917	Bascharage")
 
     def tearDown(self):
-        setattr(settings, "IMPORT_APPOINTMENT_TYPE", None)
         logging.getLogger('').removeHandler(self.warning_counter)
 
     def test_load_data(self):