
django-rangevoting
django-rangevoting
django-rangevoting allows flexible voting on django objects and includes a pluggable ranking backend to make writing custom ranking algorithms easy.
This project is 100% "under development", so watch your fingers around the rough edges.
Before you ask "why not just fork django-voting
", let me say:
- I wanted to use an aggregate table
django-voting
probably could've been extended fairly easily, but I'd prefer to start from scratch.
Setup
Add rangevoting
to INSTALLED_APPS
:
INSTALLED_APPS = (
# other apps ...
'rangevoting',
)
Sync up:
python manage.py syncdb
Included w/ the svn checkout is a test app called testapp
.
If you checked out the entire project and have testapp
, you can
run tests with:
python manage.py test rangevoting
Usage
Add the post-save
signal to each model you'd like users to vote on -- this signal
will create an entry in the VoteSummary
table, an aggregate table used to store
vote tallies and other information about each object's voting history.
``` from django.db.models.signals import post_save from rangevoting.signals import create_voting_summary
class MyModel(models.Model): ...
post_save.connect(create_voting_summary, sender=MyModel) ```
Cast a Vote on an object:
``` from django.contrib.auth.models import from rangevoting.models import Vote from myapp.models import MyModel
get a user
user1 = User.objects.get(...) user2 = User.objects.get(...)
get an object to vote on
obj = MyModel.objects.get(...)
cast the votes
Vote.objects.cast_vote(user=user1, obj=obj, score=10.0) Vote.objects.cast_vote(user=user2, obj=obj, score=6.5)
get the score for an object
VoteSummary.objects.get_obj_score(obj=obj)
{"total": 2, "score": Decimal("8.250")}
```
Ranking
django-rangevoting
comes w/ a ranking engine that lets
you hack together ranking algorithms by providing, at this time,
templated-out raw SQL. This area of the application will probably
come under considerable re-writing, but works at the moment. I'd consider
its implementation somewhere between "proof of concept" and "working draft",
but it works. And its kinda cool :)
Out-of-the-box, I've provided the following Rankers
:
- Average - ranks objects by their avg score
(vote sum / vote count)
in descending fashion. Bayesian - ranks with weighted consideration of actual worth: ``` If there is only few votes, then these votes should count less than when there are many votes and we can trust that this is how the public feels about it. In other texts this value is also refered to as “certainty” or “believability”.
- best description i've read, from http://www.thebroth.com/blog/118/bayesian-rating ```
- Hacker News (modified) - a slightly modified (see docstring) version of the Hacker News algorithm.
To rank with a Ranker
:
``` from rangevoting.rankers import AvgRanker
rank with the AvgRanker
ranked = VoteSummary.objects.rank(model=MyModel, ranker=AvgRanker) ```
ranked
receives a list of ranked MyModel
objects:
[<MyModel: ...>, <MyModel: ...>, ...]
As mentioned before, the ranking backend is still under construction. I'd love some input on it.
Note: The Hacker News algorithm relies on a MySQL function called 'unix_timestamp'. I know introduces incompatibilities -- in the future there'll be a work-around, but not right this second. In the meantime, subclass Ranker
and have at your own version.
Status/Notes
This application was most definitely "released early". I provided a JSON view to store a Vote, but its definitely more perfunctory than anything else. Until otherwise noted in giant bold letters, consider the views something to write yourself.