Managing Django Projects on Your Server with Style
This article is outdated, you’ll find an updated article here.
A friend is in the process of setting up a new slice at everbody’s favorite VPS provider Slicehost and asked for some advice. I use MySQL, Nginx & Apache/mod_wsgi on Ubuntu to serve Django. I think Slicehost’s Articles are awesome and a great place to get help you get everything installed and running, so for this article, I’ll focus on some personal preferences regarding directory structure and repo management.
One of the biggest difficulties I’ve had to tackle lately is the rapid progress of Django. I have a number of client sites hosted on my slice and have had to freeze them at certain times when backwards incompatible changes come along. My original approach made this really messy, because I only accounted for having one copy of any given third party repo, symlinked from /usr/local/src to my Python site-packages directory. My new approach takes advantage of being able to set your PYTHONPATH at runtime.
My directory structure looks like this:
|-django |--projects |---domain1.com |---domain2.com |---domain3.com
Each site has the following structure:
|-domain1.com |--apache |---django.wsgi |--django |--domain1_project |---__init__.py |---app1 |---app2 |---manage.py |---settings.py |---static |---templates |---urls.py |--third_party_repo1 |--third_party_repo2
Just drop domain1.com in your PYTHONPATH and you’re good to go. I usually start with django and the third party repos as symlinks to a current checkout of the code in /usr/local/src. When the project is completed or if the linked code passes us by, I simply replace the symlink with an actual copy of the repo. This gives me a nice compartmentalized folder of the project that would be easy to move if needed. I can also update my repos in /usr/local/src without worrying about which project might break from the new code.
Have a better way to do things? Leave a comment and let me know. Thanks for reading and tune in next week to really geek out with Nginx & Apache/mod_wsgi configuration files.
Comments
Got something to say?
This was written on August, 28 2008 and is filed in django.
Our Products
Categories
- SEO
- accessiblity
- code
- company news
- django
- gondola
- portfolio
- presentation
- screencast
- software
- subversion
- trailmapping
- wordpress
Archives
- December, 2008
- November, 2008
- September, 2008
- August, 2008
- July, 2008
- June, 2008
- May, 2008
- April, 2008
Elsewhere
What we’ve been up to online
-
Aaron Mentele, Charisma 18 ยป Turns
Electric Pulp's experience of the last economic downturn
Pete, 0 minutes ago -
pushed to master at lincolnloop/django-beancounter
Pete, 1 hour, 54 minutes ago -
committed to lincolnloop/django-beancounter
Pete, 1 hour, 54 minutes ago -
committed to lincolnloop/django-beancounter
Pete, 2 hours, 33 minutes ago -
pushed to master at lincolnloop/django-beancounter
Pete, 2 hours, 33 minutes ago -
committed to lincolnloop/django-beancounter
Pete, 2 hours, 33 minutes ago -
pushed to master at lincolnloop/django-beancounter
Pete, 2 hours, 41 minutes ago -
committed to lincolnloop/django-beancounter
Pete, 2 hours, 41 minutes ago -
committed to lincolnloop/django-beancounter
Pete, 1 day, 1 hour ago -
committed to lincolnloop/django-beancounter
Pete, 1 day, 1 hour ago -
committed to lincolnloop/django-beancounter
Pete, 1 day, 1 hour ago -
committed to lincolnloop/django-beancounter
Pete, 1 day, 1 hour ago -
committed to lincolnloop/django-beancounter
Pete, 1 day, 1 hour ago -
pushed to master at lincolnloop/django-beancounter
Pete, 1 day, 1 hour ago -
applied fork commits to lincoln-loop-deploy/master
Pete, 1 week, 5 days ago


That’s typically the point where you want to use virtualenv :)
http://pypi.python.org/pypi/virtualenv
And for mod_wsgi integration: http://code.google.com/p/modwsgi/wiki/VirtualEnvironments
One feature of virtualenv I always like to exploit: easy_install in a virtualenv looks into your PYTHONPATH to find already installed packages in it and just links to them (so no duplication). Using the multiversion flag for easy_install you could even choose between different version and share them between projects.
I wouldn’t mix python modules and non modules on the Python path. And I wouldn’t put my apps into a project module
/web/domain1.com /manage.py /static /templates /python/django /python/project1 /python/project1/settings.py /python/project1/urls.py /python/app1 /python/app2 /python/third_party_repo1 /python/third_party_repo2Put /web/domain1.com/python/ on the Python path and modify manage.py to import project1.settings instead of just settings.
I’m also a fan of virtualenv and I use Doug Hellman’s [http://www.doughellmann.com/projects/virtualenvwrapper/ virtualenvwrapper]. I’m still experimenting with finding the killer combo (for my tastes) for managing and versioning django apps and projects. I can envision a lot more discussion about distribution and deployment in the near future as projects like Pinax gain maturity and devs/admins are finding themselves dealing with large numbers of dependencies. I don’t know about you but I get pretty damned frustrated when I spend time tracking down a weird “bug” to discover it was the result of an out-of-date package somewhere higher on the path list.
I’m also excited about [http://www.blueskyonmars.com/projects/paver/index.html Paver] since it integrates so tightly with the existing tools (distutils and setuptools) plus has support for virtualenv. It looks like it will make the creation of virtualenv bootstrap scripts a little more painless.