masakari/masakari/tests/unit/api/openstack/ha/test_notifications.py

346 lines
15 KiB
Python

# Copyright (c) 2016 NTT DATA
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
"""Tests for the notifications api."""
import mock
from oslo_serialization import jsonutils
from oslo_utils import timeutils
from six.moves import http_client as http
from webob import exc
from masakari.api.openstack.ha import notifications
from masakari.engine import rpcapi as engine_rpcapi
from masakari import exception
from masakari.ha import api as ha_api
from masakari.objects import base as obj_base
from masakari.objects import notification as notification_obj
from masakari import test
from masakari.tests.unit.api.openstack import fakes
from masakari.tests.unit.objects import test_objects
from masakari.tests import uuidsentinel
NOW = timeutils.utcnow().replace(microsecond=0)
def _make_notification_obj(notification_dict):
return notification_obj.Notification(**notification_dict)
def _make_notifications_list(notifications_list):
return notification_obj.Notification(objects=[
_make_notification_obj(a) for a in notifications_list])
NOTIFICATION_DATA = {"type": "VM", "id": 1,
"payload":
{'event': 'STOPPED', 'host_status': 'NORMAL',
'cluster_status': 'ONLINE'},
"source_host_uuid": uuidsentinel.fake_host,
"generated_time": NOW,
"status": "running",
"notification_uuid": uuidsentinel.fake_notification,
"created_at": NOW,
"updated_at": None,
"deleted_at": None,
"deleted": 0
}
NOTIFICATION = _make_notification_obj(NOTIFICATION_DATA)
NOTIFICATION_LIST = [
{"type": "VM", "id": 1, "payload": {'event': 'STOPPED',
'host_status': 'NORMAL',
'cluster_status': 'ONLINE'},
"source_host_uuid": uuidsentinel.fake_host, "generated_time": NOW,
"status": "running", "notification_uuid": uuidsentinel.fake_notification,
"created_at": NOW, "updated_at": None, "deleted_at": None, "deleted": 0},
{"type": "PROCESS", "id": 2, "payload": {'event': 'STOPPED',
'process_name': 'fake_process'},
"source_host_uuid": uuidsentinel.fake_host1, "generated_time": NOW,
"status": "running", "notification_uuid": uuidsentinel.fake_notification1,
"created_at": NOW, "updated_at": None, "deleted_at": None, "deleted": 0},
]
NOTIFICATION_LIST = _make_notifications_list(NOTIFICATION_LIST)
class NotificationTestCase(test.TestCase):
"""Test Case for notifications api."""
bad_request = exception.ValidationError
@mock.patch.object(engine_rpcapi, 'EngineAPI')
def setUp(self, mock_rpc):
super(NotificationTestCase, self).setUp()
self.controller = notifications.NotificationsController()
self.req = fakes.HTTPRequest.blank('/v1/notifications',
use_admin_context=True)
self.context = self.req.environ['masakari.context']
@property
def app(self):
return fakes.wsgi_app_v1(init_only='os-hosts')
def _assert_notification_data(self, expected, actual):
self.assertTrue(obj_base.obj_equal_prims(expected, actual),
"The notifications objects were not equal")
@mock.patch.object(ha_api.NotificationAPI, 'get_all')
def test_index(self, mock_get_all):
mock_get_all.return_value = NOTIFICATION_LIST
result = self.controller.index(self.req)
result = result['notifications']
self._assert_notification_data(NOTIFICATION_LIST,
_make_notifications_list(result))
def test_index_marker_negative(self):
req = fakes.HTTPRequest.blank('/v1/notifications?limit=-1',
use_admin_context=True)
self.assertRaises(exc.HTTPBadRequest, self.controller.index, req)
def test_index_invalid_sort_key(self):
req = fakes.HTTPRequest.blank('/v1/notifications?sort_key=abcd',
use_admin_context=True)
self.assertRaises(exc.HTTPBadRequest, self.controller.index, req)
def test_index_invalid_sort_dir(self):
req = fakes.HTTPRequest.blank('/v1/notifications?sort_dir=abcd',
use_admin_context=True)
self.assertRaises(exc.HTTPBadRequest, self.controller.index, req)
@mock.patch.object(ha_api.NotificationAPI, 'get_all')
def test_index_marker_not_found(self, mock_get_all):
fake_request = fakes.HTTPRequest.blank('/v1/notifications?marker=1234',
use_admin_context=True)
mock_get_all.side_effect = exception.MarkerNotFound
self.assertRaises(exc.HTTPBadRequest, self.controller.index,
fake_request)
@mock.patch.object(ha_api.NotificationAPI, 'create_notification')
def test_create(self, mock_create):
mock_create.return_value = NOTIFICATION
result = self.controller.create(self.req, body={
"notification": {"hostname": "fake_host",
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "VM",
"generated_time": "2016-09-13T09:11:21.656788"}})
result = result['notification']
test_objects.compare_obj(self, result, NOTIFICATION_DATA)
@mock.patch('masakari.rpc.get_client')
@mock.patch.object(ha_api.NotificationAPI, 'create_notification')
def test_create_success_with_201_response_code(
self, mock_client, mock_create):
body = {
"notification": {
"hostname": "fake_host",
"payload": {
"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"
},
"type": "VM",
"generated_time": NOW
}
}
fake_req = self.req
fake_req.headers['Content-Type'] = 'application/json'
fake_req.method = 'POST'
fake_req.body = jsonutils.dump_as_bytes(body)
resp = fake_req.get_response(self.app)
self.assertEqual(http.ACCEPTED, resp.status_code)
def test_create_invalid_type(self):
body = {
"notification": {"hostname": "fake_host",
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "Fake",
"generated_time": "2016-09-13T09:11:21.656788"}}
self.assertRaises(self.bad_request, self.controller.create,
self.req, body=body)
@mock.patch.object(ha_api.NotificationAPI, 'create_notification')
def test_create_host_not_found(self, mock_create):
body = {
"notification": {"hostname": "fake_host",
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "VM",
"generated_time": "2016-09-13T09:11:21.656788"}}
mock_create.side_effect = exception.HostNotFoundByName
self.assertRaises(exc.HTTPBadRequest, self.controller.create,
self.req, body=body)
def test_create_with_no_notification_in_body(self):
body = {"hostname": "fake_host",
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "VM",
"generated_time": "2016-09-13T09:11:21.656788"}
self.assertRaises(self.bad_request, self.controller.create,
self.req, body=body)
def test_create_with_no_payload(self):
body = {"notification": {"hostname": "fake_host",
"type": "VM",
"generated_time":
"2016-09-13T09:11:21.656788"}}
self.assertRaises(self.bad_request, self.controller.create,
self.req, body=body)
def test_create_with_no_hostname(self):
body = {"notification": {"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "VM",
"generated_time":
"2016-09-13T09:11:21.656788"}}
self.assertRaises(self.bad_request, self.controller.create,
self.req, body=body)
def test_create_with_no_type(self):
body = {"notification": {"hostname": "fake_host",
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"generated_time":
"2016-09-13T09:11:21.656788"}}
self.assertRaises(self.bad_request, self.controller.create,
self.req, body=body)
def test_create_with_no_generated_time(self):
body = {"notification": {"hostname": "fake_host",
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "VM",
}}
self.assertRaises(self.bad_request, self.controller.create,
self.req, body=body)
def test_create_with_hostname_too_long(self):
body = {
"notification": {"hostname": "fake_host" * 255,
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "VM",
"generated_time": "2016-09-13T09:11:21.656788"}}
self.assertRaises(self.bad_request, self.controller.create,
self.req, body=body)
def test_create_with_extra_invalid_arg(self):
body = {
"notification": {"hostname": "fake_host",
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "VM",
"generated_time": "2016-09-13T09:11:21.656788",
"invalid_extra": "non_expected_parameter"}}
self.assertRaises(self.bad_request, self.controller.create,
self.req, body=body)
@mock.patch.object(ha_api.NotificationAPI, 'get_notification')
def test_show(self, mock_get_notification):
mock_get_notification.return_value = NOTIFICATION
result = self.controller.show(self.req, uuidsentinel.fake_notification)
result = result['notification']
self._assert_notification_data(NOTIFICATION,
_make_notification_obj(result))
@mock.patch.object(ha_api.NotificationAPI, 'get_notification')
def test_show_with_non_existing_uuid(self, mock_get_notification):
mock_get_notification.side_effect = exception.NotificationNotFound
self.assertRaises(exc.HTTPNotFound,
self.controller.show, self.req, "2")
@mock.patch('masakari.rpc.get_client')
def test_delete_notification(self, mock_client):
url = '/v1/notifications/%s' % uuidsentinel.fake_notification
fake_req = fakes.HTTPRequest.blank(url, use_admin_context=True)
fake_req.headers['Content-Type'] = 'application/json'
fake_req.method = 'DELETE'
resp = fake_req.get_response(self.app)
self.assertEqual(http.METHOD_NOT_ALLOWED, resp.status_code)
@mock.patch('masakari.rpc.get_client')
def test_update_notification(self, mock_client):
url = '/v1/notifications/%s' % uuidsentinel.fake_notification
fake_req = fakes.HTTPRequest.blank(url, use_admin_context=True)
fake_req.headers['Content-Type'] = 'application/json'
fake_req.method = 'PUT'
resp = fake_req.get_response(self.app)
self.assertEqual(http.METHOD_NOT_ALLOWED, resp.status_code)
class NotificationCasePolicyNotAuthorized(test.NoDBTestCase):
"""Test Case for notifications non admin."""
@mock.patch.object(engine_rpcapi, 'EngineAPI')
def setUp(self, mock_rpc):
super(NotificationCasePolicyNotAuthorized, self).setUp()
self.controller = notifications.NotificationsController()
self.req = fakes.HTTPRequest.blank('/v1/notifications')
self.context = self.req.environ['masakari.context']
self.rule_name = "os_masakari_api:notifications"
self.policy.set_rules({self.rule_name: "project:non_fake"})
def _check_rule(self, exc):
self.assertEqual(
"Policy doesn't allow %s to be performed." % self.rule_name,
exc.format_message())
def test_create_no_admin(self):
body = {
"notification": {"hostname": "fake_host",
"payload": {"event": "STOPPED",
"host_status": "NORMAL",
"cluster_status": "ONLINE"},
"type": "VM",
"generated_time": "2016-09-13T09:11:21.656788"}}
exc = self.assertRaises(exception.PolicyNotAuthorized,
self.controller.create,
self.req, body=body)
self._check_rule(exc)
def test_show_no_admin(self):
exc = self.assertRaises(exception.PolicyNotAuthorized,
self.controller.show,
self.req, uuidsentinel.fake_notification)
self._check_rule(exc)
def test_index_no_admin(self):
exc = self.assertRaises(exception.PolicyNotAuthorized,
self.controller.index,
self.req)
self._check_rule(exc)