I have encountered a problem with joins in Django's ORM. I had assumed
that

queryset.filter(**a) | queryset.filter(**b)

is functionally equivalent to

queryset.filter(models.Q(**a) | models.Q(**b))

for all a and b. That is not what is happening. If my assumption is
not true, then what are the intended semantics for the first form? If
my assumption is correct, then this must be a bug in Django.

The code I'm using, and the generated SQL from ._as_sql() is below.

Tags are many-to-many with Projects, Projects are many-to-many with
Profiles in two separate maps: teammembers and investigators. Profiles
are also many-to-many with Tags in two separate maps, an include
(profiles_watching) and an exclude (profiles_ignoring) map. The query
is to find Tags linked via Projects to a Profile, plus those that are
directly included and not those directly excluded. 'self' is a
models.Manager for research.Tag.

The first form is the original version that omits some of the results
I'm expecting to see.

of_interest_automatic = (
    self.filter(project__teammembers=profile) |
    self.filter(project__investigators=profile)
).distinct()
return (
    of_interest_automatic |
    self.filter(profiles_watching=profile).distinct()
).exclude(profiles_ignoring=profile).distinct()

(of_interest_automatic is actually a separate method, which is why
distinct() is repeated)

('SELECT DISTINCT U0.`id` FROM `research_tag` U0 INNER JOIN
`research_project_tags` U1 ON (U0.`id` = U1.`tag_id`) INNER JOIN
`research_project` U2 ON (U1.`project_id` = U2.`id`) LEFT OUTER JOIN
`research_project_teammembers` U3 ON (U2.`id` = U3.`project_id`) LEFT
OUTER JOIN `research_project_investigators` U5 ON (U2.`id` =
U5.`project_id`) LEFT OUTER JOIN `people_profile_tags_include` U6 ON
(U0.`id` = U6.`tag_id`) WHERE ((U3.`profile_id` = %s  OR
U5.`profile_id` = %s  OR U6.`profile_id` = %s ) AND NOT (U0.`id` IN
(SELECT U1.`tag_id` FROM `people_profile_tags_exclude` U1 WHERE
U1.`profile_id` = %s )))', (287, 287, 287, 287))

This is how I have rewritten it

return self.filter(
    Q(project__teammembers=profile) |
    Q(project__investigators=profile) |
    Q(profiles_watching=profile)
).exclude(profiles_ignoring=profile).distinct()

('SELECT DISTINCT U0.`id` FROM `research_tag` U0 LEFT OUTER JOIN
`research_project_tags` U1 ON (U0.`id` = U1.`tag_id`) LEFT OUTER JOIN
`research_project` U2 ON (U1.`project_id` = U2.`id`) LEFT OUTER JOIN
`research_project_teammembers` U3 ON (U2.`id` = U3.`project_id`) LEFT
OUTER JOIN `research_project_investigators` U5 ON (U2.`id` =
U5.`project_id`) LEFT OUTER JOIN `people_profile_tags_include` U7 ON
(U0.`id` = U7.`tag_id`) WHERE ((U3.`profile_id` = %s  OR
U5.`profile_id` = %s  OR U7.`profile_id` = %s ) AND NOT (U0.`id` IN
(SELECT U1.`tag_id` FROM `people_profile_tags_exclude` U1 WHERE
U1.`profile_id` = %s )))', (287, 287, 287, 287))

Dan
-- 
You received this message because you are subscribed to the Google Groups 
"Django users" group.
To post to this group, send email to django-us...@googlegroups.com.
To unsubscribe from this group, send email to 
django-users+unsubscr...@googlegroups.com.
For more options, visit this group at 
http://groups.google.com/group/django-users?hl=en.


Reply via email to