-
-
Notifications
You must be signed in to change notification settings - Fork 624
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Default tag ordering to the primary key #892
Merged
Merged
Changes from 20 commits
Commits
Show all changes
22 commits
Select commit
Hold shift + click to select a range
f5ba4b5
add TestTaggableManager
steverecio d96a216
fix _to_tag_model_instances to preserve order
steverecio e0c8858
ensure tag order is preserved
steverecio 0595a42
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] 5820930
missing arg
steverecio 122112b
clean up tests
steverecio 9697a97
fix conditional on ordering arg
steverecio 897ed6a
make test more explicit
steverecio cb7d00c
fix case insensitive handling and tag_kwargs
steverecio 0a8378b
fix processing logic
steverecio b297551
fixed duplicates issue
steverecio 311c1e4
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] 82922eb
fix most_common test
steverecio 8bfa7ed
a more elegant solution of imposing through table ordering only if fi…
steverecio acff592
unused import
steverecio b23715b
Make sure to only query as much as needed
rtpg e203e07
Add changelog entry
rtpg 5ba5934
Add note to tricky code
rtpg 855def6
fix style issues
rtpg 15d83bc
ignore .direnv for flake8
rtpg 1e70249
Add Steve to the AUTHORS
rtpg e4baf49
remove continue usage
rtpg File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -65,8 +65,17 @@ def __init__(self, through, model, instance, prefetch_cache_name, ordering=None) | |
self.model = model | ||
self.instance = instance | ||
self.prefetch_cache_name = prefetch_cache_name | ||
if ordering: | ||
|
||
if ordering is not None: | ||
self.ordering = ordering | ||
elif instance: | ||
# When working off an instance (i.e. instance.tags.all()) | ||
# we default to ordering by the PK of the through table | ||
# | ||
# (This ordering does not apply for queries at the model | ||
# level, i.e. Model.tags.all()) | ||
related_name = self.through.tag.field.related_query_name() | ||
self.ordering = [f"{related_name}__pk"] | ||
else: | ||
self.ordering = [] | ||
|
||
|
@@ -206,59 +215,82 @@ def add(self, *tags, through_defaults=None, tag_kwargs=None, **kwargs): | |
def _to_tag_model_instances(self, tags, tag_kwargs): | ||
""" | ||
Takes an iterable containing either strings, tag objects, or a mixture | ||
of both and returns set of tag objects. | ||
of both and returns a list of tag objects while preserving order. | ||
""" | ||
db = router.db_for_write(self.through, instance=self.instance) | ||
|
||
str_tags = set() | ||
tag_objs = set() | ||
|
||
for t in tags: | ||
if isinstance(t, self.through.tag_model()): | ||
tag_objs.add(t) | ||
elif isinstance(t, str): | ||
str_tags.add(t) | ||
else: | ||
raise ValueError( | ||
"Cannot add {} ({}). Expected {} or str.".format( | ||
t, type(t), type(self.through.tag_model()) | ||
) | ||
) | ||
|
||
case_insensitive = getattr(settings, "TAGGIT_CASE_INSENSITIVE", False) | ||
manager = self.through.tag_model()._default_manager.using(db) | ||
|
||
# tags can be instances of our through models, or strings | ||
|
||
tag_strs = [tag for tag in tags if isinstance(tag, str)] | ||
# This map from tag names to tags lets us handle deduplication | ||
# without doing extra queries along the way, all while relying on | ||
# data we were going to pull out of the database anyways | ||
# existing_tags_for_str[tag_name] = tag | ||
existing_tags_for_str = {} | ||
# we are going to first try and lookup existing tags (in a single query) | ||
if case_insensitive: | ||
# Some databases can do case-insensitive comparison with IN, which | ||
# would be faster, but we can't rely on it or easily detect it. | ||
# would be faster, but we can't rely on it or easily detect it | ||
existing = [] | ||
tags_to_create = [] | ||
|
||
for name in str_tags: | ||
for name in tag_strs: | ||
try: | ||
tag = manager.get(name__iexact=name, **tag_kwargs) | ||
existing.append(tag) | ||
existing_tags_for_str[name] = tag | ||
except self.through.tag_model().DoesNotExist: | ||
tags_to_create.append(name) | ||
else: | ||
# If str_tags has 0 elements Django actually optimizes that to not | ||
# do a query. Malcolm is very smart. | ||
existing = manager.filter(name__in=str_tags, **tag_kwargs) | ||
|
||
tags_to_create = str_tags - {t.name for t in existing} | ||
# Django is smart enough to not actually query if tag_strs is empty | ||
# but importantly, this is a single query for all potential tags | ||
existing = manager.filter(name__in=tag_strs, **tag_kwargs) | ||
# we're going to end up doing this query anyways, so here is fine | ||
for t in existing: | ||
existing_tags_for_str[t.name] = t | ||
|
||
result = [] | ||
# this set is used for deduplicating tags | ||
seen_tags = set() | ||
for t in tags: | ||
if isinstance(t, self.through.tag_model()): | ||
if t in seen_tags: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nitpicky but could be simpler here to do:
|
||
continue | ||
seen_tags.add(t) | ||
result.append(t) | ||
elif isinstance(t, str): | ||
# we are using a string, so either the tag exists (and we have the lookup) | ||
# or we need to create the value | ||
|
||
existing_tag = existing_tags_for_str.get(t, None) | ||
if existing_tag is None: | ||
# we need to create a tag | ||
# (we use get_or_create to handle potential races) | ||
if case_insensitive: | ||
lookup = {"name__iexact": t, **tag_kwargs} | ||
else: | ||
lookup = {"name": t, **tag_kwargs} | ||
existing_tag, _ = manager.get_or_create( | ||
**lookup, defaults={"name": t} | ||
) | ||
# we now have an existing tag for this string | ||
|
||
tag_objs.update(existing) | ||
# confirm if we've seen it or not (this is where case insensitivity comes | ||
# into play) | ||
if existing_tag in seen_tags: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. likewise we can eliminate the continue here |
||
continue | ||
|
||
for new_tag in tags_to_create: | ||
if case_insensitive: | ||
lookup = {"name__iexact": new_tag, **tag_kwargs} | ||
seen_tags.add(existing_tag) | ||
result.append(existing_tag) | ||
else: | ||
lookup = {"name": new_tag, **tag_kwargs} | ||
|
||
tag, create = manager.get_or_create(**lookup, defaults={"name": new_tag}) | ||
tag_objs.add(tag) | ||
raise ValueError( | ||
"Cannot add {} ({}). Expected {} or str.".format( | ||
t, type(t), type(self.through.tag_model()) | ||
) | ||
) | ||
|
||
return tag_objs | ||
return result | ||
|
||
@require_instance_manager | ||
def names(self): | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
been meaning to add this for a while....