mirror of
https://github.com/evennia/evennia.git
synced 2026-03-27 02:06:32 +01:00
get_by_tag manager can now query for multiple tag/category combinations
This commit is contained in:
parent
0f6cec5d9b
commit
7bd8fe0a34
2 changed files with 69 additions and 55 deletions
|
|
@ -221,29 +221,6 @@ class TypedObjectManager(idmapper.manager.SharedMemoryManager):
|
|||
"""
|
||||
return self.get_tag(key=key, category=category, obj=obj, tagtype="alias")
|
||||
|
||||
# @returns_typeclass_list
|
||||
# def get_by_tag(self, key=None, category=None, tagtype=None):
|
||||
# """
|
||||
# Return objects having tags with a given key or category or
|
||||
# combination of the two.
|
||||
#
|
||||
# Args:
|
||||
# key (str, optional): Tag key. Not case sensitive.
|
||||
# category (str, optional): Tag category. Not case sensitive.
|
||||
# tagtype (str or None, optional): 'type' of Tag, by default
|
||||
# this is either `None` (a normal Tag), `alias` or
|
||||
# `permission`.
|
||||
# Returns:
|
||||
# objects (list): Objects with matching tag.
|
||||
# """
|
||||
# dbmodel = self.model.__dbclass__.__name__.lower()
|
||||
# query = [("db_tags__db_tagtype", tagtype), ("db_tags__db_model", dbmodel)]
|
||||
# if key:
|
||||
# query.append(("db_tags__db_key", key.lower()))
|
||||
# if category:
|
||||
# query.append(("db_tags__db_category", category.lower()))
|
||||
# return self.filter(**dict(query))
|
||||
|
||||
def get_by_tag(self, key=None, category=None, tagtype=None):
|
||||
"""
|
||||
Return objects having tags with a given key or category or combination of the two.
|
||||
|
|
@ -253,7 +230,8 @@ class TypedObjectManager(idmapper.manager.SharedMemoryManager):
|
|||
key (str or list, optional): Tag key or list of keys. Not case sensitive.
|
||||
category (str or list, optional): Tag category. Not case sensitive. If `key` is
|
||||
a list, a single category can either apply to all keys in that list or this
|
||||
must be a list matching the `key` list element by element.
|
||||
must be a list matching the `key` list element by element. If no `key` is given,
|
||||
all objects with tags of this category are returned.
|
||||
tagtype (str, optional): 'type' of Tag, by default
|
||||
this is either `None` (a normal Tag), `alias` or
|
||||
`permission`. This always apply to all queried tags.
|
||||
|
|
@ -266,39 +244,37 @@ class TypedObjectManager(idmapper.manager.SharedMemoryManager):
|
|||
than `key`.
|
||||
|
||||
"""
|
||||
keys = make_iter(key)
|
||||
categories = make_iter(category)
|
||||
if not (key or category):
|
||||
return []
|
||||
|
||||
keys = make_iter(key) if key else []
|
||||
categories = make_iter(category) if category else []
|
||||
n_keys = len(keys)
|
||||
n_categories = len(categories)
|
||||
|
||||
dbmodel = self.model.__dbclass__.__name__.lower()
|
||||
if n_keys > 1:
|
||||
if n_categories == 1:
|
||||
category = categories[0]
|
||||
query = Q()
|
||||
for key in keys:
|
||||
query = query & \
|
||||
Q(db_tags__db_tagtype=tagtype.lower() if tagtype else tagtype,
|
||||
db_tags__db_category=category.lower() if category else category,
|
||||
db_tags__db_model=dbmodel,
|
||||
db_tags__db_key=key.lower())
|
||||
print "Query:", query
|
||||
else:
|
||||
query = Q(db_tags__db_tagtype=tagtype.lower(),
|
||||
db_tags__db_model=dbmodel)
|
||||
for ikey, key in keys:
|
||||
category = categories[ikey]
|
||||
category = category.lower() if category else category
|
||||
query = query & Q(db_tags__db_key=key.lower(),
|
||||
db_tags__db_category=category)
|
||||
return self.filter(query)
|
||||
query = self.filter(db_tags__db_tagtype__iexact=tagtype,
|
||||
db_tags__db_model__iexact=dbmodel).distinct()
|
||||
|
||||
if n_keys > 0:
|
||||
# keys and/or categories given
|
||||
if n_categories == 0:
|
||||
categories = [None for _ in range(n_keys)]
|
||||
elif n_categories == 1 and n_keys > 1:
|
||||
cat = categories[0]
|
||||
categories = [cat for _ in range(n_keys)]
|
||||
elif 1 < n_categories < n_keys:
|
||||
raise IndexError("get_by_tag needs a single category or a list of categories "
|
||||
"the same length as the list of tags.")
|
||||
for ikey, key in enumerate(keys):
|
||||
query = query.filter(db_tags__db_key__iexact=key,
|
||||
db_tags__db_category__iexact=categories[ikey])
|
||||
else:
|
||||
query = [("db_tags__db_tagtype", tagtype), ("db_tags__db_model", dbmodel)]
|
||||
if key:
|
||||
query.append(("db_tags__db_key", keys[0].lower()))
|
||||
if category:
|
||||
query.append(("db_tags__db_category", categories[0].lower()))
|
||||
return self.filter(**dict(query))
|
||||
# only one or more categories given
|
||||
for category in categories:
|
||||
query = query.filter(db_tags__db_category__iexact=category)
|
||||
|
||||
return query
|
||||
|
||||
def get_by_permission(self, key=None, category=None):
|
||||
"""
|
||||
|
|
|
|||
|
|
@ -12,10 +12,48 @@ from evennia.utils.test_resources import EvenniaTest
|
|||
|
||||
class TestTypedObjectManager(EvenniaTest):
|
||||
def _manager(self, methodname, *args, **kwargs):
|
||||
return getattr(self.obj1.__class__.objects, methodname)(*args, **kwargs)
|
||||
return list(getattr(self.obj1.__class__.objects, methodname)(*args, **kwargs))
|
||||
|
||||
def test_get_by_tag_no_category(self):
|
||||
self.obj1.tags.add("tag1")
|
||||
self.obj1.tags.add("tag2")
|
||||
self.obj1.tags.add("tag2c")
|
||||
self.obj2.tags.add("tag2")
|
||||
self.obj2.tags.add("tag3")
|
||||
self.assertEquals(list(self._manager("get_by_tag", "tag1")), [self.obj1l])
|
||||
self.obj2.tags.add("tag2a")
|
||||
self.obj2.tags.add("tag2b")
|
||||
self.obj2.tags.add("tag3 with spaces")
|
||||
self.obj2.tags.add("tag4")
|
||||
self.obj2.tags.add("tag2c")
|
||||
self.assertEquals(self._manager("get_by_tag", "tag1"), [self.obj1])
|
||||
self.assertEquals(self._manager("get_by_tag", "tag2"), [self.obj1, self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", "tag2a"), [self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", "tag3 with spaces"), [self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", ["tag2a", "tag2b"]), [self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", ["tag2a", "tag1"]), [])
|
||||
self.assertEquals(self._manager("get_by_tag", ["tag2a", "tag4", "tag2c"]), [self.obj2])
|
||||
|
||||
def test_get_by_tag_and_category(self):
|
||||
self.obj1.tags.add("tag5", "category1")
|
||||
self.obj1.tags.add("tag6", )
|
||||
self.obj1.tags.add("tag7", "category1")
|
||||
self.obj1.tags.add("tag6", "category3")
|
||||
self.obj1.tags.add("tag7", "category4")
|
||||
self.obj2.tags.add("tag5", "category1")
|
||||
self.obj2.tags.add("tag5", "category2")
|
||||
self.obj2.tags.add("tag6", "category3")
|
||||
self.obj2.tags.add("tag7", "category1")
|
||||
self.obj2.tags.add("tag7", "category5")
|
||||
self.assertEquals(self._manager("get_by_tag", "tag5", "category1"), [self.obj1, self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", "tag6", "category1"), [])
|
||||
self.assertEquals(self._manager("get_by_tag", "tag6", "category3"), [self.obj1, self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", ["tag5", "tag6"],
|
||||
["category1", "category3"]), [self.obj1, self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", ["tag5", "tag7"],
|
||||
"category1"), [self.obj1, self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", category="category1"), [self.obj1, self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", category="category2"), [self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", category=["category1", "category3"]),
|
||||
[self.obj1, self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", category=["category1", "category2"]),
|
||||
[self.obj2])
|
||||
self.assertEquals(self._manager("get_by_tag", category=["category5", "category4"]), [])
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue