testERP5Web.py 83.1 KB
Newer Older
1
# -*- coding: utf-8 -*-
2 3
##############################################################################
#
4
# Copyright (c) 2004, 2005, 2006 Nexedi SARL and Contributors.
5 6 7 8
# All Rights Reserved.
#          Romain Courteaud <romain@nexedi.com>
#
# WARNING: This program as such is intended to be used by professional
9
# programmers who take the whole responsibility of assessing all potential
10 11
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
12
# guarantees and support are strongly adviced to contract a Free Software
13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
##############################################################################

31
import re
32
import unittest
33
from unittest import expectedFailure, skip
34 35
from StringIO import StringIO
from urllib import urlencode
36
from AccessControl import Unauthorized
37
from Testing import ZopeTestCase
38
from DateTime import DateTime
39
from Products.ERP5Type.tests.ERP5TypeTestCase import ERP5TypeTestCase
40
from Products.ERP5Type.tests.utils import DummyLocalizer
41
from Products.ERP5Type.tests.utils import createZODBPythonScript
42

Rafael Monnerat's avatar
Rafael Monnerat committed
43
LANGUAGE_LIST = ('en', 'fr', 'de', 'bg', )
44 45
HTTP_OK = 200
MOVED_TEMPORARILY = 302
46

Rafael Monnerat's avatar
Rafael Monnerat committed
47

48 49 50 51
class DummyTraversalHook(object):
  def __call__(self, container, request):
    return

52

53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
class WebTraversalHookTestMixin(object):
  """Mixin class to test the WebSiteTraversalHook on both websection and website.
  """
  def getBusinessTemplateList(self):
    return ('erp5_base', 'erp5_web', )

  def test_TraversalHook_on_newContent(self):
    """a WebSiteTraversalHook is added on websections and websites automatically.
    """
    self.assertEquals(1, len(self.web_section.__before_traverse__))
    self.assertIsInstance(
      self.web_section.__before_traverse__.values()[0],
      self.traversal_hook_class)

  def test_TraversalHook_on_clone(self):
    """a WebSiteTraversalHook is correctly updated after cloning a websection/website.
    """
    cloned_web_section = self.web_section.Base_createCloneDocument(batch_mode=True)
    self.assertEquals(1, len(cloned_web_section.__before_traverse__))

  def test_TraversalHook_on_change_id(self):
    """a TraversalHook is correctly updated after changing websection id.
    """
    self.tic()
    self.web_section.setId("new_id")
    self.assertEquals(1, len(self.web_section.__before_traverse__))

80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105
  def test_TraversalHook_cleanup_on_edit(self):
    """Old traversal hooks from cloned objects are automatically cleaned up
    when section is edited.
    """
    # artificially put this websection in a similar state that websection were before
    # we fix the bug keeping traversal hooks on clone.
    from ZPublisher import BeforeTraverse
    handle = '%s/different_id_than_%s' % (self.web_section.meta_type, self.web_section.getId())
    BeforeTraverse.registerBeforeTraverse(self.web_section, self.traversal_hook_class(), handle)

    BeforeTraverse.registerBeforeTraverse(
      self.web_section,
      DummyTraversalHook(),
      'unrelated_traversal_hook_that_should_be_kept')
    self.assertEquals(3, len(self.web_section.__before_traverse__))

    self.tic()
    self.web_section.edit(title=self.id())
    # We have cleaned up the useless before traversal hook, but keep the unrelated one
    self.assertEquals(2, len(self.web_section.__before_traverse__))
    self.assertEquals(1, len([hook for hook in
      self.web_section.__before_traverse__.values() if isinstance(hook, self.traversal_hook_class)]))
    self.assertEquals(1, len([hook for hook in
      self.web_section.__before_traverse__.values() if isinstance(hook, DummyTraversalHook)]))


106 107 108 109 110 111 112 113 114
class TestWebSiteTraversalHook(WebTraversalHookTestMixin, ERP5TypeTestCase):
  def afterSetUp(self):
    super(TestWebSiteTraversalHook, self).afterSetUp()
    self.web_section = self.portal.web_site_module.newContent(
      portal_type='Web Site',
    )
    from Products.ERP5Type.Document.WebSite import WebSiteTraversalHook
    self.traversal_hook_class = WebSiteTraversalHook

115

116 117 118 119 120 121 122 123 124 125 126 127
class TestWebSectionTraversalHook(WebTraversalHookTestMixin, ERP5TypeTestCase):
  def afterSetUp(self):
    super(TestWebSectionTraversalHook, self).afterSetUp()
    web_site = self.portal.web_site_module.newContent(
      portal_type='Web Site',
    )
    self.web_section = web_site.newContent(portal_type='Web Section')

    from Products.ERP5Type.Document.WebSection import WebSectionTraversalHook
    self.traversal_hook_class = WebSectionTraversalHook


128
class TestERP5Web(ERP5TypeTestCase):
129 130 131 132
  """Test for erp5_web business template.
  """
  manager_username = 'zope'
  manager_password = 'zope'
133
  website_id = 'test'
134
  credential = '%s:%s' % (manager_username, manager_password)
Rafael Monnerat's avatar
Rafael Monnerat committed
135

136 137 138 139 140 141 142
  def getTitle(self):
    return "ERP5Web"

  def getBusinessTemplateList(self):
    """
    Return the list of required business templates.
    """
143 144
    return ('erp5_core_proxy_field_legacy',
            'erp5_base',
145
            'erp5_jquery',
146 147
            'erp5_web',
            )
148 149

  def afterSetUp(self):
150
    portal = self.getPortal()
151 152

    uf = portal.acl_users
Rafael Monnerat's avatar
Rafael Monnerat committed
153 154 155
    uf._doAddUser(self.manager_username,
                  self.manager_password,
                  ['Manager'], [])
156 157
    self.login(self.manager_username)

158 159
    self.web_page_module = self.portal.getDefaultModule('Web Page Module')
    self.web_site_module = self.portal.getDefaultModule('Web Site Module')
Rafael Monnerat's avatar
Rafael Monnerat committed
160
    portal.Localizer.manage_changeDefaultLang(language='en')
161 162
    self.portal_id = self.portal.getId()

163 164 165 166
  def clearModule(self, module):
    module.manage_delObjects(list(module.objectIds()))
    self.tic()

167
  def beforeTearDown(self):
168 169
    self.clearModule(self.portal.web_site_module)
    self.clearModule(self.portal.web_page_module)
170
    self.clearModule(self.portal.person_module)
171

172
  def setupWebSite(self, **kw):
173
    """
174 175 176
      Setup Web Site
    """
    portal = self.getPortal()
177

178 179 180
    # add supported languages for Localizer
    localizer = portal.Localizer
    for language in LANGUAGE_LIST:
Rafael Monnerat's avatar
Rafael Monnerat committed
181
      localizer.manage_addLanguage(language=language)
182

183 184 185
    # create website
    if hasattr(self.web_site_module, self.website_id):
      self.web_site_module.manage_delObjects(self.website_id)
Rafael Monnerat's avatar
Rafael Monnerat committed
186 187 188
    website = self.web_site_module.newContent(portal_type='Web Site',
                                              id=self.website_id,
                                              **kw)
189
    website.publish()
190
    self.tic()
191
    return website
192

193
  def setupWebSection(self, **kw):
194
    """
195 196
      Setup Web Section
    """
Rafael Monnerat's avatar
Rafael Monnerat committed
197
    website = self.web_site_module[self.website_id]
198
    websection = website.newContent(portal_type='Web Section', **kw)
199
    self.websection = websection
Rafael Monnerat's avatar
Rafael Monnerat committed
200
    kw = dict(criterion_property_list='portal_type',
201 202
              membership_criterion_base_category_list='',
              membership_criterion_category_list='')
203
    websection.edit(**kw)
204 205
    websection.setCriterion(property='portal_type',
                            identity=['Web Page'],
206
                            max='',
207
                            min='')
208

209
    self.tic()
210 211
    return websection

212
  def setupWebSitePages(self, prefix, suffix=None, version='0.1',
213
                        language_list=LANGUAGE_LIST, **kw):
214 215 216 217 218 219 220
    """
      Setup some Web Pages.
    """
    webpage_list = []
    # create sample web pages
    for language in language_list:
      if suffix is not None:
221
        reference = '%s-%s' % (prefix, language)
222 223
      else:
        reference = prefix
224
      webpage = self.web_page_module.newContent(portal_type='Web Page',
225 226 227 228
                                                reference=reference,
                                                version=version,
                                                language=language,
                                                **kw)
229
      webpage.publish()
230
      self.tic()
231 232 233 234
      self.assertEqual(language, webpage.getLanguage())
      self.assertEqual(reference, webpage.getReference())
      self.assertEqual(version, webpage.getVersion())
      self.assertEqual('published', webpage.getValidationState())
235
      webpage_list.append(webpage)
236

237
    return webpage_list
238

239
  def test_01_WebSiteRecatalog(self):
240 241 242 243
    """
      Test that a recataloging works for Web Site documents
    """
    self.setupWebSite()
Rafael Monnerat's avatar
Rafael Monnerat committed
244
    web_site = self.web_site_module[self.website_id]
245

246 247 248 249 250 251 252 253
    self.assertTrue(web_site is not None)
    # Recatalog the Web Site document
    portal_catalog = self.getCatalogTool()
    try:
      portal_catalog.catalog_object(web_site)
    except:
      self.fail('Cataloging of the Web Site failed.')

254 255


256
  def test_02_EditSimpleWebPage(self):
257 258
    """
      Simple Case of creating a web page.
259 260 261
    """
    page = self.web_page_module.newContent(portal_type='Web Page')
    page.edit(text_content='<b>OK</b>')
262 263
    self.assertEqual('text/html', page.getContentType())
    self.assertEqual('<b>OK</b>', page.getTextContent())
264

265 266
  def test_WebPageAsTextUTF8(self):
    """Check if Web Page's asText() returns utf-8 string correctly
267 268
    """
    page = self.web_page_module.newContent(portal_type='Web Page')
269
    page.edit(text_content='<p>Hé Hé Hé!</p>', content_type='text/html')
270
    self.tic()
271
    self.assertEqual('Hé Hé Hé!', page.asText().strip())
272 273 274 275 276 277 278 279 280 281 282 283 284 285

  def test_WebPageAsTextHTMLEntities(self):
    """Check if Web Page's asText() converts html entities properly
    """
    page = self.web_page_module.newContent(portal_type='Web Page')
    page.edit(text_content='<p>H&eacute;!</p>', content_type='text/html')
    self.tic()
    self.assertEqual('Hé!', page.asText().strip())

  def test_WebPageAsTextWrap(self):
    """Check if Web Page's asText() is wrapped by certain column width.
    """
    page = self.web_page_module.newContent(portal_type='Web Page')

286
    page.edit(text_content='<p>Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé!</p>')
287
    self.tic()
288
    self.assertEqual("""Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé Hé
289 290
Hé Hé Hé!""", page.asText().strip())

291
  @skip('WebSite_createWebSiteAccount is disabled by default.')
292
  def test_03_CreateWebSiteUser(self):
293 294 295
    """
      Create Web site User.
    """
296
    self.setupWebSite()
297 298
    portal = self.getPortal()
    request = self.app.REQUEST
Rafael Monnerat's avatar
Rafael Monnerat committed
299 300 301 302 303 304
    kw = dict(reference='web',
              first_name='TestFN',
              last_name='TestLN',
              default_email_text='test@test.com',
              password='abc',
              password_confirm='abc',)
305
    for key, item in kw.items():
Rafael Monnerat's avatar
Rafael Monnerat committed
306 307
      request.set('field_your_%s' % key, item)
    website = self.web_site_module[self.website_id]
308
    website.WebSite_createWebSiteAccount('WebSite_viewRegistrationDialog')
309

310
    self.tic()
311

312
    # find person object by reference
Rafael Monnerat's avatar
Rafael Monnerat committed
313 314
    person = website.ERP5Site_getAuthenticatedMemberPersonValue(
                                                           kw['reference'])
315 316 317 318 319
    self.assertEqual(person.getReference(), kw['reference'])
    self.assertEqual(person.getFirstName(), kw['first_name'])
    self.assertEqual(person.getLastName(), kw['last_name'])
    self.assertEqual(person.getDefaultEmailText(), kw['default_email_text'])
    self.assertEqual(person.getValidationState(), 'validated')
320

321
    # check if user account is 'loggable'
322
    uf = portal.acl_users
Rafael Monnerat's avatar
Rafael Monnerat committed
323
    user = uf.getUserById(kw['reference'])
324 325
    self.assertEqual(str(user), kw['reference'])
    self.assertEqual(1, user.has_role(('Member', 'Authenticated',)))
326
    self.login(kw['reference'])
327
    self.assertEqual(kw['reference'],
Rafael Monnerat's avatar
Rafael Monnerat committed
328
                      str(portal.portal_membership.getAuthenticatedMember()))
329 330 331

    # test redirection to person oobject
    path = website.absolute_url_path() + '/WebSite_redirectToUserView'
Rafael Monnerat's avatar
Rafael Monnerat committed
332
    response = self.publish(path, '%s:%s' % (kw['reference'], kw['password']))
333
    self.assertTrue(person.getRelativeUrl() in response.getHeader("Location"))
Rafael Monnerat's avatar
Rafael Monnerat committed
334

335 336
    # test redirecting to new Person preference
    path = website.absolute_url_path() + '/WebSite_redirectToUserPreference'
Rafael Monnerat's avatar
Rafael Monnerat committed
337
    response = self.publish(path, '%s:%s' % (kw['reference'], kw['password']))
338 339
    self.assertTrue('portal_preferences' in response.getHeader("Location"))
    # one preference should be created for user
340
    self.assertEqual(1,
Rafael Monnerat's avatar
Rafael Monnerat committed
341 342
        self.portal.portal_catalog.countResults(**{'portal_type': 'Preference',
                                              'owner': kw['reference']})[0][0])
343

344
  def test_04_WebPageTranslation(self):
345
    """
346
      Simple Case of showing the proper Web Page based on
347 348 349 350 351 352
      current user selected language in browser.
    """
    portal = self.getPortal()
    website = self.setupWebSite()
    websection = self.setupWebSection()
    page_reference = 'default-webpage'
Rafael Monnerat's avatar
Rafael Monnerat committed
353
    webpage_list = self.setupWebSitePages(prefix=page_reference)
354

355
    # set default web page for section
356
    found_by_reference = portal.portal_catalog(reference=page_reference,
Rafael Monnerat's avatar
Rafael Monnerat committed
357 358 359 360
                                               portal_type='Web Page')
    found = found_by_reference[0].getObject()
    websection.edit(categories_list=['aggregate/%s' % found.getRelativeUrl()])
    self.assertEqual([found.getReference()],
361 362 363
                      websection.getAggregateReferenceList())
    # even though we create many pages we should get only one
    # this is the most recent one since all share the same reference
364
    self.assertEqual(1, len(websection.WebSection_getDocumentValueList()))
365

366
    # use already created few pages in different languages with same reference
367
    # and check that we always get the right one based on selected
368 369 370
    # by us language
    for language in LANGUAGE_LIST:
      # set default language in Localizer only to check that we get
371
      # the corresponding web page for language.
372
      # XXX: Extend API so we can select language from REQUEST
Rafael Monnerat's avatar
Rafael Monnerat committed
373
      portal.Localizer.manage_changeDefaultLang(language=language)
374
      default_document = websection.getDefaultDocumentValue()
375
      self.assertEqual(language, default_document.getLanguage())
376

377
  def test_05_WebPageTextContentSubstitutions(self):
378
    """
Rafael Monnerat's avatar
Rafael Monnerat committed
379 380
      Simple Case of showing the proper text content with and without a
      substitution mapping method.
381
      In case of asText, the content should be replaced too
382 383
    """
    content = '<a href="${toto}">$titi</a>'
384
    asText_content = '$titi\n'
385
    substituted_content = '<a href="foo">bar</a>'
386
    substituted_asText_content = 'bar\n'
387 388 389
    mapping = dict(toto='foo', titi='bar')

    portal = self.getPortal()
390
    document = portal.web_page_module.newContent(portal_type='Web Page',
391
            text_content=content)
392

393
    # No substitution should occur.
394 395
    self.assertEqual(document.asStrippedHTML(), content)
    self.assertEqual(document.asText(), asText_content)
396 397 398 399 400 401

    klass = document.__class__
    klass.getTestSubstitutionMapping = lambda self, **kw: mapping
    document.setTextContentSubstitutionMappingMethodId('getTestSubstitutionMapping')

    # Substitutions should occur.
402 403
    self.assertEqual(document.asStrippedHTML(), substituted_content)
    self.assertEqual(document.asText(), substituted_asText_content)
404 405 406 407

    klass._getTestSubstitutionMapping = klass.getTestSubstitutionMapping
    document.setTextContentSubstitutionMappingMethodId('_getTestSubstitutionMapping')

Rafael Monnerat's avatar
Rafael Monnerat committed
408 409
    # Even with the same callable object, a restricted method
    # id should not be callable.
410 411
    self.assertRaises(Unauthorized, document.asStrippedHTML)

412
  def test_06_DefaultDocumentForWebSection(self):
413 414 415 416
    """
      Testing the default document for a Web Section.

      If a Web Section has a default document defined and if that default
417 418
      document is published, then getDefaultDocumentValue on that
      web section should return the latest version in the most
419
      appropriate language of that default document.
420 421

      Note: due to generic ERP5 Web implementation this test highly depends
422
      on WebSection_geDefaulttDocumentValueList
Ivan Tyagov's avatar
Ivan Tyagov committed
423 424 425
    """
    website = self.setupWebSite()
    websection = self.setupWebSection()
Rafael Monnerat's avatar
Rafael Monnerat committed
426
    publication_section_category_id_list = ['documentation', 'administration']
427

Ivan Tyagov's avatar
Ivan Tyagov committed
428
    # create pages belonging to this publication_section 'documentation'
Rafael Monnerat's avatar
Rafael Monnerat committed
429 430 431 432 433 434
    web_page_en = self.web_page_module.newContent(
             portal_type='Web Page',
             id='section_home',
             language='en',
             reference='NXD-DDP',
             publication_section_list=publication_section_category_id_list[:1])
Ivan Tyagov's avatar
Ivan Tyagov committed
435
    websection.setAggregateValue(web_page_en)
436
    self.tic()
437
    self.assertEqual(None, websection.getDefaultDocumentValue())
Ivan Tyagov's avatar
Ivan Tyagov committed
438 439
    # publish it
    web_page_en.publish()
440
    self.tic()
441 442 443 444 445 446 447
    self.assertEqual(web_page_en, websection.getDefaultDocumentValue())
    # and make sure that the base meta tag which is generated
    # uses the web section rather than the portal
    html_page = websection()
    from Products.ERP5.Document.Document import Document
    base_list = re.findall(Document.base_parser, str(html_page))
    base_url = base_list[0]
Rafael Monnerat's avatar
Rafael Monnerat committed
448 449
    self.assertEqual(base_url, "%s/%s/" % (websection.absolute_url(),
                                           web_page_en.getReference()))
450

451
  def test_06b_DefaultDocumentForWebSite(self):
452 453 454 455
    """
      Testing the default document for a Web Site.

      If a Web Section has a default document defined and if that default
456 457
      document is published, then getDefaultDocumentValue on that
      web section should return the latest version in the most
458
      appropriate language of that default document.
459 460

      Note: due to generic ERP5 Web implementation this test highly depends
461 462 463
      on WebSection_geDefaulttDocumentValueList
    """
    website = self.setupWebSite()
Rafael Monnerat's avatar
Rafael Monnerat committed
464
    publication_section_category_id_list = ['documentation', 'administration']
465

466
    # create pages belonging to this publication_section 'documentation'
Rafael Monnerat's avatar
Rafael Monnerat committed
467
    web_page_en = self.web_page_module.newContent(portal_type = 'Web Page',
468
                                                 id='site_home',
469 470 471
                                                 language = 'en',
                                                 reference='NXD-DDP-Site',
                                                 publication_section_list=publication_section_category_id_list[:1])
472
    website.setAggregateValue(web_page_en)
473
    self.tic()
474 475 476
    self.assertEqual(None, website.getDefaultDocumentValue())
    # publish it
    web_page_en.publish()
477
    self.tic()
478 479 480 481 482 483 484
    self.assertEqual(web_page_en, website.getDefaultDocumentValue())
    # and make sure that the base meta tag which is generated
    # uses the web site rather than the portal
    html_page = website()
    from Products.ERP5.Document.Document import Document
    base_list = re.findall(Document.base_parser, str(html_page))
    base_url = base_list[0]
485
    self.assertEqual(base_url, "%s/%s/" % (website.absolute_url(), web_page_en.getReference()))
486

487
  def test_07_WebSection_getDocumentValueList(self):
488 489 490 491 492
    """ Check getting getDocumentValueList from Web Section.
    """
    portal = self.getPortal()
    website = self.setupWebSite()
    websection = self.setupWebSection()
Rafael Monnerat's avatar
Rafael Monnerat committed
493
    publication_section_category_id_list = ['documentation', 'administration']
494

495
    #set predicate on web section using 'publication_section'
496
    websection.edit(membership_criterion_base_category = ['publication_section'],
497 498
                     membership_criterion_category=['publication_section/%s' \
                                                    % publication_section_category_id_list[0]])
499 500 501
    # clean up
    self.web_page_module.manage_delObjects(list(self.web_page_module.objectIds()))
    portal.portal_categories.publication_section.manage_delObjects(
502
                                      list(portal.portal_categories.publication_section.objectIds()))
503 504
    # create categories
    for category_id in publication_section_category_id_list:
505
      portal.portal_categories.publication_section.newContent(portal_type = 'Category',
506 507
                                                              id = category_id)

Rafael Monnerat's avatar
Rafael Monnerat committed
508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523
    property_dict = {'01': dict(language='en', version="1", reference="A"),
                     '02': dict(language='en', version="2", reference="B"),
                     '03': dict(language='en', version="3", reference="C"),
                     '04': dict(language='pt', version="1", reference="A"),
                     '05': dict(language='pt', version="2", reference="C"),
                     '06': dict(language='pt', version="3", reference="B"),
                     '07': dict(language='ja', version="1", reference="C"),
                     '08': dict(language='ja', version="2", reference="A"),
                     '09': dict(language='ja', version="3", reference="B"),
                     '10': dict(language='en', version="2", reference="D"),
                     '11': dict(language='ja', version="3", reference="E"),
                     '12': dict(language='pt', version="3", reference="F"),
                     '13': dict(language='en', version="3", reference="D"),
                     '14': dict(language='ja', version="2", reference="E"),
                     '15': dict(language='pt', version="2", reference="F"),
                     '16': dict(language='', version="1", reference="A"),
524 525
                    }
    sequence_one = property_dict.keys()
Rafael Monnerat's avatar
Rafael Monnerat committed
526 527 528 529
    sequence_two = ['01', '13', '12', '09', '06', '15', '04', '11', '02',
                    '05', '03', '07', '10', '08', '14', '16']
    sequence_three = ['05', '12', '13', '14', '06', '09', '10', '07',
                      '03', '01', '02', '11', '04', '08', '15', '16']
530

531
    sequence_count = 0
Rafael Monnerat's avatar
Rafael Monnerat committed
532
    for sequence in [sequence_one, sequence_two, sequence_three]:
533
      sequence_count += 1
534 535 536
      message = '\ntest_07_WebSection_getDocumentValueList (Sequence %s)' \
                                                              % (sequence_count)
      ZopeTestCase._print(message)
537

538
      web_page_list = []
539
      for key in sequence:
Rafael Monnerat's avatar
Rafael Monnerat committed
540
        web_page = self.web_page_module.newContent(
541
                                  title=key,
542 543
                                  portal_type = 'Web Page',
                                  publication_section_list=publication_section_category_id_list[:1])
544

545
        web_page.edit(**property_dict[key])
546
        self.tic()
547
        web_page_list.append(web_page)
548

549
      self.tic()
550
      # in draft state, no documents should belong to this Web Section
551
      self.assertEqual(0, len(websection.getDocumentValueList()))
552

553
      # when published, all web pages should belong to it
554 555
      for web_page in web_page_list:
        web_page.publish()
556
      self.tic()
557

558
      # Test for limit parameter
559
      self.assertEqual(2, len(websection.getDocumentValueList(limit=2)))
560 561

      # Testing for language parameter
562
      self.assertEqual(4, len(websection.getDocumentValueList()))
Rafael Monnerat's avatar
Rafael Monnerat committed
563 564
      self.assertEqual(['en', 'en', 'en', 'en'],
                       [w.getLanguage() for w in websection.getDocumentValueList()])
565

566 567 568 569 570
      # Check that receiving an empty string as language parameter (as done
      # when using listbox search) correctly returns user language documents.
      default_document_value_list = websection.getDocumentValueList(language='')
      self.assertEqual(4, len(default_document_value_list))
      self.assertEqual(['en', 'en', 'en', 'en'],
Rafael Monnerat's avatar
Rafael Monnerat committed
571
                       [w.getLanguage() for w in default_document_value_list])
572

573
      pt_document_value_list = websection.getDocumentValueList(language='pt')
574
      self.assertEqual(4, len(pt_document_value_list))
Rafael Monnerat's avatar
Rafael Monnerat committed
575 576
      self.assertEqual(['pt', 'pt', 'pt', 'pt'],
                           [w.getObject().getLanguage() for w in pt_document_value_list])
577

578
      ja_document_value_list = websection.getDocumentValueList(language='ja')
579
      self.assertEqual(4, len(ja_document_value_list))
Rafael Monnerat's avatar
Rafael Monnerat committed
580 581
      self.assertEqual(['ja', 'ja', 'ja', 'ja'],
                           [w.getLanguage() for w in ja_document_value_list])
582

583 584 585
      bg_document_value_list = websection.getDocumentValueList(language='bg')
      self.assertEqual(1, len(bg_document_value_list))
      self.assertEqual([''],
Rafael Monnerat's avatar
Rafael Monnerat committed
586
                       [w.getLanguage() for w in bg_document_value_list])
587

588 589 590
      # Testing for all_versions parameter
      en_document_value_list = websection.getDocumentValueList(all_versions=1)
      self.assertEqual(5, len(en_document_value_list))
Rafael Monnerat's avatar
Rafael Monnerat committed
591 592
      self.assertEqual(['en', 'en', 'en', 'en', 'en'],
                       [w.getLanguage() for w in en_document_value_list])
593 594 595 596

      pt_document_value_list = websection.getDocumentValueList(language='pt',
                                                               all_versions=1)
      self.assertEqual(5, len(pt_document_value_list))
Rafael Monnerat's avatar
Rafael Monnerat committed
597 598
      self.assertEqual(['pt', 'pt', 'pt', 'pt', 'pt'],
                       [w.getObject().getLanguage() for w in pt_document_value_list])
599

600
      ja_document_value_list = websection.getDocumentValueList(language='ja',
601 602
                                                               all_versions=1)
      self.assertEqual(5, len(ja_document_value_list))
Rafael Monnerat's avatar
Rafael Monnerat committed
603 604
      self.assertEqual(['ja', 'ja', 'ja', 'ja', 'ja'],
                           [w.getLanguage() for w in ja_document_value_list])
605

606
      # Tests for all_languages parameter (language parameter is simply ignored)
607
      en_document_value_list = websection.WebSection_getDocumentValueListBase(all_languages=1)
608
      self.assertEqual(13, len(en_document_value_list))
Rafael Monnerat's avatar
Rafael Monnerat committed
609
      self.assertEqual(4, len([w.getLanguage() for w in en_document_value_list \
610
                              if w.getLanguage() == 'en']))
611
      self.assertEqual(4, len([w.getLanguage() for w in en_document_value_list \
612
                              if w.getLanguage() == 'pt']))
613
      self.assertEqual(4, len([w.getLanguage() for w in en_document_value_list \
614
                              if w.getLanguage() == 'ja']))
615

616 617
      pt_document_value_list = websection.WebSection_getDocumentValueListBase(all_languages=1,
                                                                              language='pt')
618
      self.assertEqual(13, len(pt_document_value_list))
Rafael Monnerat's avatar
Rafael Monnerat committed
619
      self.assertEqual(4, len([w.getLanguage() for w in pt_document_value_list \
620
                              if w.getLanguage() == 'pt']))
621
      self.assertEqual(4, len([w.getLanguage() for w in pt_document_value_list \
622
                              if w.getLanguage() == 'en']))
623
      self.assertEqual(4, len([w.getLanguage() for w in pt_document_value_list \
624 625
                              if w.getLanguage() == 'ja']))

626
      # Tests for all_languages and all_versions
627 628 629 630 631 632 633 634 635 636 637
      en_document_value_list = websection.WebSection_getDocumentValueListBase(all_languages=1,
                                                                              all_versions=1)

      pt_document_value_list = websection.WebSection_getDocumentValueListBase(all_languages=1,
                                                                              all_versions=1,
                                                                              language='pt')

      ja_document_value_list = websection.WebSection_getDocumentValueListBase(all_languages=1,
                                                                              all_versions=1,
                                                                              language='ja')

Rafael Monnerat's avatar
Rafael Monnerat committed
638
      for document_value_list in [en_document_value_list, pt_document_value_list,
639 640
                                   ja_document_value_list]:

641
        self.assertEqual(16, len(document_value_list))
Rafael Monnerat's avatar
Rafael Monnerat committed
642
        self.assertEqual(5, len([w.getLanguage() for w in document_value_list \
643
                                if w.getLanguage() == 'en']))
Rafael Monnerat's avatar
Rafael Monnerat committed
644
        self.assertEqual(5, len([w.getLanguage() for w in en_document_value_list \
645
                                if w.getLanguage() == 'pt']))
Rafael Monnerat's avatar
Rafael Monnerat committed
646
        self.assertEqual(5, len([w.getLanguage() for w in en_document_value_list \
647 648 649
                                if w.getLanguage() == 'ja']))

      # Tests for sort_on parameter
Rafael Monnerat's avatar
Rafael Monnerat committed
650 651
      self.assertEqual(['A', 'B', 'C', 'D'],
                       [w.getReference() for w in \
652 653
                         websection.getDocumentValueList(sort_on=[('reference', 'ASC')])])

Rafael Monnerat's avatar
Rafael Monnerat committed
654 655
      self.assertEqual(['01', '02', '03', '13'],
                       [w.getTitle() for w in \
656 657
                         websection.getDocumentValueList(sort_on=[('title', 'ASC')])])

Rafael Monnerat's avatar
Rafael Monnerat committed
658 659
      self.assertEqual(['D', 'C', 'B', 'A'],
                       [w.getReference() for w in \
660 661
                         websection.getDocumentValueList(sort_on=[('reference', 'DESC')])])

Rafael Monnerat's avatar
Rafael Monnerat committed
662 663
      self.assertEqual(['13', '03', '02', '01'],
                       [w.getTitle() for w in \
664 665
                         websection.getDocumentValueList(sort_on=[('reference', 'DESC')])])

666
      self.assertEqual(['A', 'A', 'A', 'A', 'B', 'B', 'B', 'C', 'C', 'C', 'D', 'E', 'F'],
Rafael Monnerat's avatar
Rafael Monnerat committed
667
                       [w.getReference() for w in \
668 669
                         websection.WebSection_getDocumentValueListBase(all_languages=1,
                                            sort_on=[('reference', 'ASC')])])
670

671
      self.assertEqual(['01', '02', '03', '04', '05', '06', '07', '08', '09', '11', '12', '13', '16'],
Rafael Monnerat's avatar
Rafael Monnerat committed
672
                       [w.getTitle() for w in \
673 674 675
                         websection.WebSection_getDocumentValueListBase(all_languages=1,
                                            sort_on=[('title', 'ASC')])])

676 677
      self.assertEqual(['F', 'E', 'D', 'C', 'C', 'C', 'B', 'B', 'B', 'A', 'A', 'A', 'A'],
                                              [w.getReference() for w in \
678 679 680
                         websection.WebSection_getDocumentValueListBase(all_languages=1,
                                            sort_on=[('reference', 'DESC')])])

681
      self.assertEqual(['16', '13', '12', '11', '09', '08', '07', '06', '05', '04', '03', '02', '01'],
Rafael Monnerat's avatar
Rafael Monnerat committed
682
                       [w.getTitle() for w in \
683 684 685
                         websection.WebSection_getDocumentValueListBase(all_languages=1,
                                            sort_on=[('title', 'DESC')])])

686
      self.web_page_module.manage_delObjects(list(self.web_page_module.objectIds()))
687

688
  def test_08_AcquisitionWrappers(self):
689 690 691 692 693 694 695 696 697 698 699 700
    """Test acquisition wrappers of documents.
    Check if documents obtained by getDefaultDocumentValue, getDocumentValue
    and getDocumentValueList are wrapped appropriately.
    """
    portal = self.getPortal()

    # Make its own publication section category.
    publication_section = portal.portal_categories['publication_section']
    if publication_section._getOb('my_test_category', None) is None:
      publication_section.newContent(portal_type='Category',
                                     id='my_test_category',
                                     title='Test')
701
      self.tic()
702 703 704 705 706 707

    website = self.setupWebSite()
    websection = self.setupWebSection(
            membership_criterion_base_category_list=('publication_section',),
            membership_criterion_category=('publication_section/my_test_category',),
            )
708

709 710 711 712 713 714 715 716 717 718 719
    # Create at least two documents which belong to the publication section
    # category.
    web_page_list = self.setupWebSitePages('test1',
            language_list=('en',),
            publication_section_list=('my_test_category',))
    web_page_list2 = self.setupWebSitePages('test2',
            language_list=('en',),
            publication_section_list=('my_test_category',))

    # We need a default document.
    websection.setAggregateValue(web_page_list[0])
720
    self.tic()
721

722 723 724 725 726 727 728 729 730 731 732 733 734 735 736
    # Obtain documens in various ways.
    default_document = websection.getDefaultDocumentValue()
    self.assertNotEquals(default_document, None)

    document1 = websection.getDocumentValue('test1')
    self.assertNotEquals(document1, None)
    document2 = websection.getDocumentValue('test2')
    self.assertNotEquals(document2, None)

    document_list = websection.getDocumentValueList()
    self.assertNotEquals(document_list, None)
    self.assertNotEquals(len(document_list), 0)

    # Check if they have good acquisition wrappers.
    for doc in (default_document, document1, document2) + tuple(document_list):
