27👍
Since I was looking for some more in-depth answers, I decided to research the issue myself in depth. Please let me know if I’ve misunderstood anything.
Some general recommendation are to use a separate webserver for handling media. By separate, I mean a webserver which is not running Django. This server can be for instance:
- Lighttpd (Lighty)
- Nginx (EngineX)
- Or some other light-weight server
Then, for Django, you can go down different paths. You can either:
-
Serve Django via Apache and:
-
mod_python
This is the stable and recommended/well documented way. Cons: uses a lot of memory.
-
mod_wsgi
From what I understand, mod_wsgi is a newer alternative. It appears to be faster and easier on resources.
-
mod_fastcgi
When using FastCGI you are delegating the serving of Django to another process. Since mod_python includes a python interpreter in every request it uses a lot of memory. This is a way to bypass that problem. Also there is some security concerns.
What you do is that you start your Django FastCGI server in a separate process and then configures apache via rewrites to call this process when needed.
-
Or you can:
-
Serve Django without using Apache but with another server that supports FastCGI natively:
(The documentation mentions that you can do this if you don’t have any Apache specific needs. I guess the reason must be to save memory.)
- Lighttpd
This is the server that runs Youtube. It seems fast and easy to use, however i’ve seen reports on memoryleaks.
- nginx
I’ve seen benchmarks claiming that this server is even faster than lighttpd. It’s mostly documented in Russian though.
Another thing, due to limitations in Python your server should be running in forked mode, not threaded.
So this is my current research, but I want more opinions and experiences.
9👍
I’m using Cherokee.
According to their benchmarks (grain of salt with them), it handles load better than both Lighttpd and nginx… But that’s not why I use it.
I use it because if you type cherokee-admin
, it starts a new server that you can log into (with a one-time password) and configure the whole server through a beautifully-done webmin. That’s a killer feature. It has already saved me a lot of time. And it’s saving my server a lot of resources!
As for django, I’m running it as a threaded SCGI process. Works well. Cherokee can keep it running too. Again, very nice feature.
The current Ubuntu repo version is very old so I’d advise you use their PPA. Good luck.
- [Django]-Changing a project name in django
- [Django]-Django bulk_create with ignore rows that cause IntegrityError?
- [Django]-Django load block for css
6👍
As @Barry said, the documentation uses mod_python. I haven’t used Ubuntu as a server, but had a good experience with mod_wsgi on Solaris. You can find documentation for mod_wsgi and Django on the mod_wsgi site.
A quick review of your requirements:
- Easy to setup I’ve found apache 2.2 fairly easy to build and install.
- Fast and easy on resources I would say that this depends on your usage and traffic. * You may not want to server all files using Apache and use LightTPD (lighty) to server static files.
- Can serve media files I assume you mean images, flash files? Apache can do this.
- Multiple sites on same server Virtual server hosting on Apache.
- Rather not install other extensions Comment out anything you don’t want in the Apache config.
- [Django]-Django select only rows with duplicate field values
- [Django]-React Proxy error: Could not proxy request /api/ from localhost:3000 to http://localhost:8000 (ECONNREFUSED)
- [Django]-Are Django SECRET_KEY's per instance or per app?
5👍
The officially recommended way to deploy a django project is to use mod_python with apache. This is described in the documentation. The main pro with this is that it is the best documented, most supported, and most common way to deploy. The con is that it probably isn’t the fastest.
- [Django]-How does Django Know the Order to Render Form Fields?
- [Django]-How to get the username of the logged-in user in Django?
- [Django]-Programmatically create a django group with permissions
3👍
The best configuration is not so known I think. But here is:
- Use nginx for serving requests (dynamic to app, static content directly).
- Use python web server for serving dynamic content.
Two most speedy solutions for python-based web server is:
You need to look into google to find current best configuration for django (still in development).
- [Django]-How does Django Know the Order to Render Form Fields?
- [Django]-Nginx and supervisor setup in Ubuntu
- [Django]-SocketException: OS Error: Connection refused, errno = 111 in flutter using django backend
2👍
I’m using nginx (0.6.32 taken from Sid) with mod_wsgi. It works very well, though I can’t say whether it’s better than the alternatives because I never tried any. Nginx has memcached support built in, which can perhaps interoperate with the Django caching middleware (I don’t actually use it, instead I fill the cache manually using python-memcache and invalidate it when changes are made), so cache hits completely bypass Django (my development machine can serve about 3000 requests per second).
A caveat: nginx’ mod_wsgi
highly dislikes named locations (it tries to pass them in SCRIPT_NAME
), so the obvious ‘error_page 404 = @django
’ will cause numerous obscure errors. I had to patch mod_wsgi source to fix that.
- [Django]-Foreign Key Django Model
- [Django]-How can I disable Django's admin in a deployed project, but keep it for local development?
- [Django]-Django connection to postgres by docker-compose
2👍
I’m struggling to understand all the options as well. In this blog post I found some benefits of mod_wsgi compared to mod_python explained.
Multiple low-traffic sites on a small VPS make RAM consumption the primary concern, and mod_python seems like a bad option there. Using lighttpd and FastCGI, I’ve managed to get the minimum memory usage of a simple Django site down to 58MiB virtual and 6.5MiB resident (after restarting and serving a single non-RAM-heavy request).
I’ve noticed that upgrading from Python 2.4 to 2.5 on Debian Etch increased the minimum memory footprint of the Python processes by a few percent. On the other hand, 2.5’s better memory management might have a bigger opposite effect on long-running processes.
- [Django]-Django admin: how to sort by one of the custom list_display fields that has no database field
- [Django]-Django REST Framework – Serializing optional fields
- [Django]-Dynamic File Path in Django
2👍
Keep it simple: Django recommends Apache and mod_wsgi (or mod_python). If serving media files is a very big part of your service, consider Amazon S3 or Rackspace CloudFiles.
- [Django]-How to loop over form field choices and display associated model instance fields
- [Django]-List field in model?
- [Django]-Django Get All Users
2👍
There are many ways, approach to do this.For that reason, I recommend to read carefully the article related to the deployment process on DjangoAdvent.com:
Eric Florenzano – Deploying Django with FastCGI: http://djangoadvent.com/1.2/deploying-django-site-using-fastcgi/
Read too:
Mike Malone – Scaling Django
Stochastictechnologies Blog: The perfect Django Setup
Mikkel Hoegh Blog: 35 % Response-time-improvement-switching-uwsgi-nginx
Regards
- [Django]-How do I add a custom column with a hyperlink in the django admin interface?
- [Django]-Django MEDIA_URL and MEDIA_ROOT
- [Django]-Django Rest Framework – APIView Pagination
2👍
In my opinion best/fastest stack is varnish-nginx-uwsgi-django.
And I’m successfully using it.
- [Django]-How to annotate Count with a condition in a Django queryset
- [Django]-Sending images using Http Post
- [Django]-Accessing function as attribute in a Python class
1👍
If you’re using lighthttpd, you can also use FastCGI for serving Django. I’m not sure how the speed compares to mod_wsgi, but if memory serves correctly, you get a couple of the benefits that you would get with mod_wsgi that you wouldn’t get with mod_python. The main one being that you can give each application its own process (which is really helpful for keeping memory of different apps separated as well as for taking advantage of multi-core computers.
Edit: Just to add in regards to your update about nginix, if memory serves correctly again, nginix uses “greenlets” to handle concurrency. This means that you may need to be a little bit more careful to make sure that one app doesn’t eat up all the server’s time.
- [Django]-Django or Django Rest Framework
- [Django]-Get Timezone from City in Python/Django
- [Django]-Count number of records by date in Django
1👍
We use nginx and FastCGI for all of our Django deployments. This is mostly because we usually deploy over at Slicehost, and don’t want to donate all of our memory to Apache. I guess this would be our “use case”.
As for the remarks about the documentation being mostly in Russian — I’ve found most of the information on the English wiki to be very useful and accurate. This site has sample configurations for Django too, from which you can tweak your own nginx configuration.
- [Django]-Django Model Field Default Based Off Another Field in Same Model
- [Django]-How do I POST with jQuery/Ajax in Django?
- [Django]-How to PATCH a single field using Django Rest Framework?
1👍
I have a warning for using Cherokee. When you make changes to Django Cherokee maintains the OLD process, instead of killing it and starting a new one.
On Apache i strongly recommend this article.
http://www.djangofoo.com/17/django-mod_wsgi-deploy-exampl
Its easy to setup, easy to kill or reset after making changes.
Just type in terminal
sudo /etc/init.d/apache2 restart
and changes are seen instantly.
- [Django]-Django models.py Circular Foreign Key
- [Django]-Django aggregate or annotate
- [Django]-Pycharm: set environment variable for run manage.py Task