Jonathan, Thank you! I've tried the code you posted and it works like a charm. It really has helped me to understand how to do it. I have modified it to include also a calculation for how many teas have in common Peter and the people with at least one of the same teas as Peter. This way I can then apply some "weight" to the recommendations, so depending on how "similar" peter's and person X's tastes are (more/less teas in common), the concrete recommendation of person X is more or less important and I can add a higher or lower value to that person's recommendations.
This is the code right now: ==================== peter = Person.objects.get(name='Peter') peter_likes_ids = peter.likes.values_list('id', flat=True) # People who like at least one of the same teas as Peter, and how much of teas in common do they have (like_same_count) similarities = Person.objects.filter(likes__in=peter_likes_ids).exclude(id=peter.id).extra( select={ 'like_same_count': 'COUNT(%s.kindoftea_id)' % Person._meta.get_field('likes').m2m_db_table(), }, order_by=['id', 'name'] ) people_table = Person._meta.db_table similarities.query.group_by.extend(['%s.id' % people_table, '%s.name' % people_table]) similarities.values_list('id','name','like_same_count') # just the ids of the people who like at least one tea of the same teas as Peter people = similarities.values_list('id', flat=True) # Teas liked by the above people, excluding the teas Peter likes recommendations = KindOfTea.objects.filter( liked_by__in=people).exclude(id__in=peter_likes_ids).extra( select={ 'like_count': 'COUNT(%s.kindoftea_id)' % Person._meta.get_field('likes').m2m_db_table(), }, order_by=['-like_count', 'name'] ) tea_table = KindOfTea._meta.db_table recommendations.query.group_by.extend(['%s.id' % tea_table, '%s.name' % tea_table]) # 3-tuples of (tea id, tea name, relevant people who like it count) recommendations.values_list('id', 'name', 'like_count') I have tried to order similarities by '-like_same_count' but it always gives an error: "Cannot resolve keyword 'like_same_count' into field. Choices are: id, likes, name" . I don't know why this is happening because I have set like_same_count as an extra parameter in the select statement for similarities. I'll try to add the weight calculation and post the solution here. Regards. On 10 nov, 02:43, Jonathan Buchanan <[EMAIL PROTECTED]> wrote: > Mr.Zwrote: > > I have these models: > > > KindOfTea(models.Model): > > name = models.CharField(max_length=100, unique=True) > > description = models.TextField() > > > Person(models.Model): > > name = models.CharField(max_length=100, unique=True) > > likes = models.ManyToManyField(KindOfTea, related_name='liked_by') > > > I have 1 Person object and some related KindOfTea objects he likes. > > With this information I want to know which other new kinds of tea he > > would like. What I want to do is to get the other Person objects that > > like the same kinds of tea as this person and obtain all the other > > kinds of tea they like ordered by how many of them like the same one. > > > For example: > > ========= > > Peter likes black tea and white tea > > Laura likes black tea, green tea and red tea > > Paul likes black tea, green tea and red tea > > John likes black tea, green tea and yellow tea > > > To recommend Peter a new kind of tea I get the other persons that like > > "black tea" and/or "white tea": Laura, Paul and John. Then I get > > somehow the kinds of tea they like that are different than 'black tea' > > and 'white tea' (the kinds of tea Peter likes) and for them I store > > how many of those persons like them. So somehow I get a dictionary > > that looks like this: > > other_people_like = { 'green tea' : 3, 'red tea' : 2, 'yellow tea': > > 1 } And I can recommend Peter those kinds of tea in that order. > > > I have started with this: > > ================= > > from sets import Set > > p = Person.objects.get(name='Peter') > > > other_people = Set([]) > > > for tea in p.likes.all(): > > for person in tea.liked_by.exclude(name='Peter') > > other_people.add(person) > > > # now I have a Set with the people that like some or at least one kind > > of tea Peter likes too. now how should I do the rest? > > > Thank you. > > Here's one approach which seems to work for the sample data and expected > output you provided (tried it in SQLite, MySQL 4.1 and PostgreSQL 8.1): > > peter = Person.objects.get(name='Peter') > peter_likes_ids = peter.likes.values_list('id', flat=True) > > # People who like at least one of the same teas as Peter > people = Person.objects.filter( > likes__in=peter_likes_ids).exclude(id=peter.id).values('id') > > # Teas liked by the above people, excluding the teas Peter likes > recommendations = KindOfTea.objects.filter( > liked_by__in=people.query).exclude(id__in=peter_likes_ids).extra( > select={ > 'like_count': 'COUNT(%s.kindoftea_id)' % \ > Person._meta.get_field('likes').m2m_db_table(), > }, > order_by=['-like_count', 'name'] > ) > tea_table = KindOfTea._meta.db_table > recommendations.query.group_by.extend(['%s.id' % tea_table, > '%s.name' % tea_table]) > > # 3-tuples of (tea id, tea name, relevant people who like it count) > recommendations.values_list('id', 'name', 'like_count') > > Note that the use of group_by to insert grouping criteria isn't yet part > of the public API [1]. The resulting recommendations QuerySet will > generate a query something like the following (this is some initial > target SQL I wrote to work against, not exactly what the Django ORM will > produce): > > SELECT > app_kindoftea.id, > app_kindoftea.name, > COUNT(app_person_likes.kindoftea_id) AS like_count > FROM > app_person_likes > INNER JOIN app_kindoftea > ON app_person_likes.kindoftea_id = app_kindoftea.id > WHERE app_person_likes.person_id IN > ( > SELECT > app_person_likes.person_id > FROM > app_person_likes > INNER JOIN app_kindoftea > ON app_person_likes.kindoftea_id = app_kindoftea.id > WHERE app_kindoftea.id IN (1, 2) > AND app_person_likes.person_id != 1 > ) > AND app_kindoftea.id NOT IN (1, 2) > GROUP BY app_kindoftea.id, app_kindoftea.name > ORDER BY like_count DESC, app_kindoftea.name ASC > > Regards, > Jonathan. > > [1]http://www.eflorenzano.com/blog/post/secrets-django-orm/ --~--~---------~--~----~------------~-------~--~----~ You received this message because you are subscribed to the Google Groups "Django users" group. To post to this group, send email to django-users@googlegroups.com To unsubscribe from this group, send email to [EMAIL PROTECTED] For more options, visit this group at http://groups.google.com/group/django-users?hl=en -~----------~----~----~----~------~----~------~--~---