737 738
      self.assertEqual(doc.aq_parent, websection)
      self.assertEqual(doc.aq_parent.aq_parent, website)
739

740
  def test_09_WebSiteSkinSelection(self):
741 742 743 744 745 746 747 748 749
    """Test skin selection through a Web Site.
    Check if a Web Site can change a skin selection based on a property.
    """
    portal = self.getPortal()
    ps = portal.portal_skins
    website = self.setupWebSite()

    # First, make sure that we use the default skin selection.
    portal.changeSkin(ps.getDefaultSkin())
750
    self.tic()
751 752 753 754 755 756 757 758 759 760

    # Make some skin stuff.
    if ps._getOb('test_erp5_web', None) is not None:
      ps.manage_delObjects(['test_erp5_web'])

    addFolder = ps.manage_addProduct['OFSP'].manage_addFolder
    addFolder(id='test_erp5_web')

    if ps.getSkinPath('Test ERP5 Web') is not None:
      ps.manage_skinLayers(del_skin=1, chosen=('Test ERP5 Web',))
761

762 763 764 765 766 767 768 769 770 771 772 773 774
    path = ps.getSkinPath(ps.getDefaultSkin())
    self.assertNotEquals(path, None)
    ps.manage_skinLayers(add_skin=1, skinname='Test ERP5 Web',
            skinpath=['test_erp5_web'] + path.split(','))

    # Now we need skins which don't conflict with any other.
    createZODBPythonScript(ps.erp5_web,
            'WebSite_test_13_WebSiteSkinSelection',
            '', 'return "foo"')
    createZODBPythonScript(ps.test_erp5_web,
            'WebSite_test_13_WebSiteSkinSelection',
            '', 'return "bar"')

775
    self.tic()
776 777 778 779 780 781

    path = website.absolute_url_path() + '/WebSite_test_13_WebSiteSkinSelection'
    request = portal.REQUEST

    # With the default skin.
    request['PARENTS'] = [self.app]
782
    self.assertEqual(request.traverse(path)(), 'foo')
783 784 785

    # With the test skin.
    website.setSkinSelectionName('Test ERP5 Web')
786
    self.tic()
787 788

    request['PARENTS'] = [self.app]
789
    self.assertEqual(request.traverse(path)(), 'bar')
790

791
  def test_10_getDocumentValueList(self):
792 793 794 795 796 797 798
    """Make sure that getDocumentValueList works."""
    self.setupWebSite()
    website = self.web_site_module[self.website_id]
    website.getDocumentValueList(
      portal_type='Document',
      sort_on=[('translated_portal_type', 'ascending')])

799
  def test_11_getWebSectionValueList(self):
800 801 802 803 804 805 806 807 808
    """ Check getWebSectionValueList from Web Site.
    Only visible web section should be returned.
    """
    portal = self.getPortal()
    web_site_portal_type = 'Web Site'
    web_section_portal_type = 'Web Section'
    web_page_portal_type = 'Web Page'

    # Create web site and web section
Rafael Monnerat's avatar
Rafael Monnerat committed
809
    web_site = self.web_site_module.newContent(portal_type=web_site_portal_type)
810 811 812 813
    web_section = web_site.newContent(portal_type=web_section_portal_type)
    sub_web_section = web_section.newContent(portal_type=web_section_portal_type)

    # Create a document
Rafael Monnerat's avatar
Rafael Monnerat committed
814
    web_page = self.web_page_module.newContent(portal_type=web_page_portal_type)
815 816 817 818

    # Commit transaction
    def _commit():
      portal.portal_caches.clearAllCache()
819
      self.tic()
820 821 822 823 824 825 826 827 828 829 830 831 832 833 834

    # By default, as now Web Section is visible, nothing should be returned
    _commit()
    self.assertSameSet([], web_site.getWebSectionValueList(web_page))

    # Explicitely set both web section invisible
    web_section.setVisible(0)
    sub_web_section.setVisible(0)
    _commit()
    self.assertSameSet([], web_site.getWebSectionValueList(web_page))

    # Set parent web section visible
    web_section.setVisible(1)
    sub_web_section.setVisible(0)
    _commit()
835
    self.assertSameSet([web_section],
836 837 838 839 840 841 842
                       web_site.getWebSectionValueList(web_page))

    # Set both web section visible
    # Only leaf web section is returned
    web_section.setVisible(1)
    sub_web_section.setVisible(1)
    _commit()
843
    self.assertSameSet([sub_web_section],
844 845
                       web_site.getWebSectionValueList(web_page))

Romain Courteaud's avatar
Romain Courteaud committed
846 847
    # Set leaf web section visible, which should be returned even if parent is
    # not visible
848 849 850
    web_section.setVisible(0)
    sub_web_section.setVisible(1)
    _commit()
851
    self.assertSameSet([sub_web_section],
852 853
                       web_site.getWebSectionValueList(web_page))

854
  def test_12_getWebSiteValue(self):
855
    """
856 857
      Test that getWebSiteValue() and getWebSectionValue() always
      include selected Language.
858
    """
859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876
    website_id = self.setupWebSite().getId()
    website = self.portal.restrictedTraverse(
      'web_site_module/%s' % website_id)
    website_relative_url = website.absolute_url(relative=1)
    website_fr = self.portal.restrictedTraverse(
      'web_site_module/%s/fr' % website_id)
    website_relative_url_fr = '%s/fr' % website_relative_url

    websection_id = self.setupWebSection().getId()
    websection = self.portal.restrictedTraverse(
      'web_site_module/%s/%s' % (website_id, websection_id))
    websection_relative_url = websection.absolute_url(relative=1)
    websection_fr = self.portal.restrictedTraverse(
      'web_site_module/%s/fr/%s' % (website_id, websection_id))
    websection_relative_url_fr = '%s/%s' % (website_relative_url_fr,
                                            websection.getId())

    page_ref = 'foo'
877 878 879 880
    page = self.web_page_module.newContent(portal_type='Web Page',
                                           reference='foo',
                                           text_content='<b>OK</b>')
    page.publish()
881
    self.tic()
882

883 884 885 886 887 888 889 890 891 892
    webpage = self.portal.restrictedTraverse(
      'web_site_module/%s/%s' % (website_id, page_ref))
    webpage_fr = self.portal.restrictedTraverse(
      'web_site_module/%s/fr/%s' % (website_id, page_ref))

    webpage_module = self.portal.restrictedTraverse(
      'web_site_module/%s/web_page_module' % website_id)
    webpage_module_fr = self.portal.restrictedTraverse(
      'web_site_module/%s/fr/web_page_module' % website_id)

893
    self.assertEqual(website_relative_url,
894
                      website.getWebSiteValue().absolute_url(relative=1))
895
    self.assertEqual(website_relative_url_fr,
896
                      website_fr.getWebSiteValue().absolute_url(relative=1))
897
    self.assertEqual(website_relative_url,
898
                      webpage.getWebSiteValue().absolute_url(relative=1))
899
    self.assertEqual(website_relative_url_fr,
900
                      webpage_fr.getWebSiteValue().absolute_url(relative=1))
901
    self.assertEqual(website_relative_url,
902
                      webpage_module.getWebSiteValue().absolute_url(relative=1))
903
    self.assertEqual(website_relative_url_fr,
904 905 906 907 908 909 910 911 912 913 914 915
                      webpage_module_fr.getWebSiteValue().absolute_url(relative=1))

    webpage = self.portal.restrictedTraverse(
      'web_site_module/%s/%s/%s' % (website_id, websection_id, page_ref))
    webpage_fr = self.portal.restrictedTraverse(
      'web_site_module/%s/fr/%s/%s' % (website_id, websection_id, page_ref))

    webpage_module = self.portal.restrictedTraverse(
      'web_site_module/%s/%s/web_page_module' % (website_id, websection_id))
    webpage_module_fr = self.portal.restrictedTraverse(
      'web_site_module/%s/fr/%s/web_page_module' % (website_id, websection_id))

916
    self.assertEqual(websection_relative_url,
917
                      websection.getWebSectionValue().absolute_url(relative=1))
918
    self.assertEqual(websection_relative_url_fr,
919
                      websection_fr.getWebSectionValue().absolute_url(relative=1))
920
    self.assertEqual(websection_relative_url,
921
                      webpage.getWebSectionValue().absolute_url(relative=1))
922
    self.assertEqual(websection_relative_url_fr,
923
                      webpage_fr.getWebSectionValue().absolute_url(relative=1))
924
    self.assertEqual(websection_relative_url,
925
                      webpage_module.getWebSectionValue().absolute_url(relative=1))
926
    self.assertEqual(websection_relative_url_fr,
927
                      webpage_module_fr.getWebSectionValue().absolute_url(relative=1))
928

929
  def test_13_DocumentCache(self):
930
    """
931
      Test that when a document is modified, it can be accessed through a
932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948
      web_site, a web_section or wathever and display the last content (not an
      old cache value of the document).
    """
    portal = self.getPortal()
    request = portal.REQUEST
    request['PARENTS'] = [self.app]
    website = self.setupWebSite()
    web_section_portal_type = 'Web Section'
    web_section = website.newContent(portal_type=web_section_portal_type)

    content = '<p>initial text</p>'
    new_content = '<p>modified text<p>'
    document = portal.web_page_module.newContent(portal_type='Web Page',
            id='document_cache',
            reference='NXD-Document.Cache',
            text_content=content)
    document.publish()
949
    self.tic()
950
    self.assertEqual(document.asText().strip(), 'initial text')
951

952 953 954 955 956 957 958
    # First make sure conversion already exists on the web site
    web_document = website.restrictedTraverse('NXD-Document.Cache')
    self.assertTrue(web_document.hasConversion(format='txt'))
    web_document = web_section.restrictedTraverse('NXD-Document.Cache')
    self.assertTrue(web_document.hasConversion(format='txt'))

    # Through the web_site.
959
    path = website.absolute_url_path() + '/NXD-Document.Cache'
960
    response = self.publish(path, self.credential)
961
    self.assertNotEquals(response.getBody().find(content), -1)
962

963
    # Through a web_section.
964
    path = web_section.absolute_url_path() + '/NXD-Document.Cache'
965
    response = self.publish(path, self.credential)
966
    self.assertNotEquals(response.getBody().find(content), -1)
967 968 969

    # modified the web_page content
    document.edit(text_content=new_content)
970
    self.assertEqual(document.asText().strip(), 'modified text')
971
    self.tic()
972

973 974
    # check the cache doesn't send again the old content
    # Through the web_site.
975
    path = website.absolute_url_path() + '/NXD-Document.Cache'
976 977
    response = self.publish(path, self.credential)
    self.assertNotEquals(response.getBody().find(new_content), -1)
978

979
    # Through a web_section.
980
    path = web_section.absolute_url_path() + '/NXD-Document.Cache'
981 982
    response = self.publish(path, self.credential)
    self.assertNotEquals(response.getBody().find(new_content), -1)
983

984
  def test_13a_DocumentMovedCache(self):
