346 lines
12 KiB
Python
346 lines
12 KiB
Python
# -*- coding: utf-8; -*-
|
|
|
|
from unittest import TestCase
|
|
from unittest.mock import patch, MagicMock
|
|
|
|
from wuttaweb import menus as mod
|
|
from wuttaweb.testing import WebTestCase
|
|
|
|
|
|
class TestMenuHandler(WebTestCase):
|
|
|
|
def setUp(self):
|
|
self.setup_web()
|
|
self.handler = mod.MenuHandler(self.config)
|
|
|
|
def test_make_admin_menu(self):
|
|
|
|
# no people entry by default
|
|
menu = self.handler.make_admin_menu(self.request)
|
|
self.assertIsInstance(menu, dict)
|
|
routes = [item.get('route') for item in menu['items']]
|
|
self.assertNotIn('people', routes)
|
|
|
|
# but we can request it
|
|
menu = self.handler.make_admin_menu(self.request, include_people=True)
|
|
routes = [item.get('route') for item in menu['items']]
|
|
self.assertIn('people', routes)
|
|
|
|
def test_make_menus(self):
|
|
menus = self.handler.make_menus(self.request)
|
|
self.assertIsInstance(menus, list)
|
|
|
|
def test_is_allowed(self):
|
|
model = self.app.model
|
|
auth = self.app.get_auth_handler()
|
|
|
|
# user with perms
|
|
barney = model.User(username='barney')
|
|
self.session.add(barney)
|
|
blokes = model.Role(name="Blokes")
|
|
self.session.add(blokes)
|
|
barney.roles.append(blokes)
|
|
auth.grant_permission(blokes, 'appinfo.list')
|
|
self.request.user = barney
|
|
|
|
# perm not granted to user
|
|
item = {'perm': 'appinfo.configure'}
|
|
self.assertFalse(self.handler._is_allowed(self.request, item))
|
|
|
|
# perm *is* granted to user
|
|
item = {'perm': 'appinfo.list'}
|
|
self.assertTrue(self.handler._is_allowed(self.request, item))
|
|
|
|
# perm not required
|
|
item = {}
|
|
self.assertTrue(self.handler._is_allowed(self.request, item))
|
|
|
|
def test_mark_allowed(self):
|
|
|
|
def make_menus():
|
|
return [
|
|
{
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Foo", 'url': '#'},
|
|
{'title': "Bar", 'url': '#'},
|
|
],
|
|
},
|
|
]
|
|
|
|
mock_is_allowed = MagicMock()
|
|
with patch.object(self.handler, '_is_allowed', new=mock_is_allowed):
|
|
|
|
# all should be allowed
|
|
mock_is_allowed.return_value = True
|
|
menus = make_menus()
|
|
self.handler._mark_allowed(self.request, menus)
|
|
menu = menus[0]
|
|
self.assertTrue(menu['allowed'])
|
|
foo, bar = menu['items']
|
|
self.assertTrue(foo['allowed'])
|
|
self.assertTrue(bar['allowed'])
|
|
|
|
# none should be allowed
|
|
mock_is_allowed.return_value = False
|
|
menus = make_menus()
|
|
self.handler._mark_allowed(self.request, menus)
|
|
menu = menus[0]
|
|
self.assertFalse(menu['allowed'])
|
|
foo, bar = menu['items']
|
|
self.assertFalse(foo['allowed'])
|
|
self.assertFalse(bar['allowed'])
|
|
|
|
def test_mark_allowed_submenu(self):
|
|
|
|
def make_menus():
|
|
return [
|
|
{
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Foo", 'url': '#'},
|
|
{
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Bar", 'url': '#'},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
]
|
|
|
|
mock_is_allowed = MagicMock()
|
|
with patch.object(self.handler, '_is_allowed', new=mock_is_allowed):
|
|
|
|
# all should be allowed
|
|
mock_is_allowed.return_value = True
|
|
menus = make_menus()
|
|
self.handler._mark_allowed(self.request, menus)
|
|
menu = menus[0]
|
|
self.assertTrue(menu['allowed'])
|
|
foo, submenu = menu['items']
|
|
self.assertTrue(foo['allowed'])
|
|
self.assertTrue(submenu['allowed'])
|
|
subitem = submenu['items'][0]
|
|
self.assertTrue(subitem['allowed'])
|
|
|
|
# none should be allowed
|
|
mock_is_allowed.return_value = False
|
|
menus = make_menus()
|
|
self.handler._mark_allowed(self.request, menus)
|
|
menu = menus[0]
|
|
self.assertFalse(menu['allowed'])
|
|
foo, submenu = menu['items']
|
|
self.assertFalse(foo['allowed'])
|
|
self.assertFalse(submenu['allowed'])
|
|
subitem = submenu['items'][0]
|
|
self.assertFalse(subitem['allowed'])
|
|
|
|
def test_make_menu_key(self):
|
|
self.assertEqual(self.handler._make_menu_key('foo'), 'foo')
|
|
self.assertEqual(self.handler._make_menu_key('FooBar'), 'foobar')
|
|
self.assertEqual(self.handler._make_menu_key('Foo - $#Bar'), 'foobar')
|
|
self.assertEqual(self.handler._make_menu_key('Foo__Bar'), 'foo__bar')
|
|
|
|
def test_make_menu_entry_item(self):
|
|
item = {'title': "Foo", 'url': '#'}
|
|
entry = self.handler._make_menu_entry(self.request, item)
|
|
self.assertEqual(entry['type'], 'item')
|
|
self.assertEqual(entry['title'], "Foo")
|
|
self.assertEqual(entry['url'], '#')
|
|
self.assertTrue(entry['is_link'])
|
|
|
|
def test_make_menu_entry_item_with_no_url(self):
|
|
item = {'title': "Foo"}
|
|
entry = self.handler._make_menu_entry(self.request, item)
|
|
self.assertEqual(entry['type'], 'item')
|
|
self.assertEqual(entry['title'], "Foo")
|
|
self.assertNotIn('url', entry)
|
|
# nb. still sets is_link = True; basically it's <a> with no href
|
|
self.assertTrue(entry['is_link'])
|
|
|
|
def test_make_menu_entry_item_with_known_route(self):
|
|
item = {'title': "Foo", 'route': 'home'}
|
|
with patch.object(self.request, 'route_url', return_value='/something'):
|
|
entry = self.handler._make_menu_entry(self.request, item)
|
|
self.assertEqual(entry['type'], 'item')
|
|
self.assertEqual(entry['url'], '/something')
|
|
self.assertTrue(entry['is_link'])
|
|
|
|
def test_make_menu_entry_item_with_unknown_route(self):
|
|
item = {'title': "Foo", 'route': 'home'}
|
|
with patch.object(self.request, 'route_url', side_effect=KeyError):
|
|
entry = self.handler._make_menu_entry(self.request, item)
|
|
self.assertEqual(entry['type'], 'item')
|
|
# nb. fake url is used, based on (bad) route name
|
|
self.assertEqual(entry['url'], 'home')
|
|
self.assertTrue(entry['is_link'])
|
|
|
|
def test_make_menu_entry_sep(self):
|
|
item = {'type': 'sep'}
|
|
entry = self.handler._make_menu_entry(self.request, item)
|
|
self.assertEqual(entry['type'], 'sep')
|
|
self.assertTrue(entry['is_sep'])
|
|
self.assertFalse(entry['is_menu'])
|
|
|
|
def test_make_raw_menus(self):
|
|
# minimal test to ensure it calls the other method
|
|
with patch.object(self.handler, 'make_menus') as make_menus:
|
|
self.handler._make_raw_menus(self.request, foo='bar')
|
|
make_menus.assert_called_once_with(self.request, foo='bar')
|
|
|
|
def test_do_make_menus_prune_unallowed_item(self):
|
|
test_menus = [
|
|
{
|
|
'title': "First Menu",
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Foo", 'url': '#'},
|
|
{'title': "Bar", 'url': '#'},
|
|
],
|
|
},
|
|
]
|
|
|
|
def is_allowed(request, item):
|
|
if item.get('title') == 'Bar':
|
|
return False
|
|
return True
|
|
|
|
with patch.object(self.handler, 'make_menus', return_value=test_menus):
|
|
with patch.object(self.handler, '_is_allowed', side_effect=is_allowed):
|
|
menus = self.handler.do_make_menus(self.request)
|
|
|
|
# Foo remains but Bar is pruned
|
|
menu = menus[0]
|
|
self.assertEqual(len(menu['items']), 1)
|
|
item = menu['items'][0]
|
|
self.assertEqual(item['title'], 'Foo')
|
|
|
|
def test_do_make_menus_prune_unallowed_menu(self):
|
|
test_menus = [
|
|
{
|
|
'title': "First Menu",
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Foo", 'url': '#'},
|
|
{'title': "Bar", 'url': '#'},
|
|
],
|
|
},
|
|
{
|
|
'title': "Second Menu",
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Baz", 'url': '#'},
|
|
],
|
|
},
|
|
]
|
|
|
|
def is_allowed(request, item):
|
|
if item.get('title') == 'Baz':
|
|
return True
|
|
return False
|
|
|
|
with patch.object(self.handler, 'make_menus', return_value=test_menus):
|
|
with patch.object(self.handler, '_is_allowed', side_effect=is_allowed):
|
|
menus = self.handler.do_make_menus(self.request)
|
|
|
|
# Second/Baz remains but First/Foo/Bar are pruned
|
|
self.assertEqual(len(menus), 1)
|
|
menu = menus[0]
|
|
self.assertEqual(menu['title'], 'Second Menu')
|
|
self.assertEqual(len(menu['items']), 1)
|
|
item = menu['items'][0]
|
|
self.assertEqual(item['title'], 'Baz')
|
|
|
|
def test_do_make_menus_with_top_link(self):
|
|
test_menus = [
|
|
{
|
|
'title': "First Menu",
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Foo", 'url': '#'},
|
|
{'title': "Bar", 'url': '#'},
|
|
],
|
|
},
|
|
{
|
|
'title': "Second Link",
|
|
'type': 'link',
|
|
},
|
|
]
|
|
|
|
with patch.object(self.handler, 'make_menus', return_value=test_menus):
|
|
with patch.object(self.handler, '_is_allowed', return_value=True):
|
|
menus = self.handler.do_make_menus(self.request)
|
|
|
|
# ensure top link remains
|
|
self.assertEqual(len(menus), 2)
|
|
menu = menus[1]
|
|
self.assertEqual(menu['title'], "Second Link")
|
|
|
|
def test_do_make_menus_with_trailing_sep(self):
|
|
test_menus = [
|
|
{
|
|
'title': "First Menu",
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Foo", 'url': '#'},
|
|
{'title': "Bar", 'url': '#'},
|
|
{'type': 'sep'},
|
|
],
|
|
},
|
|
]
|
|
|
|
with patch.object(self.handler, 'make_menus', return_value=test_menus):
|
|
with patch.object(self.handler, '_is_allowed', return_value=True):
|
|
menus = self.handler.do_make_menus(self.request)
|
|
|
|
# ensure trailing sep was pruned
|
|
menu = menus[0]
|
|
self.assertEqual(len(menu['items']), 2)
|
|
foo, bar = menu['items']
|
|
self.assertEqual(foo['title'], 'Foo')
|
|
self.assertEqual(bar['title'], 'Bar')
|
|
|
|
def test_do_make_menus_with_submenu(self):
|
|
test_menus = [
|
|
{
|
|
'title': "First Menu",
|
|
'type': 'menu',
|
|
'items': [
|
|
{
|
|
'title': "First Submenu",
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Foo", 'url': '#'},
|
|
],
|
|
},
|
|
{
|
|
'title': "Second Submenu",
|
|
'type': 'menu',
|
|
'items': [
|
|
{'title': "Bar", 'url': '#'},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
]
|
|
|
|
def is_allowed(request, item):
|
|
if item.get('title') == 'Bar':
|
|
return False
|
|
return True
|
|
|
|
with patch.object(self.handler, 'make_menus', return_value=test_menus):
|
|
with patch.object(self.handler, '_is_allowed', side_effect=is_allowed):
|
|
menus = self.handler.do_make_menus(self.request)
|
|
|
|
# first submenu remains, second is pruned
|
|
menu = menus[0]
|
|
self.assertEqual(len(menu['items']), 1)
|
|
submenu = menu['items'][0]
|
|
self.assertEqual(submenu['type'], 'submenu')
|
|
self.assertEqual(submenu['title'], 'First Submenu')
|
|
self.assertEqual(len(submenu['items']), 1)
|
|
item = submenu['items'][0]
|
|
self.assertEqual(item['title'], 'Foo')
|