regluit/libraryauth/backends.py

131 lines
4.8 KiB
Python
Raw Normal View History

'''
to make a backend named <backend> you need to...
2013-12-02 04:58:58 +00:00
1. make a class <backend>
2. with a function authenticate(self, request, library)
2018-07-26 17:59:58 +00:00
returns true if can request.user can be authenticated to the library,
and attaches a credential property to the library object
returns fals if otherwise.
2013-12-02 04:58:58 +00:00
3. with a class authenticator
2018-07-26 17:59:58 +00:00
with a process((self, authenticator, success_url, deny_url) method
which is expected to return a response
4. make a libraryauth/<backend>_join.html template (authenticator will be in its context)
to insert a link or form for a user to join the library
5. if you need to show the user a form, define a model form class form with init method
__init__(self, request, library, *args, **kwargs)
and model LibraryUser
2013-12-02 04:58:58 +00:00
6. define an admin form to let the library configure its authentication
7. add new auth choice to Library.backend choices and the admin as desired
'''
import logging
from django.db.models import Q
from django import forms
from django.http import HttpResponseRedirect
from django.shortcuts import render
from .models import Block, IP, LibraryUser, CardPattern, EmailPattern
2018-07-26 17:59:58 +00:00
logger = logging.getLogger(__name__)
2013-12-02 04:58:58 +00:00
class ip:
2018-07-26 17:59:58 +00:00
def authenticate(self, request, library):
2013-12-02 04:58:58 +00:00
try:
ip = IP(request.META['REMOTE_ADDR'])
blocks = Block.objects.filter(Q(lower=ip) | Q(lower__lte=ip, upper__gte=ip))
for block in blocks:
2018-07-26 17:59:58 +00:00
if block.library == library:
2013-12-02 04:58:58 +00:00
logger.info('%s authenticated for %s from %s'%(request.user, library, ip))
2018-07-26 17:59:58 +00:00
library.credential = ip
2013-12-02 04:58:58 +00:00
return True
return False
except KeyError:
return False
2018-07-26 17:59:58 +00:00
2013-12-02 04:58:58 +00:00
class authenticator():
2013-12-02 23:19:11 +00:00
def process(self, caller, success_url, deny_url):
2013-12-02 04:58:58 +00:00
return HttpResponseRedirect(deny_url)
2018-07-26 17:59:58 +00:00
2013-12-02 04:58:58 +00:00
form = None
2018-07-26 17:59:58 +00:00
2013-12-02 04:58:58 +00:00
class admin_form(forms.ModelForm):
class Meta:
model = Block
2016-04-08 00:39:23 +00:00
exclude = ("library",)
2018-07-26 17:59:58 +00:00
class cardnum:
def authenticate(self, request, library):
2013-12-02 04:58:58 +00:00
return False
class authenticator():
2013-12-02 23:19:11 +00:00
def process(self, caller, success_url, deny_url):
2018-07-26 17:59:58 +00:00
if caller.form and caller.request.method == 'POST' and caller.form.is_valid():
2013-12-02 23:19:11 +00:00
library = caller.form.cleaned_data['library']
library.credential = caller.form.cleaned_data['credential']
2018-07-26 17:59:58 +00:00
logger.info('%s authenticated for %s from %s' % (
caller.request.user,
caller.library,
caller.form.cleaned_data.get('number'),
))
2013-12-02 23:19:11 +00:00
library.add_user(caller.form.cleaned_data['user'])
2013-12-02 04:58:58 +00:00
return HttpResponseRedirect(success_url)
2018-07-26 17:59:58 +00:00
return render(caller.request, 'libraryauth/library.html', {
'library':caller.library,
'authenticator':caller,
})
2013-12-02 04:58:58 +00:00
class admin_form(forms.ModelForm):
class Meta:
model = CardPattern
2016-04-08 00:39:23 +00:00
exclude = ("library",)
2018-07-26 17:59:58 +00:00
2013-12-02 04:58:58 +00:00
class form(forms.ModelForm):
credential = forms.RegexField(
2018-07-26 17:59:58 +00:00
label="Enter Your Library Card Number",
max_length=20,
regex=r'^\d+$',
required=True,
help_text="(digits only)",
error_messages={'invalid': "digits only!",}
)
2013-12-02 04:58:58 +00:00
def __init__(self, request, library, *args, **kwargs):
2018-07-26 17:59:58 +00:00
if request.method == "POST":
data = request.POST
2013-12-02 04:58:58 +00:00
super(cardnum.form, self).__init__(data=data)
else:
2018-07-26 17:59:58 +00:00
initial = {'user':request.user, 'library':library}
2013-12-02 04:58:58 +00:00
super(cardnum.form, self).__init__(initial=initial)
2018-07-26 17:59:58 +00:00
2013-12-02 04:58:58 +00:00
def clean(self):
library = self.cleaned_data.get('library', None)
credential = self.cleaned_data.get('credential', '')
for card_pattern in library.cardnum_auths.all():
if card_pattern.is_valid(credential):
return self.cleaned_data
raise forms.ValidationError("the library card number must be VALID.")
2018-07-26 17:59:58 +00:00
2013-12-02 04:58:58 +00:00
class Meta:
model = LibraryUser
2018-07-26 17:59:58 +00:00
widgets = {'library': forms.HiddenInput, 'user': forms.HiddenInput}
2016-07-21 19:52:07 +00:00
exclude = ()
2018-07-26 17:59:58 +00:00
class email:
def authenticate(self, request, library):
if request.user.is_anonymous:
2013-12-02 04:58:58 +00:00
return False
email = request.user.email
for email_pattern in library.email_auths.all():
if email_pattern.is_valid(email):
logger.info('%s authenticated for %s from %s'%(request.user, library, email))
2018-07-26 17:59:58 +00:00
library.credential = email
2013-12-02 04:58:58 +00:00
return True
2013-10-11 02:31:31 +00:00
return False
2018-07-26 17:59:58 +00:00
2013-12-02 04:58:58 +00:00
class authenticator():
2013-12-02 23:19:11 +00:00
def process(self, caller, success_url, deny_url):
2013-12-02 04:58:58 +00:00
return HttpResponseRedirect(deny_url)
form = None
class admin_form(forms.ModelForm):
class Meta:
model = EmailPattern
2016-04-08 00:39:23 +00:00
exclude = ("library",)