985
    """
986
      What happens to the cache if document is moved
987
      with a new ID. Can we still access content,
988
      or is the cache emptied. There is no reason
989
      that the cache should be regenerated or that the
990
      previous cache would not be emptied.
991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008

      Here, we test that the cache is not regenerated,
      not emptied, and still available.
    """
    portal = self.getPortal()
    request = portal.REQUEST
    request['PARENTS'] = [self.app]
    website = self.setupWebSite()
    web_section_portal_type = 'Web Section'
    web_section = website.newContent(portal_type=web_section_portal_type)

    content = '<p>initial text</p>'
    new_content = '<p>modified text<p>'
    document = portal.web_page_module.newContent(portal_type='Web Page',
            id='document_original_cache',
            reference='NXD-Document.Cache',
            text_content=content)
    document.publish()
1009
    self.tic()
1010
    self.assertEqual(document.asText().strip(), 'initial text')
1011 1012 1013

    # Make sure document cache keeps converted content even if ID changes
    self.assertTrue(document.hasConversion(format='txt'))
1014
    document.setId('document_new_cache')
1015
    self.assertTrue(document.hasConversion(format='txt'))
1016
    document.setId('document_original_cache')
1017 1018
    self.assertTrue(document.hasConversion(format='txt'))

1019
  def test_13b_DocumentEditCacheKey(self):
1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034
    """
      What happens if a web page is edited on a web site ?
      Is converted content cleared and updated ? Or
      is a wrong cache key created ? Here, we make sure
      that the content is updated and the cache cleared
      and reset.
    """
    portal = self.getPortal()
    request = portal.REQUEST
    request['PARENTS'] = [self.app]
    website = self.setupWebSite()
    web_section_portal_type = 'Web Section'
    web_section = website.newContent(portal_type=web_section_portal_type)

    content = '<p>initial text</p>'
1035
    new_content = '<p>modified text</p>'
1036 1037 1038 1039 1040
    document = portal.web_page_module.newContent(portal_type='Web Page',
            id='document_cache',
            reference='NXD-Document.Cache',
            text_content=content)
    document.publish()
1041
    self.tic()
1042
    self.assertEqual(document.asText().strip(), 'initial text')
1043 1044 1045

    # Through the web_site.
    path = website.absolute_url_path() + '/NXD-Document.Cache'
1046
    response = self.publish(path, self.credential)
1047
    self.assertNotEquals(response.getBody().find(content), -1)
1048 1049
    # Through a web_section.
    path = web_section.absolute_url_path() + '/NXD-Document.Cache'
1050
    response = self.publish(path, self.credential)
1051
    self.assertNotEquals(response.getBody().find(content), -1)
1052 1053 1054

    # Modify the web_page content
    # Use unrestrictedTraverse (XXX-JPS reason unknown)
1055
    web_document = website.unrestrictedTraverse('web_page_module/%s' % document.getId())
1056 1057 1058 1059 1060
    web_document.edit(text_content=new_content)
    # Make sure cached is emptied
    self.assertFalse(web_document.hasConversion(format='txt'))
    self.assertFalse(document.hasConversion(format='txt'))
    # Make sure cache is regenerated
1061
    self.assertEqual(web_document.asText().strip(), 'modified text')
1062
    self.tic()
1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073

    # First make sure conversion already exists (since it should
    # have been generated previously)
    self.assertTrue(document.hasConversion(format='txt'))
    web_document = web_section.restrictedTraverse('NXD-Document.Cache')
    self.assertTrue(web_document.hasConversion(format='txt'))
    web_document = website.restrictedTraverse('NXD-Document.Cache')
    self.assertTrue(web_document.hasConversion(format='txt'))

    # check the cache doesn't send again the old content
    # test this fist on the initial document
1074
    self.assertEqual(document.asText().strip(), 'modified text')
1075 1076 1077

    # Through a web_section.
    web_document = web_section.restrictedTraverse('NXD-Document.Cache')
1078
    self.assertEqual(web_document.asText().strip(), 'modified text')
1079
    path = web_section.absolute_url_path() + '/NXD-Document.Cache'
1080
    response = self.publish(path, self.credential)
1081
    self.assertNotEquals(response.getBody().find(new_content), -1)
1082 1083 1084

    # Through a web_site.
    web_document = website.restrictedTraverse('NXD-Document.Cache')
1085
    self.assertEqual(web_document.asText().strip(), 'modified text')
1086
    path = website.absolute_url_path() + '/NXD-Document.Cache'
1087
    response = self.publish(path, self.credential)
1088
    self.assertNotEquals(response.getBody().find(new_content), -1)
1089

1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108
  def test_14_AccessWebSiteForWithDifferentUserPreferences(self):
    """Check that Ram Cache Manager do not mix websection
    rendering between users.
    This test enable different preferences per users and check that
    those preferences doesn't affect rendering for other users.
    user          | preference

    administrator | developper_mode activated
    webeditor     | translator_mode activated
    anonymous     | developper_mode & translator_mode disabled

    The Signature used to detect enabled preferences in HTML Body are
    manage_main for developper_mode
    manage_messages for translator_mode
    """
    user = self.createUser('administrator')
    self.createUserAssignement(user, {})
    user = self.createUser('webeditor')
    self.createUserAssignement(user, {})
1109
    self.tic()
1110 1111 1112
    preference_tool = self.getPreferenceTool()
    isTransitionPossible = self.portal.portal_workflow.isTransitionPossible

1113
    # create or edit preference for administrator
1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126
    administrator_preference = self.portal.portal_catalog.getResultValue(
                                                 portal_type='Preference',
                                                 owner='administrator')
    if administrator_preference is None:
      self.login('administrator')
      administrator_preference = preference_tool.newContent(
                                              portal_type='Preference')
    if isTransitionPossible(administrator_preference, 'enable_action'):
      administrator_preference.enable()

    administrator_preference.setPreferredHtmlStyleDevelopperMode(True)
    administrator_preference.setPreferredHtmlStyleTranslatorMode(False)

1127
    # create or edit preference for webeditor
1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140
    webeditor_preference = self.portal.portal_catalog.getResultValue(
                                                  portal_type='Preference',
                                                  owner='webeditor')
    if webeditor_preference is None:
      self.login('webeditor')
      webeditor_preference = preference_tool.newContent(
                                              portal_type='Preference')
    if isTransitionPossible(webeditor_preference, 'enable_action'):
      webeditor_preference.enable()

    webeditor_preference.setPreferredHtmlStyleDevelopperMode(False)
    webeditor_preference.setPreferredHtmlStyleTranslatorMode(True)
    self.login()
1141
    self.tic()
1142 1143 1144 1145 1146 1147

    web_site = self.setupWebSite()
    websection = self.setupWebSection()

    websection_url = '%s/%s' % (self.portal.getId(), websection.getRelativeUrl())

1148
    # connect as administrator and check that only developper_mode is enable
1149 1150 1151 1152
    response = self.publish(websection_url, 'administrator:administrator')
    self.assertTrue('manage_main' in response.getBody())
    self.assertTrue('manage_messages' not in response.getBody())

1153
    # connect as webeditor and check that only translator_mode is enable
1154 1155 1156 1157
    response = self.publish(websection_url, 'webeditor:webeditor')
    self.assertTrue('manage_main' not in response.getBody())
    self.assertTrue('manage_messages' in response.getBody())

1158
    # anonymous user doesn't exists, check anonymous access without preferences
1159 1160 1161 1162
    response = self.publish(websection_url, 'anonymous:anonymous')
    self.assertTrue('manage_main' not in response.getBody())
    self.assertTrue('manage_messages' not in response.getBody())

1163
  def test_15_Check_LastModified_Header(self):
Nicolas Delaby's avatar
Nicolas Delaby committed
1164
    """Checks that Last-Modified header set by caching policy manager
1165
    is correctly filled with getModificationDate of content.
1166
    This test check 2 Policy installed by erp5_web:
1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179
    Policy ID - unauthenticated web pages
                authenticated
    """
    website = self.setupWebSite()
    web_section_portal_type = 'Web Section'
    web_section = website.newContent(portal_type=web_section_portal_type)

    content = '<p>initial text</p>'
    document = self.portal.web_page_module.newContent(portal_type='Web Page',
            id='document_cache',
            reference='NXD-Document.Cache',
            text_content=content)
    document.publish()
1180
    self.tic()
1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200
    path = website.absolute_url_path() + '/NXD-Document.Cache'
    # test Different Policy installed by erp5_web
    # unauthenticated web pages
    response = self.publish(path)
    last_modified_header = response.getHeader('Last-Modified')
    self.assertTrue(last_modified_header)
    from App.Common import rfc1123_date
    # Convert the Date into string according RFC 1123 Time Format
    modification_date = rfc1123_date(document.getModificationDate())
    self.assertEqual(modification_date, last_modified_header)

    # authenticated
    user = self.createUser('webmaster')
    self.createUserAssignement(user, {})
    response = self.publish(path, 'webmaster:webmaster')
    last_modified_header = response.getHeader('Last-Modified')
    self.assertTrue(last_modified_header)
    # Convert the Date into string according RFC 1123 Time Format
    modification_date = rfc1123_date(document.getModificationDate())
    self.assertEqual(modification_date, last_modified_header)
1201

1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216
  def test_15a_CheckCachingPolicyManager(self):
    """
    Check if caching_policy_manager is well applied even if custom
    render method is used in Web Site or Web Section.
    """
    web_site = self.setupWebSite()
    web_section_portal_type = 'Web Section'
    web_section = web_site.newContent(portal_type=web_section_portal_type)
    self.assertTrue(self.publish(web_site.absolute_url_path()).getHeader('X-Cache-Headers-Set-By'))
    web_site.setCustomRenderMethodId('WebSection_viewAsWeb')
    self.assertTrue(self.publish(web_site.absolute_url_path()).getHeader('X-Cache-Headers-Set-By'))
    self.assertTrue(self.publish(web_section.absolute_url_path()).getHeader('X-Cache-Headers-Set-By'))
    web_section.setCustomRenderMethodId('WebSection_viewAsWeb')
    self.assertTrue(self.publish(web_section.absolute_url_path()).getHeader('X-Cache-Headers-Set-By'))

1217
  def test_16_404ErrorPageIsReturned(self):
1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234
    """
      Test that when we try to access a non existing url trought a web site, a
      404 error page is returned
    """
    portal = self.getPortal()
    request = portal.REQUEST
    request['PARENTS'] = [self.app]
    website = self.setupWebSite()
    path = website.absolute_url_path() + '/a_non_existing_page'
    absolute_url = website.absolute_url() + '/a_non_existing_page'
    request = portal.REQUEST

    # Check a Not Found page is returned
    self.assertTrue('Not Found' in request.traverse(path)())
    # Check that we try to display a page with 404.error.page reference
    self.assertEqual(request.traverse(path).absolute_url().split('/')[-1],
    '404.error.page')
1235

1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251
  @expectedFailure
  def test_17_WebSectionEditionWithLanguageInURL(self):
    """
    Check that editing a web section with the language in the URL
    does not prevent indexation.

    - Create a web site
    - Activate the language in the URL
    - Create a web section
    - Access it using another language and edit it
    - Check that web section is correctly indexed
    """
    language = 'de'

    website = self.setupWebSite()
    # Check that language in defined in the URL
1252
    self.assertEqual(True, website.getStaticLanguageSelection())
1253 1254 1255
    self.assertNotEquals(language, website.getDefaultAvailableLanguage())

    websection = self.setupWebSection()
1256
    self.assertEqual(websection.getId(), websection.getTitle())
1257

1258
    self.tic()
1259
    response = self.publish('/%s/%s/%s/%s/Base_editAndEditAsWeb' % \
Rafael Monnerat's avatar
Rafael Monnerat committed
1260
                    (self.portal.getId(), website.getRelativeUrl(),
1261 1262 1263
                     language, websection.getId()),
                     basic='ERP5TypeTestCase:',
                     request_method='POST',
1264
                     stdin=StringIO(urlencode({
1265 1266 1267 1268 1269 1270
                       'form_id': 'WebSection_view',
                       'form_action': 'Base_edit',
                       'edit_document_url': '%s/%s/%s/WebSection_view' % \
                           (website.absolute_url(), language,
                             websection.getId()),
                       'field_my_title': '%s_edited' % websection.getId(),
1271
                     }))
1272
                    )
Rafael Monnerat's avatar
Rafael Monnerat committed
1273

1274
    self.assertEqual(MOVED_TEMPORARILY, response.getStatus())
1275 1276 1277
    new_location = response.getHeader('Location')
    new_location = new_location.split('/', 3)[-1]

1278
    self.tic()
1279 1280

    response = self.publish(new_location, basic='ERP5TypeTestCase:',)
1281 1282
    self.assertEqual(HTTP_OK, response.getStatus())
    self.assertEqual('text/html; charset=utf-8',
1283 1284 1285
                      response.getHeader('content-type'))
    self.assertTrue("Data updated." in response.getBody())

1286
    self.tic()
1287

1288 1289
    self.assertEqual('%s_edited' % websection.getId(), websection.getTitle())
    self.assertEqual(1, len(self.portal.portal_catalog(
Rafael Monnerat's avatar
Rafael Monnerat committed
1290
                                    relative_url=websection.getRelativeUrl(),
1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307
                                    title=websection.getTitle())))

  @expectedFailure
  def test_18_WebSiteEditionWithLanguageInURL(self):
    """
    Check that editing a web section with the language in the URL
    does not prevent indexation.

    - Create a web site
    - Activate the language in the URL
    - Access it using another language and edit it
    - Check that web site is correctly modified
    """
    language = 'de'

    website = self.setupWebSite()
    # Check that language in defined in the URL
1308
    self.assertEqual(True, website.getStaticLanguageSelection())
1309 1310
    self.assertNotEquals(language, website.getDefaultAvailableLanguage())

1311
    self.assertEqual(website.getId(), website.getTitle())
1312

1313
    self.tic()
1314 1315

    response = self.publish('/%s/%s/%s/Base_editAndEditAsWeb' % \
Rafael Monnerat's avatar
Rafael Monnerat committed
1316
                    (self.portal.getId(), website.getRelativeUrl(),
1317 1318 1319
                     language),
                     basic='ERP5TypeTestCase:',
                     request_method='POST',
1320
                     stdin=StringIO(urlencode({
1321 1322 1323 1324 1325 1326
                       'form_id': 'WebSite_view',
                       'form_action': 'Base_edit',
                       'edit_document_url': '%s/%s/WebSite_view' % \
                           (website.absolute_url(), language),
                       'field_my_title': '%s_edited' % website.getId(),
                       'field_my_id': language,
1327
                     }))
1328
                    )
Rafael Monnerat's avatar
Rafael Monnerat committed
1329

1330
    self.assertEqual(MOVED_TEMPORARILY, response.getStatus())
1331 1332 1333
    new_location = response.getHeader('Location')
    new_location = new_location.split('/', 3)[-1]

1334
    self.tic()
1335 1336

    response = self.publish(new_location, basic='ERP5TypeTestCase:',)
1337 1338
    self.assertEqual(HTTP_OK, response.getStatus())
    self.assertEqual('text/html; charset=utf-8',
1339 1340 1341
                      response.getHeader('content-type'))
    self.assertTrue("Data updated." in response.getBody())

1342
    self.tic()
1343

1344 1345
    self.assertEqual('%s_edited' % website.getId(), website.getTitle())
    self.assertEqual(1, len(self.portal.portal_catalog(
Rafael Monnerat's avatar
Rafael Monnerat committed
1346
                                    relative_url=website.getRelativeUrl(),
1347 1348
                                    title=website.getTitle())))

1349 1350
  def test_19_WebModeAndEditableMode(self):
    """
Rafael Monnerat's avatar
Rafael Monnerat committed
1351
    Check if isWebMode & isEditableMode API works.
1352 1353 1354
    """
    request = self.app.REQUEST
    website = self.setupWebSite()
Rafael Monnerat's avatar
Rafael Monnerat committed
1355

1356
    # web mode
1357 1358 1359
    self.assertEqual(False, self.portal.person_module.isWebMode())
    self.assertEqual(True, website.isWebMode())
    self.assertEqual(True, getattr(website, 'person_module').isWebMode())
Rafael Monnerat's avatar
Rafael Monnerat committed
1360

1361
    # editable mode
1362 1363 1364
    self.assertEqual(False, self.portal.person_module.isEditableMode())
    self.assertEqual(False, website.isEditableMode())
    self.assertEqual(False, getattr(website, 'person_module').isEditableMode())
Rafael Monnerat's avatar
Rafael Monnerat committed
1365

1366
    request.set('editable_mode', 1)
1367 1368 1369
    self.assertEqual(1, self.portal.person_module.isEditableMode())
    self.assertEqual(1, website.isEditableMode())
    self.assertEqual(1, getattr(website, 'person_module').isEditableMode())
1370 1371 1372

  def test_20_reStructuredText(self):
    web_page = self.portal.web_page_module.newContent(portal_type='Web Page',
1373 1374
                                                      content_type='text/x-rst')
    web_page.edit(text_content="`foo`")
1375 1376 1377
    self.assertTrue('<cite>foo</cite>' in web_page.asEntireHTML(charset='utf-8'))
    self.assertTrue('<cite>foo</cite>' in web_page.asEntireHTML())

Ivan Tyagov's avatar
Ivan Tyagov committed
1378 1379 1380 1381 1382 1383
  def test_21_WebSiteMap(self):
    """
      Test Web Site map script.
    """
    request = self.app.REQUEST
    website = self.setupWebSite()
Rafael Monnerat's avatar
Rafael Monnerat committed
1384 1385 1386
    kw = {'depth': 5,
          'include_subsection': 1}

Ivan Tyagov's avatar
Ivan Tyagov committed
1387 1388
    website.setSiteMapSectionParent(1)
    websection1 = website.newContent(portal_type='Web Section',
Rafael Monnerat's avatar
Rafael Monnerat committed
1389 1390
                                     title='Section 1',
                                     site_map_section_parent=1,
Ivan Tyagov's avatar
Ivan Tyagov committed
1391 1392
                                     visible=1)
    websection1_1 = websection1.newContent(portal_type='Web Section',
Rafael Monnerat's avatar
Rafael Monnerat committed
1393 1394 1395
                                     title='Section 1.1',
                                     site_map_section_parent=1,
                                     visible=1)
1396
    self.tic()
Rafael Monnerat's avatar
Rafael Monnerat committed
1397 1398
    site_map = website.WebSection_getSiteMapTree(depth=5, include_subsection=1)
    self.assertSameSet([websection1.getTitle()],
Ivan Tyagov's avatar
Ivan Tyagov committed
1399
                       [x['translated_title'] for x in site_map])
Rafael Monnerat's avatar
Rafael Monnerat committed
1400
    self.assertSameSet([websection1_1.getTitle()],
Ivan Tyagov's avatar
Ivan Tyagov committed
1401
                       [x['translated_title'] for x in site_map[0]['subsection']])
Rafael Monnerat's avatar
Rafael Monnerat committed
1402 1403 1404
    self.assertEqual(1, site_map[0]['level'])
    self.assertEqual(2, site_map[0]['subsection'][0]['level'])

Ivan Tyagov's avatar
Ivan Tyagov committed
1405
    # check depth works
Rafael Monnerat's avatar
Rafael Monnerat committed
1406
    site_map = website.WebSection_getSiteMapTree(depth=1, include_subsection=1)
Ivan Tyagov's avatar
Ivan Tyagov committed
1407
    self.assertEqual(None, site_map[0]['subsection'])
Rafael Monnerat's avatar
Rafael Monnerat committed
1408
    self.assertSameSet([websection1.getTitle()],
Ivan Tyagov's avatar
Ivan Tyagov committed
1409
                       [x['translated_title'] for x in site_map])
Rafael Monnerat's avatar
Rafael Monnerat committed
1410

Ivan Tyagov's avatar
Ivan Tyagov committed
1411 1412
    # hide subsections
    websection1_1.setSiteMapSectionParent(0)
Rafael Monnerat's avatar
Rafael Monnerat committed
1413
    websection1_1.setVisible(0)
1414
    self.tic()
Rafael Monnerat's avatar
Rafael Monnerat committed
1415 1416
    site_map = website.WebSection_getSiteMapTree(depth=5, include_subsection=1)
    self.assertSameSet([websection1.getTitle()],
Ivan Tyagov's avatar
Ivan Tyagov committed
1417
                       [x['translated_title'] for x in site_map])
Rafael Monnerat's avatar
Rafael Monnerat committed
1418
    self.assertEqual(None, site_map[0]['subsection'])
Ivan Tyagov's avatar
Ivan Tyagov committed
1419

1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488 1489
  def test_future_publication_date_not_visible(self):
    portal = self.portal
    reference = 'test_future_publication_date_not_visible'
    newContent = portal.web_page_module.newContent
    date1 = DateTime()
    date2 = date1 + 1
    date3 = date2 + 1
    date4 = date3 + 1

    def new(**kw):
      result = newContent(
        portal_type='Web Page',
        reference=reference,
        **kw
      )
      self.tic()
      result.publish()
      self.tic()
      return result
    document1 = new(version=1)
    document2 = new(version=2, effective_date=date2)
    # Some more documents which should never be visible.
    # Bind them to local variables for quicker debugging, if needed.
    # Later than document2.
    document3 = new(version=3, effective_date=date4)
    # Like document1, but not published
    document4 = newContent(
      portal_type='Web Page',
      reference=reference,
    )
    # Like document2, but not published
    document5 = newContent(
      portal_type='Web Page',
      reference=reference,
      effective_date=date2,
    )
    self.tic()
    site = portal.web_site_module.newContent(
      portal_type='Web Site',
    )
    site.publish()
    section = site.newContent(
      portal_type='Web Section',
    )
    section.setCriterionProperty('reference')
    section.setCriterion(
      'reference',
      max='',
      identity=[reference],
      min='',
    )
    self.tic()

    self.assertEqual(document1.getValidationState(), 'published')
    self.assertEqual(document2.getValidationState(), 'published')
    self.assertEqual(document3.getValidationState(), 'published')
    self.assertEqual(document4.getValidationState(), 'draft')
    self.assertEqual(document5.getValidationState(), 'draft')
    def check(expected_document, date):
      document = section.WebSection_getDocumentValue(reference, now=date)
      self.assertNotEqual(document, None)
      self.assertEqual(document.getPath(), expected_document.getPath())
      document_list = section.WebSection_getDocumentValueList(now=date)
      self.assertEqual(len(document_list), 1)
      self.assertEqual(document_list[0].getPath(), expected_document.getPath())
    # document1 is visible & listed before date2
    check(document1, date1)
    # document2 is visible & listed at and above date2
    check(document2, date2)
    check(document2, date3)
1490

1491 1492 1493 1494 1495 1496 1497
class TestERP5WebWithSimpleSecurity(ERP5TypeTestCase):
  """
  Test for erp5_web with simple security.
  """

  def getBusinessTemplateList(self):
    return ('erp5_base',
1498
            'erp5_pdm',
1499 1500
            'erp5_trade',
            'erp5_project',
1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511
            'erp5_web',
            )

  def getTitle(self):
    return "Web"

  def createUser(self, name, role_list):
    user_folder = self.getPortal().acl_users
    user_folder._doAddUser(name, 'password', role_list, [])

  def afterSetUp(self):
