2013-10-10 19:56:40 +00:00
|
|
|
'''
|
|
|
|
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)
|
2013-10-10 19:56:40 +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
|
2013-10-10 19:56:40 +00:00
|
|
|
with a process((self, authenticator, success_url, deny_url) method which is expected to return a response
|
2013-12-02 04:58:58 +00:00
|
|
|
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)
|
2013-10-10 19:56:40 +00:00
|
|
|
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
|
2013-10-10 19:56:40 +00:00
|
|
|
|
|
|
|
'''
|
2013-10-08 19:37:22 +00:00
|
|
|
import logging
|
|
|
|
from django.db.models import Q
|
2013-10-10 06:41:50 +00:00
|
|
|
from django import forms
|
2013-10-08 19:37:22 +00:00
|
|
|
from django.http import HttpResponseRedirect
|
|
|
|
from django.shortcuts import render
|
2013-10-02 16:02:08 +00:00
|
|
|
|
2013-11-30 20:40:45 +00:00
|
|
|
from .models import Block, IP, LibraryUser, CardPattern, EmailPattern
|
|
|
|
|
2013-10-08 19:37:22 +00:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2013-12-02 04:58:58 +00:00
|
|
|
class ip:
|
|
|
|
def authenticate(self,request, library):
|
|
|
|
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:
|
|
|
|
if block.library==library:
|
|
|
|
logger.info('%s authenticated for %s from %s'%(request.user, library, ip))
|
|
|
|
library.credential=ip
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
except KeyError:
|
|
|
|
return False
|
2013-10-08 19:37:22 +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
|
2013-10-02 16:02:08 +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",)
|
2013-10-08 19:37:22 +00:00
|
|
|
|
2013-12-02 04:58:58 +00:00
|
|
|
class cardnum:
|
|
|
|
def authenticate(self,request, library):
|
|
|
|
return False
|
|
|
|
|
|
|
|
class authenticator():
|
2013-12-02 23:19:11 +00:00
|
|
|
def process(self, caller, success_url, deny_url):
|
|
|
|
if caller.form and caller.request.method=='POST' and caller.form.is_valid():
|
|
|
|
library = caller.form.cleaned_data['library']
|
|
|
|
library.credential = caller.form.cleaned_data['credential']
|
|
|
|
logger.info('%s authenticated for %s from %s'%(caller.request.user, caller.library, caller.form.cleaned_data.get('number')))
|
|
|
|
library.add_user(caller.form.cleaned_data['user'])
|
2013-12-02 04:58:58 +00:00
|
|
|
return HttpResponseRedirect(success_url)
|
|
|
|
else:
|
2013-12-02 23:19:11 +00:00
|
|
|
return render(caller.request, 'libraryauth/library.html', {
|
|
|
|
'library':caller.library,
|
|
|
|
'authenticator':caller,
|
2013-12-02 04:58:58 +00:00
|
|
|
})
|
2013-10-10 06:41:50 +00:00
|
|
|
|
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",)
|
2013-11-30 20:40:45 +00:00
|
|
|
|
2013-12-02 04:58:58 +00:00
|
|
|
class form(forms.ModelForm):
|
|
|
|
credential = forms.RegexField(
|
|
|
|
label="Enter Your Library Card Number",
|
|
|
|
max_length=20,
|
|
|
|
regex=r'^\d+$',
|
|
|
|
required = True,
|
|
|
|
help_text = "(digits only)",
|
|
|
|
error_messages = {'invalid': "digits only!",}
|
|
|
|
)
|
|
|
|
def __init__(self, request, library, *args, **kwargs):
|
|
|
|
if request.method=="POST":
|
|
|
|
data=request.POST
|
|
|
|
super(cardnum.form, self).__init__(data=data)
|
|
|
|
else:
|
|
|
|
initial={'user':request.user, 'library':library}
|
|
|
|
super(cardnum.form, self).__init__(initial=initial)
|
2013-10-10 06:41:50 +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.")
|
2013-10-11 02:31:31 +00:00
|
|
|
|
2013-12-02 04:58:58 +00:00
|
|
|
class Meta:
|
|
|
|
model = LibraryUser
|
|
|
|
widgets = { 'library': forms.HiddenInput, 'user': forms.HiddenInput }
|
2016-07-21 19:52:07 +00:00
|
|
|
exclude = ()
|
2013-12-02 04:58:58 +00:00
|
|
|
class email:
|
|
|
|
def authenticate(self,request, library):
|
|
|
|
if request.user.is_anonymous():
|
|
|
|
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))
|
|
|
|
library.credential=email
|
|
|
|
return True
|
2013-10-11 02:31:31 +00:00
|
|
|
return False
|
|
|
|
|
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",)
|