My Django app currently has URLs which are protected by 'permission_required()' functions.
This function is called in three different ways.
- As a decorator in views.py, with hardcoded parameters.
- As a plain function, with autogenerated parameter, in custom Class Based Generic Views.
- As a function invoking views in urls.py, with hardcoded parameters.
I'm now adding a menu system to the app, and I need to make menu entries reflect whether the user has permission to request the URL of each menu entry. (Either by greying-out or hiding said entries.)
Is there a way of query the permissions required to a URL without requesting the URL?
The only solution I've thought of so far is to replace the decorator with a parameterless 'menu_permssion_required()' decorator and hardcode all of the permissions into a Python structure. This seems like a step backwards, as my custom Class Based Generic Views already autogenerate their required permissions.
Any suggestions on how to make a menu system which reflects URL permissions for the current user?
Here is an example of how to solve your problem:
First, Create a decorator wrapper to use instead of permission_required:
Then, use it to decorate your views:
Then, add a tag to use for your menu items:
And finally, in your templates, when building the menu tree:
N.B. I've not tested the last part with the tag, but I hope you got the idea. The magic thing here is to add the permissions to the view_func in the decorator, and then you can access this using resolve(path). I'm not sure how this will behave in terms of performance, but after all that's just an idea.
EDIT: Just fixed a bug in the example..
User.has_perm() and User.has_module_perms()
I really like this question, because it concerns anyone that makes a website with django, so I find it really relevant. I've been through that myself and even coded a menu "system" in my first django project back in 2008. But since then I tried Pinax, and one of the (so many) things I learnt from their example projects is that it is completely unnecessary bloat.
So, I have no suggestion which I would support on how to make a menu "system" which respects the request user permissions.
I do have a suggestion on how to make a simple menu which respects the request user permissions, so that might not be completely unrelated.
Just make your menu in plain HTML, it's not like it's going to change so often that it has to be generated. That will also keep your Python code simpler.
Add to
settings.TEMPLATE_CONTEXT_PROCESSORS
:'django.core.context_processors.PermWrapper'
Use the
{{ perms }} proxy
to User.has_perms.Example:
That's how I keep navigation simple, stupid, and out of the way. But also I always include an autocomplete not far from the menus to allows the user to navigate to any detail page easily. So, that's all I know about navigation in django projects, I'm eager to read other answers !
I had a similar issue, but it went a little deeper. Instead of just permissions, I also wanted other tests based on the lidded in user (ie,
is_staff
, oruser.units.count() > 1
). Duplicating these in the view and the template seems prone to errors.You can introspect a view object, and see all of the decorators wrapping it, and work out if they are checks (in my case: the first argument I'd
u
oruser
). If they all pass, then allow rendering the link.Get all decorators wrapping a function describes the technique in a little more detail. You can find the app that wraps this up into a handy replacement for
{% url %}
at Django-menus.