# -*- coding: utf-8 -*- import datetime import logging import unittest from zope.component import createObject from zope.component import getMultiAdapter from Products.PloneTestCase.ptc import PloneTestCase from plone.app.discussion.tests.layer import DiscussionLayer from plone.app.discussion.interfaces import IComment, IConversation, IReplies from plone.app.discussion.browser.comment import View logger = logging.getLogger('plone.app.discussion.tests') logger.addHandler(logging.StreamHandler()) class CommentTest(PloneTestCase): layer = DiscussionLayer def afterSetUp(self): """Create a document. """ self.loginAsPortalOwner() self.portal.invokeFactory(id='doc1', title='Document 1', type_name='Document') def test_factory(self): comment1 = createObject('plone.Comment') self.assert_(IComment.providedBy(comment1)) def test_UTCDates(self): utc_to_local_diff = datetime.datetime.now() - datetime.datetime.utcnow() utc_to_local_diff = abs(utc_to_local_diff.seconds) if utc_to_local_diff < 60: logger.warning("Your computer is living in a timezone where local " "time equals utc time. Some potential errors can " "get hidden by that") comment1 = createObject('plone.Comment') local_utc = datetime.datetime.utcnow() for date in (comment1.creation_date, comment1.modification_date): difference = abs(date - local_utc) difference = difference.seconds # We hope that between comment1 and local_utc happen less than # 10 seconds self.assertFalse(difference / 10) def test_id(self): comment1 = createObject('plone.Comment') comment1.comment_id = 123 self.assertEquals('123', comment1.id) self.assertEquals('123', comment1.getId()) self.assertEquals(u'123', comment1.__name__) def test_title(self): conversation = IConversation(self.portal.doc1) comment1 = createObject('plone.Comment') comment1.creator = "Jim Fulton" conversation.addComment(comment1) self.assertEquals("Jim Fulton on Document 1", comment1.Title()) def test_no_name_title(self): conversation = IConversation(self.portal.doc1) comment1 = createObject('plone.Comment') conversation.addComment(comment1) self.assertEquals("Anonymous on Document 1", comment1.Title()) def test_title_special_characters(self): self.portal.invokeFactory(id='doc_sp_chars', title=u'Document äüö', type_name='Document') conversation = IConversation(self.portal.doc_sp_chars) comment1 = createObject('plone.Comment') comment1.creator = u"Tarek Ziadé" conversation.addComment(comment1) self.assertEquals(u"Tarek Ziadé on Document äüö", comment1.Title()) def test_creator(self): comment1 = createObject('plone.Comment') comment1.creator = "Jim" self.assertEquals("Jim", comment1.Creator()) def test_type(self): comment1 = createObject('plone.Comment') self.assertEquals(comment1.Type(), 'Comment') def test_traversal(self): # make sure comments are traversable, have an id, absolute_url and # physical path conversation = IConversation(self.portal.doc1) comment1 = createObject('plone.Comment') comment1.text = 'Comment text' new_comment1_id = conversation.addComment(comment1) comment = self.portal.doc1.restrictedTraverse( '++conversation++default/%s' % new_comment1_id) self.assert_(IComment.providedBy(comment)) self.assertEquals(('', 'plone', 'doc1', '++conversation++default', str(new_comment1_id)), comment.getPhysicalPath()) self.assertEquals('http://nohost/plone/doc1/++conversation++default/' + str(new_comment1_id), comment.absolute_url()) def test_workflow(self): """Basic test for the 'comment_review_workflow' """ self.portal.portal_workflow.setChainForPortalTypes( ('Discussion Item',), ('comment_review_workflow,')) conversation = IConversation(self.portal.doc1) comment1 = createObject('plone.Comment') new_comment1_id = conversation.addComment(comment1) comment = conversation[new_comment1_id] # Make sure comments use the 'comment_review_workflow' chain = self.portal.portal_workflow.getChainFor(comment) self.assertEquals(('comment_review_workflow',), chain) # Ensure the initial state was entered and recorded self.assertEquals(1, len(comment.workflow_history['comment_review_workflow'])) self.assertEquals(None, comment.workflow_history['comment_review_workflow'][0]\ ['action']) self.assertEquals('pending', self.portal.portal_workflow.getInfoFor(comment, 'review_state')) def test_fti(self): # test that we can look up an FTI for Discussion Item self.assert_("Discussion Item" in self.portal.portal_types.objectIds()) comment1 = createObject('plone.Comment') fti = self.portal.portal_types.getTypeInfo(comment1) self.assertEquals('Discussion Item', fti.getTypeInfo(comment1).getId()) def test_view(self): # make sure that the comment view is there and redirects to the right # URL # 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) # Create a comment comment1 = createObject('plone.Comment') comment1.text = 'Comment text' # Add comment to the conversation new_comment1_id = conversation.addComment(comment1) comment = self.portal.doc1.restrictedTraverse( '++conversation++default/%s' % new_comment1_id) # make sure the view is there self.failUnless(getMultiAdapter((comment, self.app.REQUEST), name='view')) # make sure the HTTP redirect (status code 302) works when a comment # is called directly view = View(comment, self.app.REQUEST) View.__call__(view) self.assertEquals(self.app.REQUEST.response.status, 302) class RepliesTest(PloneTestCase): # test the IReplies adapter on a comment layer = DiscussionLayer def afterSetUp(self): # First we need to create some content. self.loginAsPortalOwner() self.portal.invokeFactory(id='doc1', title='Document 1', type_name='Document') def test_add_comment(self): # Add comments to a CommentReplies adapter # 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) # Add a comment to the conversation replies = IReplies(conversation) comment = createObject('plone.Comment') comment.text = 'Comment text' new_id = replies.addComment(comment) comment = self.portal.doc1.restrictedTraverse( '++conversation++default/%s' % new_id) # Add a reply to the CommentReplies adapter of the first comment re_comment = createObject('plone.Comment') re_comment.text = 'Comment text' replies = IReplies(comment) new_re_id = replies.addComment(re_comment) # check that replies provides the IReplies interface self.assert_(IReplies.providedBy(replies)) # Make sure our comment was added self.failUnless(new_re_id in replies) # Make sure it is also reflected in the conversation self.failUnless(new_re_id in conversation) # Make sure the conversation has the correct comment id self.assertEquals(conversation[new_re_id].comment_id, new_re_id) def test_delete_comment(self): # Add and remove a comment to a CommentReplies adapter # 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) # Add a comment to the conversation replies = IReplies(conversation) comment = createObject('plone.Comment') comment.text = 'Comment text' new_id = replies.addComment(comment) comment = self.portal.doc1.restrictedTraverse( '++conversation++default/%s' % new_id) # Add a reply to the CommentReplies adapter of the first comment re_comment = createObject('plone.Comment') re_comment.text = 'Comment text' replies = IReplies(comment) new_re_id = replies.addComment(re_comment) # Remove the reply to the CommentReplies adapter del replies[new_re_id] # Make sure there is no comment left in CommentReplies self.assertEquals(len(replies), 0) # Make sure the first comment is still in the conversation self.assertEquals(conversation.total_comments, 1) def test_traversal(self): # Create a nested structure of comment replies and check the traversal # make sure comments are traversable, have an id, absolute_url and # physical path conversation = IConversation(self.portal.doc1) comment1 = createObject('plone.Comment') comment1.text = 'Comment text' conversation.addComment(comment1) comment = createObject('plone.Comment') comment.text = 'Comment text' new_id = conversation.addComment(comment) comment = self.portal.doc1.restrictedTraverse( '++conversation++default/%s' % new_id) # Add a reply to the CommentReplies adapter of the first comment re_comment = createObject('plone.Comment') re_comment.text = 'Comment text' replies = IReplies(comment) new_re_id = replies.addComment(re_comment) re_comment = self.portal.doc1.restrictedTraverse( '++conversation++default/%s' % new_re_id) # Add a reply to the reply re_re_comment = createObject('plone.Comment') re_re_comment.text = 'Comment text' replies = IReplies(re_comment) new_re_re_id = replies.addComment(re_re_comment) re_re_comment = self.portal.doc1.restrictedTraverse( '++conversation++default/%s' % new_re_re_id) # Add a reply to the replies reply re_re_re_comment = createObject('plone.Comment') re_re_re_comment.text = 'Comment text' replies = IReplies(re_re_comment) new_re_re_re_id = replies.addComment(re_re_re_comment) re_re_re_comment = self.portal.doc1.restrictedTraverse( '++conversation++default/%s' % new_re_re_re_id) self.assertEquals(('', 'plone', 'doc1', '++conversation++default', str(new_id)), comment.getPhysicalPath()) self.assertEquals('http://nohost/plone/doc1/++conversation++default/' + str(new_id), comment.absolute_url()) self.assertEquals(('', 'plone', 'doc1', '++conversation++default', str(new_re_id)), re_comment.getPhysicalPath()) self.assertEquals('http://nohost/plone/doc1/++conversation++default/' + str(new_re_id), re_comment.absolute_url()) self.assertEquals(('', 'plone', 'doc1', '++conversation++default', str(new_re_re_id)), re_re_comment.getPhysicalPath()) self.assertEquals('http://nohost/plone/doc1/++conversation++default/' + str(new_re_re_id), re_re_comment.absolute_url()) self.assertEquals(('', 'plone', 'doc1', '++conversation++default', str(new_re_re_re_id)), re_re_re_comment.getPhysicalPath()) self.assertEquals('http://nohost/plone/doc1/++conversation++default/' + str(new_re_re_re_id), re_re_re_comment.absolute_url()) def test_suite(): return unittest.defaultTestLoader.loadTestsFromName(__name__)