Moved the indexing to be a catalog wrapper.

svn path=/plone.app.discussion/trunk/; revision=27005
This commit is contained in:
Lennart Regebro 2009-05-17 18:38:45 +00:00
parent e35f761939
commit acf00c9de9
6 changed files with 73 additions and 48 deletions

View File

@ -115,7 +115,9 @@ class Conversation(Persistent, Explicit):
if not reply_to in self._children:
self._children[reply_to] = LLSet()
self._children[reply_to].insert(id)
notify(ObjectAddedEvent(comment, self, id))
# Notify that the object is added. The object must here be
# acquisition wrapped or the indexing will fail.
notify(ObjectAddedEvent(comment.__of__(self), self, id))
notify(ContainerModifiedEvent(self))
# Dict API
@ -137,6 +139,9 @@ class Conversation(Persistent, Explicit):
def keys(self):
return self._comments.keys()
def getPhysicalPath(self):
return self.aq_parent.getPhysicalPath() + (self.id,)
# TODO: Update internal data structures when items added or removed
@implementer(IConversation)

View File

@ -3,7 +3,7 @@
<utilities>
<utility
interface="plone.app.discussion.interfaces.ICommentingTool"
factory="plone.app.discussion.tool.CommentingTool"
object="portal_discussion"
/>
</utilities>
</componentregistry>

View File

@ -0,0 +1,5 @@
<?xml version="1.0"?>
<tool-setup>
<required tool_id="portal_discussion"
class="plone.app.discussion.tool.CommentingTool"/>
</tool-setup>

View File

@ -29,7 +29,10 @@ class ConversationTest(TestCase):
# Create a conversation. In this case we doesn't assign it to an
# object, as we just want to check the Conversation object API.
conversation = IConversation(self.portal.doc1)
# Pretend that we have traversed to the comment by aq wrapping it.
# XXX implement traversal to commenting and change this:
conversation = conversation.__of__(self.portal.doc1)
# Add a comment. reply_to=0 means it's not a reply
comment = Comment(conversation=conversation, reply_to=0)
comment.title = 'Comment 1'

View File

@ -29,6 +29,9 @@ class ToolTest(TestCase):
# Create a conversation. In this case we doesn't assign it to an
# object, as we just want to check the Conversation object API.
conversation = IConversation(self.portal.doc1)
# Pretend that we have traversed to the comment by aq wrapping it.
# XXX implement traversal to commenting and change this:
conversation = conversation.__of__(self.portal.doc1)
# Add a comment. reply_to=0 means it's not a reply
comment = Comment(conversation=conversation, reply_to=0)
@ -39,8 +42,10 @@ class ToolTest(TestCase):
# Check that the comment got indexed in the tool:
tool = getUtility(ICommentingTool)
comment = list(tool.search())[0]
self.assertEquals(comment['text'], 'Comment text')
comment = list(tool.searchResults())
self.assert_(len(comment) == 1, "There is only one comment, but we got"
" %s results in the search" % len(comment))
self.assertEquals(comment[0].Title, 'Comment 1')
def test_suite():

View File

@ -2,58 +2,65 @@ import time
from zope import interface
from zope.component import getUtility
from BTrees.OOBTree import OOBTree, OOSet, intersection
from interfaces import ICommentingTool
from interfaces import ICommentingTool, IComment
# The commenting tool, which is a local utility
class CommentingTool(object):
from Products.CMFCore.utils import UniqueObject, getToolByName
from OFS.SimpleItem import SimpleItem
class CommentingTool(UniqueObject, SimpleItem):
interface.implements(ICommentingTool)
def __init__(self):
self._id2uid = OOBTree() # The comment ID to object UID
self._id2text = OOBTree() # The text for a comment
self._wfstate2id = OOBTree() # To search on wf states
self._creator2id = OOBTree() # To search/order on creator ids
def index(self, comment):
# Store the object in the store:
id = comment.comment_id
self._id2uid[id] = comment.__parent__._parent_uid
self._id2text[id] = comment.text
meta_type = 'plone.app.discussion tool'
id = 'portal_discussion'
def reindexObject(self, object):
"""Remove from catalog.
"""
catalog = getToolByName(self, 'portal_catalog')
return catalog.reindexObject(object)
indexObject = reindexObject
# TODO
## Index on workflow state
#wfstate = comment.getWorkflowState()
#if not wfstate in self._wfstate2id:
#self._wfstate2id[wfstate] = OOSet()
#self._wfstate2id[wfstate].insert(id)
def unindexObject(self, object):
"""Remove from catalog.
"""
catalog = getToolByName(self, 'portal_catalog')
return catalog.unindexObject(object)
def uniqueValuesFor(self, name):
""" return unique values for FieldIndex name """
catalog = getToolByName(self, 'portal_catalog')
return catalog.uniqueValuesFor(name)
# Index on creator
creator = comment.creator
if not creator in self._creator2id:
self._creator2id[creator] = OOSet()
self._creator2id[creator].insert(id)
def search(self, creator=None):
if creator is not None:
# Get all replies for a certain object
ids = self._creator2ids.get(creator, None)
if ids is None:
raise StopIteration
else:
ids = self._id2uid.keys()
for id in ids:
yield {'id': id,
'text': self._id2text[id]
# TODO: More data + maybe brains or something?
}
def searchResults(self, REQUEST=None, **kw):
"""
Calls ZCatalog.searchResults with extra arguments that
limit the results to what the user is allowed to see.
"""
catalog = getToolByName(self, 'portal_catalog')
object_provides = [IComment.__identifier__]
if 'object_provides' in kw:
kw_provides = kw['object_provides']
if isinstance(str, kw_provides):
object_provides.append(kw_provides)
else:
object_provides.extend(kw_provides)
if REQUEST is not None and 'object_provides' in REQUEST.form:
rq_provides = REQUEST.form['object_provides']
del REQUEST.form['object_provides']
if isinstance(str, rq_provides):
object_provides.append(rq_provides)
else:
object_provides.extend(rq_provides)
kw['object_provides'] = object_provides
return catalog.searchResults(REQUEST, **kw)
def object_added_handler(obj, event):
tool = getUtility(ICommentingTool)
tool.index(obj)
tool.indexObject(obj)
def object_removed_handler(obj, event):
tool = getUtility(ICommentingTool)
tool.unindex(obj)
tool.unindexObject(obj)