plone.app.discussion/plone/app/discussion/catalog.py

167 lines
4.4 KiB
Python

"""Catalog indexers, using plone.indexer. These will populate standard catalog
indexes with values based on the IComment interface.
Also provide event handlers to actually catalog the comments.
"""
from DateTime import DateTime
from plone.app.discussion.interfaces import IComment
from plone.app.discussion.interfaces import IConversation
from plone.base.utils import safe_text
from plone.indexer import indexer
from plone.uuid.interfaces import IUUID
from Products.CMFCore.interfaces import IContentish
from Products.ZCatalog.interfaces import IZCatalog
MAX_DESCRIPTION = 25
# Conversation Indexers
@indexer(IContentish, IZCatalog)
def total_comments(object):
# Total number of comments on a conversation
# Indexers won't work on old discussion items
if object.meta_type != "Discussion Item":
try:
conversation = IConversation(object)
return conversation.total_comments()
except TypeError: # pragma: no cover
# The item is contentish but nobody
# implemented an adapter for it
pass
@indexer(IContentish, IZCatalog)
def last_comment_date(object):
# Date of the latest comment on a conversation
# Indexers won't work on old discussion items
if object.meta_type != "Discussion Item":
try:
conversation = IConversation(object)
return conversation.last_comment_date
except TypeError: # pragma: no cover
# The item is contentish but nobody
# implemented an adapter for it
pass
@indexer(IContentish, IZCatalog)
def commentators(object):
# List of commentators on a conversation
# Indexers won't work on old discussion items
if object.meta_type != "Discussion Item":
try:
conversation = IConversation(object)
return conversation.public_commentators
except TypeError: # pragma: no cover
# The item is contentish but nobody
# implemented an adapter for it
pass
# Comment Indexers
@indexer(IComment)
def title(object):
return object.Title()
@indexer(IComment)
def creator(object):
if not object.creator:
return
value = safe_text(object.creator)
return value
@indexer(IComment)
def description(object):
# Return the first 25 words of the comment text and append ' [...]'
text = " ".join(
object.getText(targetMimetype="text/plain").split()[:MAX_DESCRIPTION],
)
if len(object.getText().split()) > 25:
text += " [...]"
return text
@indexer(IComment)
def searchable_text(object):
return object.getText(targetMimetype="text/plain")
@indexer(IComment)
def in_response_to(object):
# Always returns the content object the comment is added to.
# Do not confuse this with the in_reply_to attribute of a comment!
return object.__parent__.__parent__.title_or_id()
@indexer(IComment)
def effective(object):
# the catalog index needs Zope DateTime instead of Python datetime
return DateTime(
object.creation_date.year,
object.creation_date.month,
object.creation_date.day,
object.creation_date.hour,
object.creation_date.minute,
object.creation_date.second,
object.creation_date.tzname(),
)
@indexer(IComment)
def created(object):
# the catalog index needs Zope DateTime instead of Python datetime
return DateTime(
object.creation_date.year,
object.creation_date.month,
object.creation_date.day,
object.creation_date.hour,
object.creation_date.minute,
object.creation_date.second,
object.creation_date.tzname(),
)
@indexer(IComment)
def modified(object):
# the catalog index needs Zope DateTime instead of Python datetime
return DateTime(
object.modification_date.year,
object.modification_date.month,
object.modification_date.day,
object.modification_date.hour,
object.modification_date.minute,
object.modification_date.second,
object.modification_date.tzname(),
)
# Override the conversation indexers for comments
@indexer(IComment)
def comments_total_comments(object):
return None
@indexer(IComment)
def comments_last_comment_date(object):
return None
@indexer(IComment)
def comments_commentators(object):
return None
# Make sure comments don't inherit their container's UID
@indexer(IComment)
def UID(object):
if IUUID:
return IUUID(object, None)