Subscription and notification interfaces add http basic auth support
[modeling/etsicatalog.git] / catalog / packages / tests / test_vnf_pkg_subscription.py
1 # Copyright (C) 2019 Verizon. All Rights Reserved
2 #
3 # Licensed under the Apache License, Version 2.0 (the "License");
4 # you may not use this file except in compliance with the License.
5 # You may obtain a copy of the License at
6 #
7 #         http://www.apache.org/licenses/LICENSE-2.0
8 #
9 # Unless required by applicable law or agreed to in writing, software
10 # distributed under the License is distributed on an "AS IS" BASIS,
11 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 # See the License for the specific language governing permissions and
13 # limitations under the License.
14
15 import json
16 import os
17 import uuid
18
19 import mock
20 from django.test import TestCase
21 from rest_framework import status
22 from rest_framework.test import APIClient
23 from requests.auth import HTTPBasicAuth
24
25 import catalog.pub.utils.timeutil
26 from catalog.packages import const
27 from catalog.packages.biz.notificationsutil import PkgNotifications
28 from catalog.packages.biz.vnf_pkg_subscription import QuerySubscription, TerminateSubscription
29 from catalog.pub.config import config as pub_config
30 from catalog.pub.config.config import CATALOG_ROOT_PATH
31 from catalog.pub.database.models import VnfPkgSubscriptionModel, VnfPackageModel
32 from catalog.pub.exceptions import SubscriptionDoesNotExistsException
33 from catalog.pub.utils import toscaparser
34 from .const import vnf_subscription_data, vnfd_data
35
36
37 class TestNfPackageSubscription(TestCase):
38     def setUp(self):
39         self.client = APIClient()
40         VnfPkgSubscriptionModel.objects.filter().delete()
41         self.vnf_subscription_data = vnf_subscription_data
42
43     def tearDown(self):
44         pass
45
46     @mock.patch("requests.get")
47     @mock.patch.object(uuid, 'uuid4')
48     def test_create_vnf_subscription(self, mock_uuid4, mock_requests):
49         temp_uuid = "99442b18-a5c7-11e8-998c-bf1755941f13"
50         mock_requests.return_value.status_code = 204
51         mock_requests.get.status_code = 204
52         mock_uuid4.return_value = temp_uuid
53         response = self.client.post(
54             "/api/vnfpkgm/v1/subscriptions",
55             data=self.vnf_subscription_data,
56             format='json'
57         )
58         self.assertEqual(201, response.status_code)
59         self.assertEqual(
60             self.vnf_subscription_data["callbackUri"],
61             response.data["callbackUri"]
62         )
63         self.assertEqual(temp_uuid, response.data["id"])
64
65     @mock.patch("requests.get")
66     @mock.patch.object(uuid, 'uuid4')
67     def test_duplicate_subscriptions(self, mock_uuid4, mock_requests):
68         temp_uuid = "99442b18-a5c7-11e8-998c-bf1755941f13"
69         temp1_uuid = "00342b18-a5c7-11e8-998c-bf1755941f12"
70         mock_requests.return_value.status_code = 204
71         mock_requests.get.status_code = 204
72         mock_uuid4.side_effect = [temp_uuid, temp1_uuid]
73         response = self.client.post(
74             "/api/vnfpkgm/v1/subscriptions",
75             data=self.vnf_subscription_data,
76             format='json'
77         )
78         self.assertEqual(201, response.status_code)
79         self.assertEqual(
80             self.vnf_subscription_data["callbackUri"],
81             response.data["callbackUri"]
82         )
83         self.assertEqual(temp_uuid, response.data["id"])
84         temp_uuid = "00442b18-a5c7-11e8-998c-bf1755941f12"
85         mock_requests.return_value.status_code = 204
86         mock_requests.get.status_code = 204
87         mock_uuid4.return_value = temp_uuid
88         response = self.client.post(
89             "/api/vnfpkgm/v1/subscriptions",
90             data=self.vnf_subscription_data,
91             format='json'
92         )
93         self.assertEqual(303, response.status_code)
94
95     @mock.patch("requests.get")
96     @mock.patch.object(uuid, 'uuid4')
97     def test_get_subscriptions(self, mock_uuid4, mock_requests):
98         temp_uuid = "99442b18-a5c7-11e8-998c-bf1755941f13"
99         mock_requests.return_value.status_code = 204
100         mock_requests.get.status_code = 204
101         mock_uuid4.return_value = temp_uuid
102         self.client.post(
103             "/api/vnfpkgm/v1/subscriptions",
104             data=self.vnf_subscription_data,
105             format='json'
106         )
107         response = self.client.get(
108             "/api/vnfpkgm/v1/subscriptions?usageState=IN_USE",
109             format='json'
110         )
111         self.assertEqual(200, response.status_code)
112         self.assertEqual(1, len(response.data))
113
114     @mock.patch("requests.get")
115     @mock.patch.object(uuid, 'uuid4')
116     def test_get_subscriptions_with_invalid_params(self, mock_uuid4, mock_requests):
117         temp_uuid = "99442b18-a5c7-11e8-998c-bf1755941f13"
118         mock_requests.return_value.status_code = 204
119         mock_requests.get.status_code = 204
120         mock_uuid4.return_value = temp_uuid
121         self.client.post(
122             "/api/vnfpkgm/v1/subscriptions",
123             data=self.vnf_subscription_data,
124             format='json'
125         )
126         response = self.client.get(
127             "/api/vnfpkgm/v1/subscriptions?dummy=dummy",
128             format='json'
129         )
130         self.assertEqual(400, response.status_code)
131
132     @mock.patch("requests.get")
133     @mock.patch.object(uuid, 'uuid4')
134     def test_get_subscription_with_id(self, mock_uuid4, mock_requests):
135         temp_uuid = "99442b18-a5c7-11e8-998c-bf1755941f13"
136         mock_requests.return_value.status_code = 204
137         mock_requests.get.status_code = 204
138         mock_uuid4.return_value = temp_uuid
139         self.client.post(
140             "/api/vnfpkgm/v1/subscriptions",
141             data=self.vnf_subscription_data,
142             format='json'
143         )
144         response = self.client.get(
145             "/api/vnfpkgm/v1/subscriptions/%s" % temp_uuid,
146             format='json'
147         )
148         self.assertEqual(200, response.status_code)
149         self.assertEqual(temp_uuid, response.data["id"])
150
151     @mock.patch("requests.get")
152     @mock.patch.object(uuid, 'uuid4')
153     def test_get_subscription_with_id_not_exists(self, mock_uuid4, mock_requests):
154         temp_uuid = "99442b18-a5c7-11e8-998c-bf1755941f13"
155         dummy_uuid = str(uuid.uuid4())
156         mock_requests.return_value.status_code = 204
157         mock_requests.get.status_code = 204
158         mock_uuid4.return_value = temp_uuid
159         self.client.post(
160             "/api/vnfpkgm/v1/subscriptions",
161             data=self.vnf_subscription_data,
162             format='json'
163         )
164         response = self.client.get(
165             "/api/vnfpkgm/v1/subscriptions/%s" % dummy_uuid,
166             format='json'
167         )
168         self.assertEqual(404, response.status_code)
169
170     @mock.patch("requests.get")
171     @mock.patch.object(uuid, 'uuid4')
172     def test_delete_subscription_with_id(self, mock_uuid4, mock_requests):
173         temp_uuid = "99442b18-a5c7-11e8-998c-bf1755941f13"
174         dummy_uuid = str(uuid.uuid4())
175         mock_requests.return_value.status_code = 204
176         mock_requests.get.status_code = 204
177         mock_uuid4.return_value = temp_uuid
178         self.client.post(
179             "/api/vnfpkgm/v1/subscriptions",
180             data=self.vnf_subscription_data,
181             format='json'
182         )
183         self.client.get(
184             "/api/vnfpkgm/v1/subscriptions/%s" % dummy_uuid,
185             format='json'
186         )
187         response = self.client.delete("/api/vnfpkgm/v1/subscriptions/%s" % temp_uuid)
188         self.assertEqual(204, response.status_code)
189
190     @mock.patch("requests.get")
191     @mock.patch.object(uuid, 'uuid4')
192     def test_delete_subscription_with_id_not_exists(self, mock_uuid4, mock_requests):
193         dummy_uuid = str(uuid.uuid4())
194         response = self.client.delete("/api/vnfpkgm/v1/subscriptions/%s" % dummy_uuid)
195         self.assertEqual(404, response.status_code)
196
197     @mock.patch("requests.get")
198     @mock.patch.object(toscaparser, 'parse_vnfd')
199     @mock.patch("requests.post")
200     @mock.patch("uuid.uuid4")
201     @mock.patch.object(catalog.pub.utils.timeutil, "now_time")
202     def test_vnfpkg_subscript_notify(self, mock_nowtime, mock_uuid, mock_requests_post, mock_parse_vnfd,
203                                      mock_requests_get):
204         mock_nowtime.return_value = "2019-02-16 14:41:16"
205         uuid_subscriptid = "99442b18-a5c7-11e8-998c-bf1755941f13"
206         uuid_vnfPackageId = "3fa85f64-5717-4562-b3fc-2c963f66afa6"
207         uuid_vnfdid = "00342b18-a5c7-11e8-998c-bf1755941f12"
208         mock_uuid.side_effect = [uuid_subscriptid, "1111"]
209         mock_requests_get.return_value.status_code = 204
210         mock_parse_vnfd.return_value = json.JSONEncoder().encode(vnfd_data)
211
212         response = self.client.post(
213             "/api/vnfpkgm/v1/subscriptions",
214             data=vnf_subscription_data,
215             format='json')
216         self.assertEqual(201, response.status_code)
217
218         data = {'file': open(os.path.join(CATALOG_ROOT_PATH, "empty.txt"), "rt")}
219         VnfPackageModel.objects.create(
220             vnfPackageId=uuid_vnfPackageId,
221             onboardingState="CREATED"
222         )
223
224         response = self.client.put("/api/vnfpkgm/v1/vnf_packages/%s/package_content" % uuid_vnfPackageId, data=data)
225         vnf_pkg = VnfPackageModel.objects.filter(vnfPackageId=uuid_vnfPackageId)
226         self.assertEqual(uuid_vnfdid, vnf_pkg[0].vnfdId)
227         self.assertEqual(const.PKG_STATUS.ONBOARDED, vnf_pkg[0].onboardingState)
228         self.assertEqual(response.status_code, status.HTTP_202_ACCEPTED)
229
230         expect_notification = {
231             'id': "1111",
232             'notificationType': const.PKG_NOTIFICATION_TYPE.ONBOARDING,
233             'timeStamp': "2019-02-16 14:41:16",
234             'vnfPkgId': uuid_vnfPackageId,
235             'vnfdId': uuid_vnfdid,
236             "subscriptionId": uuid_subscriptid,
237             '_links': {
238                 'subscription': {
239                     'href': 'http://%s:%s/%s%s' % (pub_config.MSB_SERVICE_IP,
240                                                    pub_config.MSB_SERVICE_PORT,
241                                                    const.VNFPKG_SUBSCRIPTION_ROOT_URI,
242                                                    uuid_subscriptid)},
243                 'vnfPackage': {
244                     'href': 'http://%s:%s/%s/vnf_packages/%s' % (pub_config.MSB_SERVICE_IP,
245                                                                  pub_config.MSB_SERVICE_PORT,
246                                                                  const.PKG_URL_PREFIX,
247                                                                  uuid_vnfPackageId)
248                 }
249             }
250         }
251         mock_requests_post.assert_called_with(vnf_subscription_data["callbackUri"], data=expect_notification,
252                                               headers={'Connection': 'close'}, auth=HTTPBasicAuth("admin", "pwd1234"))
253
254     def test_service_query_single_subscription_not_found(self):
255         try:
256             subscription_id = "test_not_found"
257             QuerySubscription().query_single_subscription(subscription_id)
258         except SubscriptionDoesNotExistsException as e:
259             self.assertEqual("Subscription with ID: %s does not exist" % subscription_id, e.args[0])
260
261     def test_service_delete_single_subscription_not_found(self):
262         try:
263             subscription_id = "test_not_found"
264             TerminateSubscription().terminate(subscription_id)
265         except SubscriptionDoesNotExistsException as e:
266             self.assertEqual("Subscription with ID: %s does not exist" % subscription_id, e.args[0])
267
268
269 class NotificationTest(TestCase):
270     def setUp(self):
271         VnfPackageModel.objects.all().delete()
272         VnfPkgSubscriptionModel.objects.all().delete()
273
274     def tearDown(self):
275         VnfPackageModel.objects.all().delete()
276         VnfPkgSubscriptionModel.objects.all().delete()
277
278     @mock.patch("requests.post")
279     @mock.patch("uuid.uuid4")
280     @mock.patch.object(catalog.pub.utils.timeutil, "now_time")
281     def test_vnfpkg_manual_notify(self, mock_nowtime, mock_uuid, mock_requests_post):
282         VnfPackageModel(vnfPackageId="vnfpkgid1",
283                         vnfdId="vnfdid1"
284                         ).save()
285
286         VnfPkgSubscriptionModel(subscription_id="1",
287                                 callback_uri="http://127.0.0.1/self",
288                                 notification_types=const.NOTIFICATION_TYPES,
289                                 vnfd_id="vnfdid1",
290                                 vnf_pkg_id="vnfpkgid1"
291                                 ).save()
292         mock_nowtime.return_value = "2019-12-16 14:41:16"
293         mock_uuid.return_value = "1111"
294         notify = PkgNotifications(const.PKG_NOTIFICATION_TYPE.CHANGE, "vnfpkgid1",
295                                   const.PKG_CHANGE_TYPE.OP_STATE_CHANGE, operational_state=None)
296
297         notify.send_notification()
298         expect_callbackuri = "http://127.0.0.1/self"
299         expect_notification = {
300             'id': "1111",
301             'notificationType': const.PKG_NOTIFICATION_TYPE.CHANGE,
302             'timeStamp': "2019-12-16 14:41:16",
303             'vnfPkgId': "vnfpkgid1",
304             'vnfdId': "vnfdid1",
305             'changeType': const.PKG_CHANGE_TYPE.OP_STATE_CHANGE,
306             'operationalState': None,
307             "subscriptionId": "1",
308             '_links': {
309                 'subscription': {
310                     'href': 'http://%s:%s/%s%s' % (pub_config.MSB_SERVICE_IP,
311                                                    pub_config.MSB_SERVICE_PORT,
312                                                    const.VNFPKG_SUBSCRIPTION_ROOT_URI,
313                                                    "1")},
314                 'vnfPackage': {
315                     'href': 'http://%s:%s/%s/vnf_packages/%s' % (pub_config.MSB_SERVICE_IP,
316                                                                  pub_config.MSB_SERVICE_PORT,
317                                                                  const.PKG_URL_PREFIX,
318                                                                  "vnfpkgid1")
319                 }
320             }
321         }
322         mock_requests_post.assert_called_with(expect_callbackuri, data=expect_notification,
323                                               headers={'Connection': 'close'})