An application designed to ban users. The application is plugged and gives the current User model a new feature: ban, unban, and check ban for users. Bans are accumulable and can be checked, reverted, forgiven, and expired.
-
Install "detention" in your project (i.e. add "detention" in the INSTALLED_APPS tuple).
-
If you have a custom User model, you can specify (in DETENTION_USER_CREATOR setting) a custom function that gets-or-creates a user based on a username. The default implementation looks like:
def create_special_user(model_class, username): return model_class.objects.get_or_create(username=username, defaults={'email': unicode(username) + u'@example.com'} #It will install 3 users (currently __auto_ban__ is not used in the system).
-
The service.DetentionService wrapper class allows you to:
- Check wether the user is banned or not (update: see item 6. Resource-specific banning).
- Ban another user (update: see item 6. Resource-specific banning).
- Forgive/Revert a ban.
This example is a how-to for this app:
klass = get_user_model() admin = klass.objects.get("someUser") admin_wrapper = service.DetentionService(admin) client = klass.objects.get("anotherUser") client_wrapper = service.DetentionService(client) new_ban = admin_wrapper.ban(client, "1d", "one-day sample ban") current_ban = client_wrapper.my_current_ban() assert new_ban == current_ban, "They must match" dead_ban = admin_wrapper.forgive(new_ban, "forgiving sample ban") #admin_wrapper.revert works as well, with same parameters
-
You can wrap your views with class-based decorators that make your view behave differently depending on whether the current user in session is banned or not.
These decorators, upon anonymous user or user being banned, process an alternative flow to the wrapped view.from decorators import ifban, ifban_forbid, ifban_redirect, ifban_same #You can subclass those decorators. You can override: class my_ifban(ifban): def on_anonymous(self, request, *args, **kwargs): #return a response here def on_banned(self, request, *args, **kwargs): #return a response here class my_ifban_redirect(ifban_redirect): def on_anonymous(self, request, *args, **kwargs): #return a response here def get_redirection(self, request, *args, **kwargs): #return a (url, isPermanentRedirect:Boolean) tuple #like ('/', True) being a Permanent Redirect to '/'. class my_ifban_forbid(ifban_forbid): def on_anonymous(self, request, *args, **kwargs): #return a response here def get_content(self, request, *args, **kwargs): #return a (text, content-type name) tuple class my_ifban_same(ifban_same): def on_anonymous(self, request, *args, **kwargs): #return a response here #and instantiate them as: my_ifban(allow_anonymous, ban_attr_name='current_ban', service_attr_name='service', view_attr_name='view') # the 1st param will determine wether not being logged in or not "auth" installed will call the original view or the on_anonymous view. # the 2nd param will be the name of the request attribute holding the current ban (for the on_banned view). # the 3rd param will be the name of the request attribute holding the wrapped user (for the on_banned view, and the original view; may be None if allowed anonymous and no user is logged in). # the 4th param will be the name of the request attribute holding the wrapped view (for the on_banned view, and the on_anonymous view). #finally, wrap views @my_ifban(True) def my_view(request, *args, **kwargs): #return a response here.
-
In detention.signals there are signals that react to user banning and ban termination.
#in each case, the sender will be the banned user. # ban_applied: new_ban (the newly created ban) # ban_terminated: ban (the one that was terminated by an explicit call) # bans_expired: current_ban (if any), ban_list (expired ban list in the current call) #use them with the usual @receiver decorator @receiver(signals.bans_expired, dispatch_uid="example.unbanned") def bans_expired_handler(sender, **kwargs): #... handle pass
-
Resource-specific banning: you can specify a custom resource to indicate the user is banned to operate in that resource.
Whatever you specify as resource must either be any (SAVED, remember it: SAVED) model instance, or leave it blank defaulting to None (it is considered as just another possible value).
Take an example: administrator a1 can ban a user c1 from accessing a room r1 for one day and "you suck" reason. Whatever you specify is business-specific and is up to you to define the meaning (even the None value).
Based on the given example, syntax would be: DetentionService(a1).ban(c1, "1d", "you suck", r1) #or DetentionService(a1).ban(c1, "1d", "you suck", resource=r1).
So, checking for DetentionService(c1).my_current_ban() would give no active ban (considering the only made ban was the previous one).
And so, checking for DetentionService(c1).my_current_ban(r1) would give the recently created ban (assuming you call this in the scope of the duration).
Assuming ds is a DetentionService instance, remember that ds.ban(c1, "1d", "you suck") and ds.my_current_ban() are the same as ds.ban(c1, "1d", "you suck", None) and ds.my_current_ban(None).
These resource-enabled features are not enabled in the ifban views. -
For a more detailed example check the example application. I know this documentation is a crap but dude I hate doc-ing and 'm not in the best mood today, so deal with it.