1512
    self.web_site_module = self.portal.web_site_module
1513 1514 1515
    self.portal.Localizer = DummyLocalizer()
    self.createUser('admin', ['Manager'])
    self.createUser('erp5user', ['Auditor', 'Author'])
1516
    self.createUser('webmaster', ['Assignor'])
1517 1518
    self.tic()

1519 1520 1521 1522
  def clearModule(self, module):
    module.manage_delObjects(list(module.objectIds()))
    self.tic()

1523
  def beforeTearDown(self):
1524 1525
    self.clearModule(self.portal.web_site_module)
    self.clearModule(self.portal.web_page_module)
1526

1527
  def test_01_AccessWebPageByReference(self):
1528
    self.login('admin')
1529 1530 1531 1532 1533 1534 1535 1536 1537 1538 1539
    site = self.portal.web_site_module.newContent(portal_type='Web Site',
                                                  id='site')
    section = site.newContent(portal_type='Web Section', id='section')

    self.tic()

    section.setCriterionProperty('portal_type')
    section.setCriterion('portal_type', max='', identity=['Web Page'], min='')

    self.tic()

1540
    self.login('erp5user')
1541 1542 1543 1544 1545 1546 1547 1548 1549 1550 1551 1552 1553 1554 1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571
    page_en = self.portal.web_page_module.newContent(portal_type='Web Page')
    page_en.edit(reference='my-first-web-page',
                 language='en',
                 version='1',
                 text_format='text/plain',
                 text_content='Hello, World!')

    self.tic()

    page_en.publish()

    self.tic()

    page_ja = self.portal.web_page_module.newContent(portal_type='Web Page')
    page_ja.edit(reference='my-first-web-page',
                 language='ja',
                 version='1',
                 text_format='text/plain',
                 text_content='こんにちは、世界!')

    self.tic()

    page_ja.publish()

    self.tic()

    # By Anonymous
    self.logout()

    self.portal.Localizer.changeLanguage('en')

1572
    target = self.portal.unrestrictedTraverse('web_site_module/site/section/my-first-web-page')
1573 1574 1575 1576
    self.assertEqual('Hello, World!', target.getTextContent())

    self.portal.Localizer.changeLanguage('ja')

1577
    target = self.portal.unrestrictedTraverse('web_site_module/site/section/my-first-web-page')
1578
    self.assertEqual('こんにちは、世界!', target.getTextContent())
1579

1580
  def test_02_LocalRolesFromRoleDefinition(self):
1581 1582 1583 1584 1585 1586
    """ Test setting local roles on Web Site/ Web Sectio using ERP5 Role Definition objects . """
    portal = self.portal
    person_reference = 'webuser'
    site = portal.web_site_module.newContent(portal_type='Web Site',
                                                  id='site')
    section = site.newContent(portal_type='Web Section', id='section')
1587
    person = portal.person_module.newContent(portal_type = 'Person',
1588
                                             reference = person_reference)
1589
    # add Role Definition for site and section
1590 1591
    site_role_definition = site.newContent(portal_type = 'Role Definition',
                                           role_name = 'Assignee',
1592
                                           agent = person.getRelativeUrl())
1593 1594
    section_role_definition = section.newContent(portal_type = 'Role Definition',
                                                 role_name = 'Associate',
1595
                                                 agent = person.getRelativeUrl())
1596 1597
    self.tic()
    # check if Role Definition have create local roles
1598
    self.assertSameSet(('Assignee',),
1599
                          site.get_local_roles_for_userid(person_reference))
1600
    self.assertSameSet(('Associate',),
1601
                          section.get_local_roles_for_userid(person_reference))
1602 1603
    self.assertRaises(Unauthorized, site_role_definition.edit,
                      role_name='Manager')
1604

1605 1606 1607 1608
    # delete Role Definition and check again (local roles must be gone too)
    site.manage_delObjects(site_role_definition.getId())
    section.manage_delObjects(section_role_definition.getId())
    self.tic()
1609
    self.assertSameSet((),
1610 1611 1612 1613
                       site.get_local_roles_for_userid(person_reference))
    self.assertSameSet((),
                       section.get_local_roles_for_userid(person_reference))

1614
  def test_03_WebSection_getDocumentValueListSecurity(self):
1615
    """ Test WebSection_getDocumentValueList behaviour and security"""
1616
    self.login('admin')
1617
    site = self.portal.web_site_module.newContent(portal_type='Web Site',
1618
                                      id='site')
1619
    site.publish()
1620

1621
    section = site.newContent(portal_type='Web Section',
1622 1623 1624 1625 1626
                              id='section')

    self.tic()

    section.setCriterionProperty('portal_type')
1627
    section.setCriterion('portal_type', max='',
1628 1629 1630 1631
                         identity=['Web Page'], min='')

    self.tic()

1632
    self.login('erp5user')
1633 1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660
    page_en_0 = self.portal.web_page_module.newContent(portal_type='Web Page')
    page_en_0.edit(reference='my-first-web-page',
                 language='en',
                 version='1',
                 text_format='text/plain',
                 text_content='Hello, World!')

    page_en_1 = self.portal.web_page_module.newContent(portal_type='Web Page')
    page_en_1.edit(reference='my-first-web-page',
                 language='en',
                 version='2',
                 text_format='text/plain',
                 text_content='Hello, World!')

    page_en_2 = self.portal.web_page_module.newContent(portal_type='Web Page')
    page_en_2.edit(reference='my-second-web-page',
                 language='en',
                 version='2',
                 text_format='text/plain',
                 text_content='Hello, World!')

    page_jp_0 = self.portal.web_page_module.newContent(portal_type='Web Page')
    page_jp_0.edit(reference='my-first-japonese-page',
                 language='jp',
                 version='1',
                 text_format='text/plain',
                 text_content='Hello, World!')

1661
    self.commit()
1662
    self.login('erp5user')
1663 1664
    self.tic()
    self.portal.Localizer.changeLanguage('en')
1665

1666
    self.assertEqual(0, len(section.WebSection_getDocumentValueList()))
1667

1668
    self.login('erp5user')
1669 1670 1671 1672
    page_en_0.publish()
    self.tic()

    self.portal.Localizer.changeLanguage('en')
1673 1674
    self.assertEqual(1, len(section.WebSection_getDocumentValueList()))
    self.assertEqual(page_en_0.getUid(),
1675 1676 1677
                      section.WebSection_getDocumentValueList()[0].getUid())

    self.portal.Localizer.changeLanguage('jp')
1678
    self.assertEqual(0, len(section.WebSection_getDocumentValueList()))
1679 1680 1681 1682

    # By Anonymous
    self.logout()
    self.portal.Localizer.changeLanguage('en')
1683 1684
    self.assertEqual(1, len(section.WebSection_getDocumentValueList()))
    self.assertEqual(page_en_0.getUid(),
1685 1686
                      section.WebSection_getDocumentValueList()[0].getUid())
    self.portal.Localizer.changeLanguage('jp')
1687
    self.assertEqual(0, len(section.WebSection_getDocumentValueList()))
1688 1689

    # Second Object
1690
    self.login('erp5user')
1691 1692 1693 1694
    page_en_1.publish()
    self.tic()

    self.portal.Localizer.changeLanguage('en')
1695 1696
    self.assertEqual(1, len(section.WebSection_getDocumentValueList()))
    self.assertEqual(page_en_1.getUid(),
1697 1698
                      section.WebSection_getDocumentValueList()[0].getUid())
    self.portal.Localizer.changeLanguage('jp')
1699
    self.assertEqual(0, len(section.WebSection_getDocumentValueList()))
1700 1701 1702 1703

    # By Anonymous
    self.logout()
    self.portal.Localizer.changeLanguage('en')
1704 1705
    self.assertEqual(1, len(section.WebSection_getDocumentValueList()))
    self.assertEqual(page_en_1.getUid(),
1706 1707 1708
                      section.WebSection_getDocumentValueList()[0].getUid())

    # Trird Object
1709
    self.login('erp5user')
1710 1711 1712 1713
    page_en_2.publish()
    self.tic()

    self.portal.Localizer.changeLanguage('en')
1714
    self.assertEqual(2, len(section.WebSection_getDocumentValueList()))
1715
    self.portal.Localizer.changeLanguage('jp')
1716
    self.assertEqual(0, len(section.WebSection_getDocumentValueList()))
1717 1718 1719 1720

    # By Anonymous
    self.logout()
    self.portal.Localizer.changeLanguage('en')
1721
    self.assertEqual(2, len(section.WebSection_getDocumentValueList()))
1722
    self.portal.Localizer.changeLanguage('jp')
1723
    self.assertEqual(0, len(section.WebSection_getDocumentValueList()))
1724

1725
    # First Japanese Object
1726
    self.login('erp5user')
1727 1728 1729 1730
    page_jp_0.publish()
    self.tic()

    self.portal.Localizer.changeLanguage('en')
1731
    self.assertEqual(2, len(section.WebSection_getDocumentValueList()))
1732
    self.portal.Localizer.changeLanguage('jp')
1733
    self.assertEqual(1, len(section.WebSection_getDocumentValueList()))
1734 1735 1736 1737

    # By Anonymous
    self.logout()
    self.portal.Localizer.changeLanguage('en')
1738
    self.assertEqual(2, len(section.WebSection_getDocumentValueList()))
1739
    self.portal.Localizer.changeLanguage('jp')
1740 1741
    self.assertEqual(1, len(section.WebSection_getDocumentValueList()))
    self.assertEqual(page_jp_0.getUid(),
1742
                      section.WebSection_getDocumentValueList()[0].getUid())
1743

1744
  def test_04_ExpireUserAction(self):
1745
    """ Test the expire user action"""
1746
    self.login('admin')
1747
    site = self.portal.web_site_module.newContent(portal_type='Web Site', id='site')
1748 1749 1750 1751 1752 1753 1754 1755

    # create websections in a site and in anothers web sections
    section_1 = site.newContent(portal_type='Web Section', id='section_1')
    section_2 = site.newContent(portal_type='Web Section', id='section_2')
    section_3 = site.newContent(portal_type='Web Section', id='section_3')
    section_4 = site.newContent(portal_type='Web Section', id='section_4')
    section_5 = section_3.newContent(portal_type='Web Section', id='section_5')
    section_6 = section_4.newContent(portal_type='Web Section', id='section_6')
1756 1757
    self.tic()

1758 1759 1760 1761 1762 1763
    # test if a manager can expire them
    try:
      section_1.expire()
      section_5.expire()
    except Unauthorized:
      self.fail("Admin should be able to expire a Web Section.")
1764

1765
    # test if a user (ASSIGNOR) can expire them
1766
    self.login('webmaster')
1767 1768 1769 1770 1771
    try:
      section_2.expire()
      section_6.expire()
    except Unauthorized:
      self.fail("An user should be able to expire a Web Section.")
1772

1773
  def test_05_createWebSite(self):
1774
    """ Test to create or clone web sites with many users """
1775
    self.login('admin')
1776
    web_site_module = self.portal.web_site_module
1777 1778 1779 1780 1781 1782 1783 1784

    # test for admin
    try:
      site_1 = web_site_module.newContent(portal_type='Web Site', id='site_1')
    except Unauthorized:
      self.fail("Admin should be able to create a Web Site.")

    # test as a web user (assignor)
1785
    self.login('webmaster')
