Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Reversing namespaced URLs in Django: multiple instances of the same app

I've been working with Django for a while now (currently on version 1.2), but just recently started working on an app that needs to support multiple instances. E.g., the project urls.py file will include it twice, under two different namespaces, like this:

urlpatterns = patterns('',
    (r'^instance1/', include('myapp.urls', namespace='instance1')),
    (r'^instance2/', include('myapp.urls', namespace='instance2')),
)

I was going along fine, until I realized I needed to figure out what to do about all the internal calls to reverse() (or the template calls to the {% url %} filter). For instance, let's say I'm doing something like the following in one of my views:

return HttpResponseRedirect(reverse('view_name'))

or something like this in one of my templates:

<a href="{% url view_name %}">link text</a>

...where view_name is the name of a URL pattern contained in myapp.urls. Since I'm using namespaces, this will raise an error: there is no view called view_name. Rather, I have to tell it either instance1:view_name or instance2:view_name. But doing this dynamically is stumping me.

I did some looking and it looks like the current_app argument, passed to either Context or RequestContext, was designed to help with this, but it's not clear at all how to dynamically pass the right application name to current_app. So what's the right way to tell Django which namespace to use?

EDIT: My use case is to use a single installation of the app multiple times. That is, it only exists on disk once, but gets included multiple times in the project's root urls.py (each time under a different namespace, as in my example above). With this in mind, is there any good way to keep track of which namespace a view/template is being called from, and make any use of reverse() or {% url %} stick within the same namespace? I know Django 1.3 will provide some extra features that could help with this (namely, the new and improved resolve()), but surely there's a good way to do this now...

like image 253
mjjohnson Avatar asked Nov 20 '10 01:11

mjjohnson


4 Answers

A lot has changed since the question was posted but for future googlers (like myself) it might be useful to point out that request has the namespace now (at least since 1.7 as shown in this example.

From what I understood we should be able to simply pass current_app positional argument to reverse/redirect but I couldn't get it to work so I ended up creating an help method for this purpose:

def __redirect(request, viewname, *args, **kwargs):
    to = viewname

    if callable(to):
        to = viewname.__module__ + '.' + viewname.__name__
    if request.resolver_match.namespace:
        to = '%s:%s' % (request.resolver_match.namespace, to)

    return redirect(
        to,
        *args,
        **kwargs
    )

I'm using redirect here but all the arguments are passed on to reverse, so it's the same.

like image 181
Filipe Pina Avatar answered Oct 31 '22 21:10

Filipe Pina


Not a very nice solution, but since you use the same text for your namespace and initial part of the URL path, you can extract that element from request.path (request.path.split('/')[1]) and set that as current_app in the request context, or just use it as the namespace in views.

http://docs.djangoproject.com/en/dev/topics/http/urls/#url-namespaces point 2.

You could do that e.g. in a context processor (if you want to use the namespace in a template).

For views you could write a decorator that feeds your function an extra kwarg "namespace" and use it as:

@feed_namespace
def view1(request, *args, **kwargs):
    ns = kwargs['namespace']

or just write a reverse_namespaced function with an extra param (the request) where the function gets the namespace from, and use it instead of reverse.

Of course if you do this you will always have to use a request path/namespace for this app

like image 30
Carles Barrobés Avatar answered Oct 31 '22 22:10

Carles Barrobés


There is a doc page about reversing namespaced urls.

http://docs.djangoproject.com/en/dev/topics/http/urls/#topics-http-reversing-url-namespaces

Either reverse('instance1:myapp.urls.some_view') or reverse('instance1:view_name') should work, or both :) - i've never tried this myself.

like image 2
Evgeny Avatar answered Oct 31 '22 21:10

Evgeny


The current_app variable is something you have to set yourself to something you like.

Personally I'd recommend setting it to something like __name__.rsplit('.', 1)[0] so you get spam in spam/views.py.

But you can define it to be anything you like, as long as your app name is consistent with what you define in your urls file.

like image 2
Wolph Avatar answered Oct 31 '22 22:10

Wolph