# Licensed to the Apache Software Foundation (ASF) under one or more # contributor license agreements. See the NOTICE file distributed with # this work for additional information regarding copyright ownership. # The ASF licenses this file to You 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. import pytest import mock from aria.cli.env import _Environment from aria.core import Core from aria.exceptions import DependentActiveExecutionsError, DependentAvailableNodesError from aria.modeling.exceptions import ParameterException from aria.storage import exceptions as storage_exceptions from .base_test import ( # pylint: disable=unused-import TestCliBase, raise_exception, assert_exception_raised, mock_storage ) from ..mock import models as mock_models class TestServicesList(TestCliBase): @pytest.mark.parametrize('sort_by, order, sort_by_in_output, order_in_output', [ ('', '', 'created_at', 'asc'), ('', ' --descending', 'created_at', 'desc'), (' --sort-by name', '', 'name', 'asc'), (' --sort-by name', ' --descending', 'name', 'desc') ]) def test_no_specified_service_template(self, monkeypatch, mock_storage, sort_by, order, sort_by_in_output, order_in_output): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services list{sort_by}{order}'.format(sort_by=sort_by, order=order)) assert 'Listing all services...' in self.logger_output_string assert 'Listing services for service template' not in self.logger_output_string mock_storage.service.list.assert_called_once_with(sort={sort_by_in_output: order_in_output}, filters={}) assert 'Services:' in self.logger_output_string assert mock_models.SERVICE_TEMPLATE_NAME in self.logger_output_string assert mock_models.SERVICE_NAME in self.logger_output_string @pytest.mark.parametrize('sort_by, order, sort_by_in_output, order_in_output', [ ('', '', 'created_at', 'asc'), ('', ' --descending', 'created_at', 'desc'), (' --sort-by name', '', 'name', 'asc'), (' --sort-by name', ' --descending', 'name', 'desc') ]) def test_specified_service_template(self, monkeypatch, mock_storage, sort_by, order, sort_by_in_output, order_in_output): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services list -t test_st{sort_by}{order}'.format(sort_by=sort_by, order=order)) assert 'Listing services for service template test_st...' in self.logger_output_string assert 'Listing all services...' not in self.logger_output_string mock_storage.service.list.assert_called_once_with(sort={sort_by_in_output: order_in_output}, filters={'service_template': mock.ANY}) assert 'Services:' in self.logger_output_string assert mock_models.SERVICE_TEMPLATE_NAME in self.logger_output_string assert mock_models.SERVICE_NAME in self.logger_output_string class TestServicesCreate(TestCliBase): def test_header_string(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services create -t test_st test_s') assert 'Creating new service from service template test_st...' in self.logger_output_string def test_no_exception(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) m = mock.MagicMock(return_value=mock_models.create_service_with_dependencies()) monkeypatch.setattr(Core, 'create_service', m) self.invoke('services create -t test_st test_s') assert "Service created. The service's name is test_s" in self.logger_output_string def test_raises_storage_error_resulting_from_name_uniqueness(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) monkeypatch.setattr(Core, 'create_service', raise_exception(storage_exceptions.NotFoundError, msg='UNIQUE constraint failed')) assert_exception_raised( self.invoke('services create -t test_st test_s'), expected_exception=storage_exceptions.NotFoundError, expected_msg='There already a exists a service with the same name') assert "Service created. The service's name is test_s" not in self.logger_output_string def test_raises_other_storage_error(self, monkeypatch, mock_object): monkeypatch.setattr(_Environment, 'model_storage', mock_object) monkeypatch.setattr(Core, 'create_service', raise_exception(storage_exceptions.NotFoundError)) assert_exception_raised( self.invoke('services create -t test_st test_s'), expected_exception=storage_exceptions.NotFoundError) assert "Service created. The service's name is test_s" not in self.logger_output_string def test_raises_inputs_exception(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) monkeypatch.setattr(Core, 'create_service', raise_exception(ParameterException)) assert_exception_raised( self.invoke('services create -t with_inputs test_s'), expected_exception=ParameterException) assert "Service created. The service's name is test_s" not in self.logger_output_string class TestServicesDelete(TestCliBase): def test_header_string(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services delete test_s') assert 'Deleting service test_s...' in self.logger_output_string def test_delete_no_exception(self, monkeypatch, mock_storage, mock_object): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) monkeypatch.setattr(Core, 'delete_service', mock_object) self.invoke('services delete test_s') assert 'Service test_s deleted' in self.logger_output_string def test_delete_active_execution_error(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) mock_service_with_execution = \ mock.MagicMock(return_value=mock_models.create_service_with_dependencies( include_execution=True)) monkeypatch.setattr(mock_storage.service, 'get', mock_service_with_execution) assert_exception_raised( self.invoke('services delete test_s'), expected_exception=DependentActiveExecutionsError, expected_msg="Can't delete service `{name}` - there is an active execution " "for this service. Active execution ID: 1".format( name=mock_models.SERVICE_NAME)) def test_delete_available_nodes_error(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) assert_exception_raised( self.invoke('services delete test_s'), expected_exception=DependentAvailableNodesError, expected_msg="Can't delete service `{name}` - there are available nodes " "for this service. Available node IDs: 1".format( name=mock_models.SERVICE_NAME)) def test_delete_available_nodes_error_with_force(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services delete service_with_available_nodes --force') assert mock_storage.service.delete.call_count == 1 assert 'Service service_with_available_nodes deleted' in self.logger_output_string class TestServicesOutputs(TestCliBase): def test_header_string(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services outputs test_s') assert 'Showing outputs for service test_s...' in self.logger_output_string def test_outputs_no_outputs(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services outputs service_with_no_outputs') assert 'No outputs' in self.logger_output_string assert 'output1' not in self.logger_output_string assert 'value1' not in self.logger_output_string def test_outputs_one_output(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) s = mock_models.create_service_with_dependencies(include_output=True) monkeypatch.setattr(mock_storage.service, 'get_by_name', mock.MagicMock(return_value=s)) self.invoke('services outputs test_s') assert 'output1' in self.logger_output_string assert 'value1' in self.logger_output_string assert 'No outputs' not in self.logger_output_string class TestServicesInputs(TestCliBase): def test_header_string(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services inputs test_s') assert 'Showing inputs for service test_s...' in self.logger_output_string def test_inputs_no_inputs(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) self.invoke('services inputs service_with_no_inputs') assert 'No inputs' in self.logger_output_string assert 'input1' not in self.logger_output_string assert 'value1' not in self.logger_output_string def test_inputs_one_input(self, monkeypatch, mock_storage): monkeypatch.setattr(_Environment, 'model_storage', mock_storage) s = mock_models.create_service_with_dependencies(include_input=True) monkeypatch.setattr(mock_storage.service, 'get_by_name', mock.MagicMock(return_value=s)) self.invoke('services inputs test_s') assert 'input1' in self.logger_output_string assert 'value1' in self.logger_output_string assert 'No inputs' not in self.logger_output_string