Commit 87ed7f27 authored by Tres Seaver's avatar Tres Seaver

Clean up tests for Z2 ZPT expresssion machinery:

o Move unneeded global imports to point of use.

o Wrap long lines.

o Make setUp / tearDown explicit.

o Avoid use of testcase instance variables.

o Split apart edge cases into discrete testcase methods.

o Use stock 'assertEqual', 'failUnless' methods of unittest.

Test both the 'trusted' and the 'untrusted' versions of the engine,
using a common base class.

  XXX:  This split will allow new tests which verify the security
        semantics of each version of the engine.
parent b24bb10c
...@@ -2,50 +2,51 @@ ...@@ -2,50 +2,51 @@
import unittest import unittest
import zope.component from zope.component.testing import PlacelessSetup
import zope.component.testing
from zope.traversing.adapters import DefaultTraversable class EngineTestsBase(PlacelessSetup):
from Products.PageTemplates import Expressions
from Products.PageTemplates.DeferExpr import LazyWrapper
from Products.PageTemplates.DeferExpr import DeferWrapper
from Products.PageTemplates.unicodeconflictresolver import \
DefaultUnicodeEncodingConflictResolver, \
StrictUnicodeEncodingConflictResolver, \
ReplacingUnicodeEncodingConflictResolver, \
IgnoringUnicodeEncodingConflictResolver
from Products.PageTemplates.interfaces import IUnicodeEncodingConflictResolver
class Dummy:
__allow_access_to_unprotected_subobjects__ = 1
def __call__(self):
return 'dummy'
class DummyDocumentTemplate:
__allow_access_to_unprotected_subobjects__ = 1
isDocTemp = True
def __call__(self, client=None, REQUEST={}, RESPONSE=None, **kw):
return 'dummy'
class ExpressionTests(zope.component.testing.PlacelessSetup, unittest.TestCase):
def setUp(self): def setUp(self):
super(ExpressionTests, self).setUp() from zope.component import provideAdapter
zope.component.provideAdapter(DefaultTraversable, (None,)) from zope.traversing.adapters import DefaultTraversable
PlacelessSetup.setUp(self)
provideAdapter(DefaultTraversable, (None,))
def tearDown(self):
PlacelessSetup.tearDown(self)
def _makeEngine(self):
# subclasses must override
raise NotImplementedError
def _makeContext(self, bindings=None):
class Dummy:
__allow_access_to_unprotected_subobjects__ = 1
def __call__(self):
return 'dummy'
self.e = e = Expressions.getEngine() class DummyDocumentTemplate:
self.ec = e.getContext( __allow_access_to_unprotected_subobjects__ = 1
isDocTemp = True
def __call__(self, client=None, REQUEST={}, RESPONSE=None, **kw):
return 'dummy'
_DEFAULT_BINDINGS = dict(
one = 1, one = 1,
d = {'one': 1, 'b': 'b', '': 'blank', '_': 'under'}, d = {'one': 1, 'b': 'b', '': 'blank', '_': 'under'},
blank = '', blank = '',
dummy = Dummy(), dummy = Dummy(),
dummy2 = DummyDocumentTemplate() dummy2 = DummyDocumentTemplate(),
) )
def testCompile(self): if bindings is None:
'''Test expression compilation''' bindings = _DEFAULT_BINDINGS
e = self.e return self._makeEngine().getContext(bindings)
def test_compile(self):
#Test expression compilation
e = self._makeEngine()
for p in ('x', 'x/y', 'x/y/z'): for p in ('x', 'x/y', 'x/y/z'):
e.compile(p) e.compile(p)
e.compile('path:a|b|c/d/e') e.compile('path:a|b|c/d/e')
...@@ -55,98 +56,181 @@ class ExpressionTests(zope.component.testing.PlacelessSetup, unittest.TestCase): ...@@ -55,98 +56,181 @@ class ExpressionTests(zope.component.testing.PlacelessSetup, unittest.TestCase):
e.compile('python: 2 + 2') e.compile('python: 2 + 2')
e.compile('python: 2 \n+\n 2\n') e.compile('python: 2 \n+\n 2\n')
def testSimpleEval(self): def test_evaluate_simple_path_binding(self):
'''Test simple expression evaluation''' ec = self._makeContext()
ec = self.ec self.assertEqual(ec.evaluate('one'), 1)
assert ec.evaluate('one') == 1
assert ec.evaluate('d/one') == 1 def test_evaluate_simple_path_dict_key_int_value(self):
assert ec.evaluate('d/b') == 'b' ec = self._makeContext()
self.assertEqual(ec.evaluate('d/one'), 1)
def test_evaluate_simple_path_dict_key_string_value(self):
ec = self._makeContext()
self.assertEqual(ec.evaluate('d/b'), 'b')
def testRenderedEval(self): def test_evaluate_with_render_simple_callable(self):
ec = self.ec ec = self._makeContext()
self.assertEquals(ec.evaluate('dummy'), 'dummy') self.assertEquals(ec.evaluate('dummy'), 'dummy')
def test_evaluate_with_render_DTML_template(self):
# http://www.zope.org/Collectors/Zope/2232 # http://www.zope.org/Collectors/Zope/2232
# DTML templates could not be called from a Page Template # DTML templates could not be called from a Page Template
# due to an ImportError # due to an ImportError
ec = self._makeContext()
self.assertEquals(ec.evaluate('dummy2'), 'dummy') self.assertEquals(ec.evaluate('dummy2'), 'dummy')
def testEval1(self): def test_evaluate_alternative_first_missing(self):
'''Test advanced expression evaluation 1''' ec = self._makeContext()
ec = self.ec self.failUnless(ec.evaluate('x | nothing') is None)
assert ec.evaluate('x | nothing') is None
def DONT_test_evaluate_with_empty_element(self):
# empty path elements aren't supported anymore, for the lack # empty path elements aren't supported anymore, for the lack
# of a use case # of a use case
#assert ec.evaluate('d/') == 'blank' ec = self._makeContext()
assert ec.evaluate('d/_') == 'under' self.assertEqual(ec.evaluate('d/'), 'blank')
#assert ec.evaluate('d/ | nothing') == 'blank'
assert ec.evaluate('d/?blank') == 'blank' def DONT_test_evaluate_with_empty_element_and_alternative(self):
# empty path elements aren't supported anymore, for the lack
def testHybrid(self): # of a use case
'''Test hybrid path expressions''' ec = self._makeContext()
ec = self.ec self.assertEqual(ec.evaluate('d/ | nothing'), 'blank')
assert ec.evaluate('x | python:1+1') == 2
assert ec.evaluate('x | python:int') == int def test_evaluate_dict_key_as_underscore(self):
assert ec.evaluate('x | string:x') == 'x' ec = self._makeContext()
assert ec.evaluate('x | string:$one') == '1' self.assertEqual(ec.evaluate('d/_'), 'under')
assert ec.evaluate('x | not:exists:x')
def test_evaluate_dict_with_key_from_expansion(self):
def testIteratorZRPythonExpr(self): ec = self._makeContext()
'''Test access to iterator functions from Python expressions''' self.assertEqual(ec.evaluate('d/?blank'), 'blank')
ec = self.ec
def test_hybrid_with_python_expression_int_value(self):
ec = self._makeContext()
self.assertEqual(ec.evaluate('x | python:1+1'), 2)
def test_hybrid_with_python_expression_type_value_not_called(self):
ec = self._makeContext()
self.assertEqual(ec.evaluate('x | python:int'), int)
def test_hybrid_with_string_expression(self):
ec = self._makeContext()
self.assertEqual(ec.evaluate('x | string:x'), 'x')
def test_hybrid_with_string_expression_and_expansion(self):
ec = self._makeContext()
self.assertEqual(ec.evaluate('x | string:$one'), '1')
def test_hybrid_with_compound_expression_int_value(self):
ec = self._makeContext()
self.failUnless(ec.evaluate('x | not:exists:x'))
def test_access_iterator_from_python_expression(self):
ec = self._makeContext()
ec.beginScope() ec.beginScope()
ec.setRepeat('loop', "python:[1,2,3]") ec.setRepeat('loop', "python:[1,2,3]")
assert ec.evaluate("python:repeat['loop'].odd()") self.failUnless(ec.evaluate("python:repeat['loop'].odd()"))
ec.endScope() ec.endScope()
def testWrappers(self): def test_defer_expression_returns_wrapper(self):
"""Test if defer and lazy are returning their wrappers from Products.PageTemplates.DeferExpr import DeferWrapper
""" ec = self._makeContext()
ec = self.ec
defer = ec.evaluate('defer: b') defer = ec.evaluate('defer: b')
lazy = ec.evaluate('lazy: b')
self.failUnless(isinstance(defer, DeferWrapper)) self.failUnless(isinstance(defer, DeferWrapper))
def test_lazy_expression_returns_wrapper(self):
from Products.PageTemplates.DeferExpr import LazyWrapper
ec = self._makeContext()
lazy = ec.evaluate('lazy: b')
self.failUnless(isinstance(lazy, LazyWrapper)) self.failUnless(isinstance(lazy, LazyWrapper))
def test_empty_ZopePathExpr(self): def test_empty_path_expression_explicit(self):
"""Test empty path expressions. ec = self._makeContext()
"""
ec = self.ec
self.assertEquals(ec.evaluate('path:'), None) self.assertEquals(ec.evaluate('path:'), None)
def test_empty_path_expression_explicit_with_trailing_whitespace(self):
ec = self._makeContext()
self.assertEquals(ec.evaluate('path: '), None) self.assertEquals(ec.evaluate('path: '), None)
def test_empty_path_expression_implicit(self):
ec = self._makeContext()
self.assertEquals(ec.evaluate(''), None) self.assertEquals(ec.evaluate(''), None)
def test_empty_path_expression_implicit_with_trailing_whitespace(self):
ec = self._makeContext()
self.assertEquals(ec.evaluate(' \n'), None) self.assertEquals(ec.evaluate(' \n'), None)
class UntrustedEngineTests(EngineTestsBase, unittest.TestCase):
class UnicodeEncodingConflictResolverTests(zope.component.testing.PlacelessSetup, unittest.TestCase): def _makeEngine(self):
from Products.PageTemplates.Expressions import createZopeEngine
return createZopeEngine()
# XXX: add tests that show security checks being enforced
class TrustedEngineTests(EngineTestsBase, unittest.TestCase):
def _makeEngine(self):
from Products.PageTemplates.Expressions import createTrustedZopeEngine
return createTrustedZopeEngine()
# XXX: add tests that show security checks *not* being enforced
class UnicodeEncodingConflictResolverTests(PlacelessSetup, unittest.TestCase):
def testDefaultResolver(self): def testDefaultResolver(self):
zope.component.provideUtility(DefaultUnicodeEncodingConflictResolver, from zope.component import getUtility
IUnicodeEncodingConflictResolver) from zope.component import provideUtility
resolver = zope.component.getUtility(IUnicodeEncodingConflictResolver) from Products.PageTemplates.interfaces \
self.assertRaises(UnicodeDecodeError, resolver.resolve, None, '', None) import IUnicodeEncodingConflictResolver
from Products.PageTemplates.unicodeconflictresolver \
import DefaultUnicodeEncodingConflictResolver
provideUtility(DefaultUnicodeEncodingConflictResolver,
IUnicodeEncodingConflictResolver)
resolver = getUtility(IUnicodeEncodingConflictResolver)
self.assertRaises(UnicodeDecodeError,
resolver.resolve, None, '', None)
def testStrictResolver(self): def testStrictResolver(self):
zope.component.provideUtility(StrictUnicodeEncodingConflictResolver, from zope.component import getUtility
from zope.component import provideUtility
from Products.PageTemplates.interfaces \
import IUnicodeEncodingConflictResolver
from Products.PageTemplates.unicodeconflictresolver \
import StrictUnicodeEncodingConflictResolver
provideUtility(StrictUnicodeEncodingConflictResolver,
IUnicodeEncodingConflictResolver) IUnicodeEncodingConflictResolver)
resolver = zope.component.getUtility(IUnicodeEncodingConflictResolver) resolver = getUtility(IUnicodeEncodingConflictResolver)
self.assertRaises(UnicodeDecodeError, resolver.resolve, None, '', None) self.assertRaises(UnicodeDecodeError,
resolver.resolve, None, '', None)
def testIgnoringResolver(self): def testIgnoringResolver(self):
zope.component.provideUtility(IgnoringUnicodeEncodingConflictResolver, from zope.component import getUtility
from zope.component import provideUtility
from Products.PageTemplates.interfaces \
import IUnicodeEncodingConflictResolver
from Products.PageTemplates.unicodeconflictresolver \
import IgnoringUnicodeEncodingConflictResolver
provideUtility(IgnoringUnicodeEncodingConflictResolver,
IUnicodeEncodingConflictResolver) IUnicodeEncodingConflictResolver)
resolver = zope.component.getUtility(IUnicodeEncodingConflictResolver) resolver = getUtility(IUnicodeEncodingConflictResolver)
self.assertEqual(resolver.resolve(None, '', None), '') self.assertEqual(resolver.resolve(None, '', None), '')
def testReplacingResolver(self): def testReplacingResolver(self):
zope.component.provideUtility(ReplacingUnicodeEncodingConflictResolver, from zope.component import getUtility
from zope.component import provideUtility
from Products.PageTemplates.interfaces \
import IUnicodeEncodingConflictResolver
from Products.PageTemplates.unicodeconflictresolver \
import ReplacingUnicodeEncodingConflictResolver
provideUtility(ReplacingUnicodeEncodingConflictResolver,
IUnicodeEncodingConflictResolver) IUnicodeEncodingConflictResolver)
resolver = zope.component.getUtility(IUnicodeEncodingConflictResolver) resolver = getUtility(IUnicodeEncodingConflictResolver)
self.assertEqual(resolver.resolve(None, '', None), u'\ufffd\ufffd\ufffd') self.assertEqual(resolver.resolve(None, '', None),
u'\ufffd\ufffd\ufffd')
def test_suite(): def test_suite():
return unittest.TestSuite(( return unittest.TestSuite((
unittest.makeSuite(ExpressionTests), unittest.makeSuite(UntrustedEngineTests),
unittest.makeSuite(TrustedEngineTests),
unittest.makeSuite(UnicodeEncodingConflictResolverTests) unittest.makeSuite(UnicodeEncodingConflictResolverTests)
)) ))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment