aboutsummaryrefslogtreecommitdiffstats
path: root/src/notification/signals.py
blob: 936c25ba4eab19d84dc23955951af2f6d39de65c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
##############################################################################
# Copyright (c) 2016 Max Breitenfeldt and others.
#
# All rights reserved. This program and the accompanying materials
# are made available under the terms of the Apache License, Version 2.0
# which accompanies this distribution, and is available at
# http://www.apache.org/licenses/LICENSE-2.0
##############################################################################


from django.db.models.signals import post_save
from django.dispatch import receiver

from booking.models import Booking
from notification.models import BookingNotification


@receiver(post_save, sender=Booking)
def booking_notification_handler(sender, instance, **kwargs):
    BookingNotification.objects.update_or_create(
        booking=instance, type='booking_start', defaults={'submit_time': instance.start}
    )
    BookingNotification.objects.update_or_create(
        booking=instance, type='booking_end', defaults={'submit_time': instance.end}
    )
r: #a6e22e } /* Name.Function */ .highlight .nl { color: #f8f8f2 } /* Name.Label */ .highlight .nn { color: #f8f8f2 } /* Name.Namespace */ .highlight .nx { color: #a6e22e } /* Name.Other */ .highlight .py { color: #f8f8f2 } /* Name.Property */ .highlight .nt { color: #f92672 } /* Name.Tag */ .highlight .nv { color: #f8f8f2 } /* Name.Variable */ .highlight .ow { color: #f92672 } /* Operator.Word */ .highlight .w { color: #f8f8f2 } /* Text.Whitespace */ .highlight .mb { color: #ae81ff } /* Literal.Number.Bin */ .highlight .mf { color: #ae81ff } /* Literal.Number.Float */ .highlight .mh { color: #ae81ff } /* Literal.Number.Hex */ .highlight .mi { color: #ae81ff } /* Literal.Number.Integer */ .highlight .mo { color: #ae81ff } /* Literal.Number.Oct */ .highlight .sa { color: #e6db74 } /* Literal.String.Affix */ .highlight .sb { color: #e6db74 } /* Literal.String.Backtick */ .highlight .sc { color: #e6db74 } /* Literal.String.Char */ .highlight .dl { color: #e6db74 } /* Literal.String.Delimiter */ .highlight .sd { color: #e6db74 } /* Literal.String.Doc */ .highlight .s2 { color: #e6db74 } /* Literal.String.Double */ .highlight .se { color: #ae81ff } /* Literal.String.Escape */ .highlight .sh { color: #e6db74 } /* Literal.String.Heredoc */ .highlight .si { color: #e6db74 } /* Literal.String.Interpol */ .highlight .sx { color: #e6db74 } /* Literal.String.Other */ .highlight .sr { color: #e6db74 } /* Literal.String.Regex */ .highlight .s1 { color: #e6db74 } /* Literal.String.Single */ .highlight .ss { color: #e6db74 } /* Literal.String.Symbol */ .highlight .bp { color: #f8f8f2 } /* Name.Builtin.Pseudo */ .highlight .fm { color: #a6e22e } /* Name.Function.Magic */ .highlight .vc { color: #f8f8f2 } /* Name.Variable.Class */ .highlight .vg { color: #f8f8f2 } /* Name.Variable.Global */ .highlight .vi { color: #f8f8f2 } /* Name.Variable.Instance */ .highlight .vm { color: #f8f8f2 } /* Name.Variable.Magic */ .highlight .il { color: #ae81ff } /* Literal.Number.Integer.Long */ } @media (prefers-color-scheme: light) { .highlight .hll { background-color: #ffffcc } .highlight .c { color: #888888 } /* Comment */ .highlight .err { color: #a61717; background-color: #e3d2d2 } /* Error */ .highlight .k { color: #008800; font-weight: bold } /* Keyword */ .highlight .ch { color: #888888 } /* Comment.Hashbang */ .highlight .cm { color: #888888 } /* Comment.Multiline */ .highlight .cp { color: #cc0000; font-weight: bold } /* Comment.Preproc */ .highlight .cpf { color: #888888 } /* Comment.PreprocFile */ .highlight .c1 { color: #888888 } /* Comment.Single */ .highlight .cs { color: #cc0000; font-weight: bold; background-color: #fff0f0 } /* Comment.Special */ .highlight .gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */ .highlight .ge { font-style: italic } /* Generic.Emph */ .highlight .gr { color: #aa0000 } /* Generic.Error */ .highlight .gh { color: #333333 } /* Generic.Heading */ .highlight .gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */ .highlight .go { color: #888888 } /* Generic.Output */ .highlight .gp { color: #555555 } /* Generic.Prompt */ .highlight .gs { font-weight: bold } /* Generic.Strong */ .highlight .gu { color: #666666 } /* Generic.Subheading */ .highlight .gt { color: #aa0000 } /* Generic.Traceback */ .highlight .kc { color: #008800; font-weight: bold } /* Keyword.Constant */ .highlight .kd { color: #008800; font-weight: bold } /* Keyword.Declaration */ .highlight .kn { color: #008800; font-weight: bold } /* Keyword.Namespace */ .highlight .kp { color: #008800 } /* Keyword.Pseudo */ .highlight .kr { color: #008800; font-weight: bold } /* Keyword.Reserved */ .highlight .kt { color: #888888; font-weight: bold } /* Keyword.Type */ .highlight .m { color: #0000DD; font-weight: bold } /* Literal.Number */ .highlight .s { color: #dd2200; background-color: #fff0f0 } /* Literal.String */ .highlight .na { color: #336699 } /* Name.Attribute */ .highlight .nb { color: #003388 } /* Name.Builtin */ .highlight .nc { color: #bb0066; font-weight: bold } /* Name.Class */ .highlight .no { color: #003366; font-weight: bold } /* Name.Constant */ .highlight .nd { color: #555555 } /* Name.Decorator */ .highlight .ne { color: #bb0066; font-weight: bold } /* Name.Exception */ .highlight .nf { color: #0066bb; font-weight: bold } /* Name.Function */ .highlight .nl { color: #336699; font-style: italic } /* Name.Label */ .highlight .nn { color: #bb0066; font-weight: bold } /* Name.Namespace */ .highlight .py { color: #336699; font-weight: bold } /* Name.Property */ .highlight .nt { color: #bb0066; font-weight: bold } /* Name.Tag */ .highlight .nv { color: #336699 } /* Name.Variable */ .highlight .ow { color: #008800 } /* Operator.Word */ .highlight .w { color: #bbbbbb } /* Text.Whitespace */ .highlight .mb { color: #0000DD; font-weight: bold } /* Literal.Number.Bin */ .highlight .mf { color: #0000DD; font-weight: bold } /* Literal.Number.Float */ .highlight .mh { color: #0000DD; font-weight: bold } /* Literal.Number.Hex */ .highlight .mi { color: #0000DD; font-weight: bold } /* Literal.Number.Integer */ .highlight .mo { color: #0000DD; font-weight: bold } /* Literal.Number.Oct */ .highlight .sa { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Affix */ .highlight .sb { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Backtick */ .highlight .sc { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Char */ .highlight .dl { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Delimiter */ .highlight .sd { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Doc */ .highlight .s2 { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Double */ .highlight .se { color: #0044dd; background-color: #fff0f0 } /* Literal.String.Escape */ .highlight .sh { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Heredoc */ .highlight .si { color: #3333bb; background-color: #fff0f0 } /* Literal.String.Interpol */ .highlight .sx { color: #22bb22; background-color: #f0fff0 } /* Literal.String.Other */ .highlight .sr { color: #008800; background-color: #fff0ff } /* Literal.String.Regex */ .highlight .s1 { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Single */ .highlight .ss { color: #aa6600; background-color: #fff0f0 } /* Literal.String.Symbol */ .highlight .bp { color: #003388 } /* Name.Builtin.Pseudo */ .highlight .fm { color: #0066bb; font-weight: bold } /* Name.Function.Magic */ .highlight .vc { color: #336699 } /* Name.Variable.Class */ .highlight .vg { color: #dd7700 } /* Name.Variable.Global */ .highlight .vi { color: #3333bb } /* Name.Variable.Instance */ .highlight .vm { color: #336699 } /* Name.Variable.Magic */ .highlight .il { color: #0000DD; font-weight: bold } /* Literal.Number.Integer.Long */ }
#!/usr/bin/env python

# Copyright (c) 2016 Orange and others.
#
# All rights reserved. This program and the accompanying materials
# are made available under the terms of the Apache License, Version 2.0
# which accompanies this distribution, and is available at
# http://www.apache.org/licenses/LICENSE-2.0

import errno
import logging
import mock
import os
import unittest

from robot.errors import RobotError

from functest.core import TestCasesBase
from functest.opnfv_tests.Controllers.ODL import OpenDaylightTesting
from functest.utils import functest_constants as ft_constants


class ODLTestCasesTesting(unittest.TestCase):

    logging.disable(logging.CRITICAL)

    _keystone_ip = "127.0.0.1"
    _neutron_ip = "127.0.0.2"
    _sdn_controller_ip = "127.0.0.3"
    _os_tenantname = "admin"
    _os_username = "admin"
    _os_password = "admin"
    _odl_webport = "8080"
    _odl_restconfport = "8181"
    _odl_username = "admin"
    _odl_password = "admin"

    def setUp(self):
        ft_constants.OS_USERNAME = self._os_username
        ft_constants.OS_PASSWORD = self._os_password
        ft_constants.OS_TENANT_NAME = self._os_tenantname
        self.test = OpenDaylightTesting.ODLTestCases()

    @mock.patch('fileinput.input', side_effect=Exception())
    def test_set_robotframework_vars_failed(self, *args):
        self.assertFalse(self.test.set_robotframework_vars())

    @mock.patch('fileinput.input', return_value=[])
    def test_set_robotframework_vars(self, args):
        self.assertTrue(self.test.set_robotframework_vars())

    @classmethod
    def _fake_url_for(cls, service_type='identity', **kwargs):
        if service_type == 'identity':
            return "http://{}:5000/v2.0".format(
                ODLTestCasesTesting._keystone_ip)
        elif service_type == 'network':
            return "http://{}:9696".format(ODLTestCasesTesting._neutron_ip)
        else:
            return None

    @classmethod
    def _get_fake_keystone_client(cls):
        kclient = mock.Mock()
        kclient.service_catalog = mock.Mock()
        kclient.service_catalog.url_for = mock.Mock(
            side_effect=cls._fake_url_for)
        return kclient

    def _get_main_kwargs(self, key=None):
        kwargs = {'odlusername': self._odl_username,
                  'odlpassword': self._odl_password,
                  'keystoneip': self._keystone_ip,
                  'neutronip': self._neutron_ip,
                  'osusername': self._os_username,
                  'ostenantname': self._os_tenantname,
                  'ospassword': self._os_password,
                  'odlip': self._sdn_controller_ip,
                  'odlwebport': self._odl_webport,
                  'odlrestconfport': self._odl_restconfport}
        if key:
            del kwargs[key]
        return kwargs

    def _test_main(self, status, *args):
        kwargs = self._get_main_kwargs()
        self.assertEqual(self.test.main(**kwargs), status)
        odl_res_dir = OpenDaylightTesting.ODLTestCases.res_dir
        if len(args) > 0:
            args[0].assert_called_once_with(odl_res_dir)
        if len(args) > 1:
            variable = ['KEYSTONE:{}'.format(self._keystone_ip),
                        'NEUTRON:{}'.format(self._neutron_ip),
                        'OSUSERNAME:"{}"'.format(self._os_username),
                        'OSTENANTNAME:"{}"'.format(self._os_tenantname),
                        'OSPASSWORD:"{}"'.format(self._os_password),
                        'ODL_SYSTEM_IP:{}'.format(self._sdn_controller_ip),
                        'PORT:{}'.format(self._odl_webport),
                        'RESTCONFPORT:{}'.format(self._odl_restconfport)]
            output_file = os.path.join(odl_res_dir, 'output.xml')
            args[1].assert_called_once_with(
                OpenDaylightTesting.ODLTestCases.basic_suite_dir,
                OpenDaylightTesting.ODLTestCases.neutron_suite_dir,
                log='NONE',
                output=output_file,
                report='NONE',
                stdout=mock.ANY,
                variable=variable)
        if len(args) > 2:
            stdout_file = os.path.join(odl_res_dir, 'stdout.txt')
            args[2].assert_called_with(stdout_file)

    def _test_main_missing_keyword(self, key):
        kwargs = self._get_main_kwargs(key)
        self.assertEqual(self.test.main(**kwargs),
                         TestCasesBase.TestCasesBase.EX_RUN_ERROR)

    def test_main_missing_odlusername(self):
        self._test_main_missing_keyword('odlusername')

    def test_main_missing_odlpassword(self):
        self._test_main_missing_keyword('odlpassword')

    def test_main_missing_keystoneip(self):
        self._test_main_missing_keyword('keystoneip')

    def test_main_missing_neutronip(self):
        self._test_main_missing_keyword('neutronip')

    def test_main_missing_osusername(self):
        self._test_main_missing_keyword('osusername')

    def test_main_missing_ostenantname(self):
        self._test_main_missing_keyword('ostenantname')

    def test_main_missing_ospassword(self):
        self._test_main_missing_keyword('ospassword')

    def test_main_missing_odlip(self):
        self._test_main_missing_keyword('odlip')

    def test_main_missing_odlwebport(self):
        self._test_main_missing_keyword('odlwebport')

    def test_main_missing_odlrestconfport(self):
        self._test_main_missing_keyword('odlrestconfport')

    def test_main_set_robotframework_vars_failed(self):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=False):
            self._test_main(TestCasesBase.TestCasesBase.EX_RUN_ERROR)
            self.test.set_robotframework_vars.assert_called_once_with(
                self._odl_username, self._odl_password)

    @mock.patch('os.makedirs', side_effect=Exception)
    def test_main_makedirs_exception(self, mock_method):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True), \
                self.assertRaises(Exception):
            self._test_main(TestCasesBase.TestCasesBase.EX_RUN_ERROR,
                            mock_method)

    @mock.patch('os.makedirs', side_effect=OSError)
    def test_main_makedirs_oserror(self, mock_method):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True):
            self._test_main(TestCasesBase.TestCasesBase.EX_RUN_ERROR,
                            mock_method)

    @mock.patch('robot.run', side_effect=RobotError)
    @mock.patch('os.makedirs')
    def test_main_robot_run_failed(self, *args):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True), \
                self.assertRaises(RobotError):
            self._test_main(TestCasesBase.TestCasesBase.EX_RUN_ERROR, *args)

    @mock.patch('robot.run')
    @mock.patch('os.makedirs')
    def test_main_parse_results_failed(self, *args):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True), \
                mock.patch.object(self.test, 'parse_results',
                                  side_effect=RobotError):
            self._test_main(TestCasesBase.TestCasesBase.EX_RUN_ERROR, *args)

    @mock.patch('os.remove', side_effect=Exception)
    @mock.patch('robot.run')
    @mock.patch('os.makedirs')
    def test_main_remove_exception(self, *args):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True), \
                mock.patch.object(self.test, 'parse_results'), \
                self.assertRaises(Exception):
            self._test_main(TestCasesBase.TestCasesBase.EX_OK, *args)

    @mock.patch('os.remove')
    @mock.patch('robot.run')
    @mock.patch('os.makedirs')
    def test_main(self, *args):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True), \
                mock.patch.object(self.test, 'parse_results'):
            self._test_main(TestCasesBase.TestCasesBase.EX_OK, *args)

    @mock.patch('os.remove')
    @mock.patch('robot.run')
    @mock.patch('os.makedirs', side_effect=OSError(errno.EEXIST, ''))
    def test_main_makedirs_oserror17(self, *args):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True), \
                mock.patch.object(self.test, 'parse_results'):
            self._test_main(TestCasesBase.TestCasesBase.EX_OK, *args)

    @mock.patch('os.remove')
    @mock.patch('robot.run', return_value=1)
    @mock.patch('os.makedirs')
    def test_main_testcases_in_failure(self, *args):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True), \
                mock.patch.object(self.test, 'parse_results'):
            self._test_main(TestCasesBase.TestCasesBase.EX_OK, *args)

    @mock.patch('os.remove', side_effect=OSError)
    @mock.patch('robot.run')
    @mock.patch('os.makedirs')
    def test_main_remove_oserror(self, *args):
        with mock.patch.object(self.test, 'set_robotframework_vars',
                               return_value=True), \
                mock.patch.object(self.test, 'parse_results'):
            self._test_main(TestCasesBase.TestCasesBase.EX_OK, *args)

    def _test_run_missing_env_var(self, var):
        if var == 'OS_USERNAME':
            ft_constants.OS_USERNAME = None
        elif var == 'OS_PASSWORD':
            ft_constants.OS_PASSWORD = None
        elif var == 'OS_TENANT_NAME':
            ft_constants.OS_TENANT_NAME = None

        self.assertEqual(self.test.run(),
                         TestCasesBase.TestCasesBase.EX_RUN_ERROR)

    def _test_run(self, status=TestCasesBase.TestCasesBase.EX_OK,
                  exception=None, odlip="127.0.0.3", odlwebport="8080"):
        with mock.patch('functest.utils.openstack_utils.get_keystone_client',
                        return_value=self._get_fake_keystone_client()):
            if exception:
                self.test.main = mock.Mock(side_effect=exception)
            else:
                self.test.main = mock.Mock(return_value=status)
            self.assertEqual(self.test.run(), status)
            self.test.main.assert_called_once_with(
                keystoneip=self._keystone_ip, neutronip=self._neutron_ip,
                odlip=odlip, odlpassword=self._odl_password,
                odlrestconfport=self._odl_restconfport,
                odlusername=self._odl_username, odlwebport=odlwebport,
                ospassword=self._os_password, ostenantname=self._os_tenantname,
                osusername=self._os_username)

    def test_run_missing_os_username(self):
        self._test_run_missing_env_var("OS_USERNAME")

    def test_run_missing_os_password(self):
        self._test_run_missing_env_var("OS_PASSWORD")

    def test_run_missing_os_tenant_name(self):
        self._test_run_missing_env_var("OS_TENANT_NAME")

    def test_run_main_false(self):
        ft_constants.CI_INSTALLER_TYPE = None
        ft_constants.SDN_CONTROLLER_IP = self._sdn_controller_ip
        self._test_run(TestCasesBase.TestCasesBase.EX_RUN_ERROR,
                       odlip=self._sdn_controller_ip,
                       odlwebport=self._odl_webport)

    def test_run_main_exception(self):
        ft_constants.CI_INSTALLER_TYPE = None
        ft_constants.SDN_CONTROLLER_IP = self._sdn_controller_ip
        with self.assertRaises(Exception):
            self._test_run(status=TestCasesBase.TestCasesBase.EX_RUN_ERROR,
                           exception=Exception(),
                           odlip=self._sdn_controller_ip,
                           odlwebport=self._odl_webport)

    def test_run_missing_sdn_controller_ip(self):
        with mock.patch('functest.utils.openstack_utils.get_keystone_client',
                        return_value=self._get_fake_keystone_client()):
            ft_constants.CI_INSTALLER_TYPE = None
            ft_constants.SDN_CONTROLLER_IP = None
            self.assertEqual(self.test.run(),
                             TestCasesBase.TestCasesBase.EX_RUN_ERROR)

    def test_run_without_installer_type(self):
        ft_constants.SDN_CONTROLLER_IP = self._sdn_controller_ip
        ft_constants.CI_INSTALLER_TYPE = None
        self._test_run(TestCasesBase.TestCasesBase.EX_OK,
                       odlip=self._sdn_controller_ip,
                       odlwebport=self._odl_webport)

    def test_run_fuel(self):
        ft_constants.CI_INSTALLER_TYPE = "fuel"
        self._test_run(TestCasesBase.TestCasesBase.EX_OK,
                       odlip=self._neutron_ip, odlwebport='8282')

    def test_run_apex_missing_sdn_controller_ip(self):
        with mock.patch('functest.utils.openstack_utils.get_keystone_client',
                        return_value=self._get_fake_keystone_client()):
            ft_constants.CI_INSTALLER_TYPE = "apex"
            ft_constants.SDN_CONTROLLER_IP = None
            self.assertEqual(self.test.run(),
                             TestCasesBase.TestCasesBase.EX_RUN_ERROR)

    def test_run_apex(self):
        ft_constants.SDN_CONTROLLER_IP = self._sdn_controller_ip
        ft_constants.CI_INSTALLER_TYPE = "apex"
        self._test_run(TestCasesBase.TestCasesBase.EX_OK,
                       odlip=self._sdn_controller_ip, odlwebport='8181')

    def test_run_joid_missing_sdn_controller(self):
        with mock.patch('functest.utils.openstack_utils.get_keystone_client',
                        return_value=self._get_fake_keystone_client()):
            ft_constants.CI_INSTALLER_TYPE = "joid"
            ft_constants.SDN_CONTROLLER = None
            self.assertEqual(self.test.run(),
                             TestCasesBase.TestCasesBase.EX_RUN_ERROR)

    def test_run_joid(self):
        ft_constants.SDN_CONTROLLER = self._sdn_controller_ip
        ft_constants.CI_INSTALLER_TYPE = "joid"
        self._test_run(TestCasesBase.TestCasesBase.EX_OK,
                       odlip=self._sdn_controller_ip,
                       odlwebport=self._odl_webport)

    def test_run_compass(self, *args):
        ft_constants.CI_INSTALLER_TYPE = "compass"
        self._test_run(TestCasesBase.TestCasesBase.EX_OK,
                       odlip=self._neutron_ip, odlwebport='8181')


if __name__ == "__main__":
    unittest.main(verbosity=2)