1786 1787 1788 1789 1790
    try:
      site_2 = web_site_module.newContent(portal_type='Web Site', id='site_2')
    except Unauthorized:
      self.fail("A webmaster should be able to create a Web Site.")

1791 1792 1793
    site_2_copy = web_site_module.manage_copyObjects(ids=(site_2.getId(),))
    site_2_clone = web_site_module[web_site_module.manage_pasteObjects(
      site_2_copy)[0]['new_id']]
1794
    self.assertEqual(site_2_clone.getPortalType(), 'Web Site')
1795

1796
  def test_06_createWebSection(self):
1797
    """ Test to create or clone web sections with many users """
1798
    self.login('admin')
1799
    site = self.portal.web_site_module.newContent(portal_type='Web Site', id='site')
1800

1801 1802 1803 1804 1805 1806 1807
    # test for admin
    try:
      section_1 = site.newContent(portal_type='Web Section', id='section_1')
      section_2 = section_1.newContent(portal_type='Web Section', id='section_2')
    except Unauthorized:
      self.fail("Admin should be able to create a Web Section.")

1808
    # test as a webmaster (assignor)
1809
    self.login('webmaster')
1810 1811 1812 1813
    try:
      section_2 = site.newContent(portal_type='Web Section', id='section_2')
      section_3 = section_2.newContent(portal_type='Web Section', id='section_3')
    except Unauthorized:
1814
      self.fail("A webmaster should be able to create a Web Section.")
1815 1816 1817
    section_2_copy = site.manage_copyObjects(ids=(section_2.getId(),))
    section_2_clone = site[site.manage_pasteObjects(
      section_2_copy)[0]['new_id']]
1818
    self.assertEqual(section_2_clone.getPortalType(), 'Web Section')
1819 1820 1821
    section_3_copy = section_2.manage_copyObjects(ids=(section_3.getId(),))
    section_3_clone = section_2[section_2.manage_pasteObjects(
      section_3_copy)[0]['new_id']]
1822
    self.assertEqual(section_3_clone.getPortalType(), 'Web Section')
1823

1824
  def test_07_createCategory(self):
1825
    """ Test to create or clone categories with many users """
1826
    self.login('admin')
1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839 1840 1841 1842
    portal_categories = self.portal.portal_categories
    publication_section = portal_categories.publication_section

    # test for admin
    try:
      base_category_1 = portal_categories.newContent(portal_type='Base Category', id='base_category_1')
    except Unauthorized:
      self.fail("Admin should be able to create a Base Category.")
    try:
      category_1 = publication_section.newContent(portal_type='Category', id='category_1')
      category_2 = category_1.newContent(portal_type='Category', id='category_3')
    except Unauthorized:
      self.fail("Admin should be able to create a Category.")
    category_1_copy = publication_section.manage_copyObjects(ids=(category_1.getId(),))
    category_1_clone = publication_section[publication_section.manage_pasteObjects(
      category_1_copy)[0]['new_id']]
1843
    self.assertEqual(category_1_clone.getPortalType(), 'Category')
1844 1845 1846
    category_2_copy = category_1.manage_copyObjects(ids=(category_2.getId(),))
    category_2_clone = category_1[category_1.manage_pasteObjects(
      category_2_copy)[0]['new_id']]
1847
    self.assertEqual(category_2_clone.getPortalType(), 'Category')
1848 1849

    # test as a web user (assignor)
1850
    self.login('webmaster')
1851 1852 1853 1854 1855 1856 1857 1858 1859 1860 1861 1862 1863 1864 1865
    try:
      base_category_2 = portal_categories.newContent(portal_type='Base Category', id='base_category_2')
      self.fail("A webmaster should not be able to create a Base Category.")
    except Unauthorized:
      pass
    try:
      category_3 = publication_section.newContent(portal_type='Category', id='category_3')
      category_4 = category_3.newContent(portal_type='Category', id='category_4')
    except Unauthorized:
      self.fail("A webmaster should be able to create a Category.")
    # try to clone a sub category of the same owner whose parent is a
    # base category.
    category_3_copy = publication_section.manage_copyObjects(ids=(category_3.getId(),))
    category_3_clone = publication_section[publication_section.manage_pasteObjects(
      category_3_copy)[0]['new_id']]
1866
    self.assertEqual(category_3_clone.getPortalType(), 'Category')
1867 1868 1869 1870
    # try to clone a sub category of the different owner
    category_2_copy = category_1.manage_copyObjects(ids=(category_2.getId(),))
    category_2_clone = category_1[category_1.manage_pasteObjects(
      category_2_copy)[0]['new_id']]
1871
    self.assertEqual(category_2_clone.getPortalType(), 'Category')
1872 1873 1874 1875
    # try to clone a sub category of the same owner
    category_4_copy = category_3.manage_copyObjects(ids=(category_4.getId(),))
    category_4_clone = category_3[category_3.manage_pasteObjects(
      category_4_copy)[0]['new_id']]
1876
    self.assertEqual(category_4_clone.getPortalType(), 'Category')
1877

1878
  def test_08_createAndrenameCategory(self):
1879
    """ Test to create or rename categories with many users """
1880
    self.login('admin')
1881 1882
    portal_categories = self.portal.portal_categories
    publication_section = portal_categories.publication_section
1883

1884 1885 1886 1887 1888 1889 1890 1891
    # test for admin
    try:
      new_base_category_1 = portal_categories.newContent(portal_type='Base Category', id='new_base_category_1')
    except Unauthorized:
      self.fail("Admin should be able to create a Base Category.")
    try:
      new_category_1 = publication_section.newContent(portal_type='Category', id='new_category_1')
      new_category_2 = new_category_1.newContent(portal_type='Category',
1892
      id='new_category_2')
1893 1894 1895 1896
    except Unauthorized:
      self.fail("Admin should be able to create a Category.")
    self.tic()
    try:
1897
      new_cat_1_renamed = new_category_1.edit(id='new_cat_1_renamed')
1898 1899 1900 1901
      new_cat_2_renamed = new_category_2.edit(id='new_cat_2_renamed')
    except Unauthorized:
      self.fail("Admin should be able to rename a Category.")
    # test as a web user (assignor)
1902
    self.login('webmaster')
1903 1904 1905 1906 1907 1908 1909
    try:
      base_category_2 = portal_categories.newContent(portal_type='Base Category', id='base_category_2')
      self.fail("A webmaster should not be able to create a Base Category.")
    except Unauthorized:
      pass
    try:
      new_category_3 = publication_section.newContent(
Rafael Monnerat's avatar
Rafael Monnerat committed
1910
      portal_type='Category', id='new_category_3')
1911
      new_category_4 = new_category_3.newContent(portal_type='Category',
1912
          id='new_category_4')
1913 1914 1915 1916
    except Unauthorized:
      self.fail("A webmaster should be able to create a Category.")
    self.tic()
    try:
1917
      new_cat_3_renamed = new_category_3.edit(id='new_cat_3_renamed')
1918 1919 1920
      new_cat_4_renamed = new_category_4.edit(id='new_cat_4_renamed')
    except Unauthorized:
      self.fail("A webmaster should be able to rename a Category.")
1921

1922 1923 1924 1925 1926
  def test_getDocumentValueList_AnonymousUser(self):
    """
      For a given Web Site with Predicates:
      - membership_criterion_base_category: follow_up
      - membership_criterion_document_list: follow_up/project/object_id
1927

1928 1929 1930 1931 1932 1933 1934 1935 1936 1937 1938 1939
      When you access website/WebSection_viewContentListAsRSS:
      - with super user you get the correct result
      - with anonymous user you do not get the correct result

      In this case, both Web Pages are returned for Anonymous user and this
      it not the expected behavior.

      Note: The ListBox into WebSection_viewContentListAsRSS has
            getDocumentValueList defined as ListMethod.
    """
    project = self.portal.project_module.newContent(portal_type='Project')
    project.validate()
1940
    self.tic()
1941

1942
    website = self.portal.web_site_module.newContent(portal_type='Web Site',
1943
                                                     id='site')
1944
    website.publish()
1945 1946 1947
    website.setMembershipCriterionBaseCategory('follow_up')
    website.setMembershipCriterionDocumentList(['follow_up/%s' %
                                                  project.getRelativeUrl()])
1948
    self.tic()
1949 1950 1951 1952 1953 1954 1955 1956 1957 1958

    web_page_module = self.portal.web_page_module
    web_page_follow_up = web_page_module.newContent(portal_type="Web Page",
                                      follow_up=project.getRelativeUrl(),
                                      id='test_web_page_with_follow_up',
                                      reference='NXD-Document.Follow.Up.Test',
                                      version='001',
                                      language='en',
                                      text_content='test content')
    web_page_follow_up.publish()
1959
    self.tic()
1960

1961 1962 1963 1964 1965 1966 1967
    web_page_no_follow_up = web_page_module.newContent(portal_type="Web Page",
                                      id='test_web_page_no_follow_up',
                                      reference='NXD-Document.No.Follow.Up.Test',
                                      version='001',
                                      language='en',
                                      text_content='test content')
    web_page_no_follow_up.publish()
1968
    self.tic()
1969

1970
    self.assertEqual(1, len(website.WebSection_getDocumentValueList()))
1971 1972

    self.logout()
1973
    self.assertEqual(1, len(website.WebSection_getDocumentValueList()))
1974

1975 1976
  def test_WebSiteModuleDefaultSecurity(self):
    """
1977
      Test that by default Anonymous User cannot access Web Site Module
1978 1979
    """
    self.logout()
1980
    self.assertRaises(Unauthorized, self.portal.web_site_module.view)
1981

1982

1983 1984 1985 1986 1987 1988 1989 1990 1991 1992 1993 1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004 2005 2006 2007 2008 2009 2010 2011 2012
class TestERP5WebCategoryPublicationWorkflow(ERP5TypeTestCase):
  """Tests possible transitions for category_publication_workflow"""
  def getBusinessTemplateList(self):
    return ('erp5_base',
            'erp5_web',
            )

  def afterSetUp(self):
    base_category = self.getPortal().portal_categories\
        .newContent(portal_type='Base Category')
    self.doActionFor = self.getPortal().portal_workflow.doActionFor
    self.category = base_category.newContent(portal_type='Category')
    self.assertEqual('embedded', self.category.getValidationState())

  def test_category_embedded_expired(self):
    self.doActionFor(self.category, 'expire_action')
    self.assertEqual('expired', self.category.getValidationState())

  def test_category_embedded_protected_expired(self):
    self.doActionFor(self.category, 'protect_action')
    self.assertEqual('protected', self.category.getValidationState())
    self.doActionFor(self.category, 'expire_action')
    self.assertEqual('expired_protected', self.category.getValidationState())

  def test_category_embedded_published_expired(self):
    self.doActionFor(self.category, 'publish_action')
    self.assertEqual('published', self.category.getValidationState())
    self.doActionFor(self.category, 'expire_action')
    self.assertEqual('expired_published', self.category.getValidationState())

Rafael Monnerat's avatar
Rafael Monnerat committed
2013

2014 2015 2016
def test_suite():
  suite = unittest.TestSuite()
  suite.addTest(unittest.makeSuite(TestERP5Web))
2017
  suite.addTest(unittest.makeSuite(TestERP5WebWithSimpleSecurity))
2018
  suite.addTest(unittest.makeSuite(TestERP5WebCategoryPublicationWorkflow))
2019 2020
  suite.addTest(unittest.makeSuite(TestWebSiteTraversalHook))
  suite.addTest(unittest.makeSuite(TestWebSectionTraversalHook))
2021
  return suite