Add ability to manage several environments at once in dashboard

Patch makes it possible for user to select several environments and
execute one of the three actions to them: delete, abandon or
deploy.
Tests to check new functionality are added.

Change-Id: I2b395cc2ed80a59d39eb984c920f83ac6b687572
Closes-bug: #1582193
This commit is contained in:
Valerii Kovalchuk 2016-03-31 14:18:33 +03:00
parent 4ebf7f56b6
commit 70a0cfbbea
5 changed files with 121 additions and 7 deletions

View File

@ -48,6 +48,16 @@ def _get_environment_status_and_version(request, table):
return status, version return status, version
def _check_row_actions_allowed(action, request):
envs = action.table.data
if not envs:
return False
for env in envs:
if action.allowed(request, env):
return True
return False
class AddApplication(tables.LinkAction): class AddApplication(tables.LinkAction):
name = 'AddApplication' name = 'AddApplication'
verbose_name = _('Add Component') verbose_name = _('Add Component')
@ -107,10 +117,13 @@ class DeleteEnvironment(tables.DeleteAction):
) )
def allowed(self, request, environment): def allowed(self, request, environment):
if environment: # table action case: action allowed if any row action allowed
return environment.status not in (consts.STATUS_ID_DEPLOYING, if not environment:
consts.STATUS_ID_DELETING) return _check_row_actions_allowed(self, request)
return True
# row action case
return environment.status not in (consts.STATUS_ID_DEPLOYING,
consts.STATUS_ID_DELETING)
def action(self, request, environment_id): def action(self, request, environment_id):
try: try:
@ -152,6 +165,12 @@ class AbandonEnvironment(tables.DeleteAction):
* environment is new * environment is new
* app added to env, but not deploy is not started * app added to env, but not deploy is not started
""" """
# table action case: action allowed if any row action allowed
if not environment:
return _check_row_actions_allowed(self, request)
# row action case
status = getattr(environment, 'status', None) status = getattr(environment, 'status', None)
if status in [consts.STATUS_ID_NEW, consts.STATUS_ID_PENDING]: if status in [consts.STATUS_ID_NEW, consts.STATUS_ID_PENDING]:
return False return False
@ -234,6 +253,12 @@ class DeployEnvironment(tables.BatchAction):
* no new services added to the environment (after env creation * no new services added to the environment (after env creation
or successful deploy or delete failure) or successful deploy or delete failure)
""" """
# table action case: action allowed if any row action allowed
if not environment:
return _check_row_actions_allowed(self, request)
# row action case
status = getattr(environment, 'status', None) status = getattr(environment, 'status', None)
if (status != consts.STATUS_ID_DEPLOY_FAILURE and if (status != consts.STATUS_ID_DEPLOY_FAILURE and
not environment.has_new_services): not environment.has_new_services):
@ -371,10 +396,10 @@ class EnvironmentsTable(tables.DataTable):
row_class = UpdateEnvironmentRow row_class = UpdateEnvironmentRow
status_columns = ['status'] status_columns = ['status']
no_data_message = _('NO ENVIRONMENTS') no_data_message = _('NO ENVIRONMENTS')
table_actions = (CreateEnvironment,) table_actions = (CreateEnvironment, DeployEnvironment,
DeleteEnvironment, AbandonEnvironment)
row_actions = (ShowEnvironmentServices, DeployEnvironment, row_actions = (ShowEnvironmentServices, DeployEnvironment,
DeleteEnvironment, AbandonEnvironment) DeleteEnvironment, AbandonEnvironment)
multi_select = False
def get_service_details_link(service): def get_service_details_link(service):

View File

@ -46,7 +46,7 @@ $(function() {
} }
var $newEnvTr = $('<tr class="new_env">' + var $newEnvTr = $('<tr class="new_env">' +
'<td id="input_create_env" class="normal_column row"></td>' + '<td id="input_create_env" class="normal_column row" colspan="2"></td>' +
'<td class="normal_column">New</td>' + '<td class="normal_column">New</td>' +
'<td class="actions_column">' + '<td class="actions_column">' +
'<div class="btn-group">' + '<div class="btn-group">' +

View File

@ -17,6 +17,7 @@ DeleteImageMeta = TestImage + "//td//button[contains(text(), 'Delete Metadata')]
ImageMeta = "//dl[dt[contains(text(), 'murano_image_info')]]/dd" ImageMeta = "//dl[dt[contains(text(), 'murano_image_info')]]/dd"
More = "//tr[contains(@id, '{0}__row__{1}')]//a[contains(@class, dropdown-toggle) and @href='#']" # noqa More = "//tr[contains(@id, '{0}__row__{1}')]//a[contains(@class, dropdown-toggle) and @href='#']" # noqa
Status = "//td[contains(text(), '{0}')]" Status = "//td[contains(text(), '{0}')]"
EnvStatus = "//tr[contains(@data-display, '{0}')]/td[contains(text(), '{1}')]"
CellStatus = "//td[contains(@class, 'status_{0}')]" CellStatus = "//td[contains(@class, 'status_{0}')]"
Row = "//tr[contains(@id, 'services__row__{0}')]" Row = "//tr[contains(@id, 'services__row__{0}')]"
ErrorMessage = '//span[contains(@class, "help-block") and contains(text(), "{0}")]' # noqa ErrorMessage = '//span[contains(@class, "help-block") and contains(text(), "{0}")]' # noqa
@ -31,11 +32,15 @@ HotFlavorField = '//div[contains(@class, "has-error")]//input'
ButtonSubmit = ".//*[@name='wizard_goto_step'][2]" ButtonSubmit = ".//*[@name='wizard_goto_step'][2]"
InputSubmit = "//input[@type='submit']" InputSubmit = "//input[@type='submit']"
ConfirmDeletion = "//div[@class='modal-footer']//a[contains(text(), 'Delete')]" # noqa ConfirmDeletion = "//div[@class='modal-footer']//a[contains(text(), 'Delete')]" # noqa
ConfirmAbandon = "//div[@class='modal-footer']//a[contains(text(), 'Abandon')]" # noqa
UploadPackage = 'packages__action_upload_package' UploadPackage = 'packages__action_upload_package'
ImportBundle = 'packages__action_import_bundle' ImportBundle = 'packages__action_import_bundle'
CreateEnvironment = ".add_env .btn" CreateEnvironment = ".add_env .btn"
DeployEnvironment = "services__action_deploy_env" DeployEnvironment = "services__action_deploy_env"
DeleteEnvironment = "//button[contains(@id, 'action_delete')]" DeleteEnvironment = "//button[contains(@id, 'action_delete')]"
DeployEnvironments = ".btn#environments__action_deploy"
DeleteEnvironments = ".btn#environments__action_delete"
AbandonEnvironments = ".btn#environments__action_abandon"
ConfirmCreateEnvironment = 'confirm_create_env' ConfirmCreateEnvironment = 'confirm_create_env'
AddComponent = "services__action_AddApplication" AddComponent = "services__action_AddApplication"
AddCategory = "categories__action_add_category" AddCategory = "categories__action_add_category"

View File

@ -2071,3 +2071,81 @@ class TestSuiteCategoriesPagination(base.PackageTestCase):
self.check_element_on_page(by.By.XPATH, c.Status.format(name)) self.check_element_on_page(by.By.XPATH, c.Status.format(name))
if i != len(pages_itself): if i != len(pages_itself):
self.driver.find_element_by_xpath(c.PrevBtn).click() self.driver.find_element_by_xpath(c.PrevBtn).click()
class TestSuiteMultipleEnvironments(base.ApplicationTestCase):
def test_create_two_environments_and_delete_them_at_once(self):
"""Test check ability to create and delete multiple environments
Scenario:
1. Create two environments
2. Navigate to environment list
3. Check created environments
4. Delete created environments at once
"""
self.go_to_submenu('Environments')
self.create_environment('test_create_del_env_1')
self.go_to_submenu('Environments')
self.create_environment('test_create_del_env_2', by_id=True)
self.go_to_submenu('Environments')
self.driver.find_element_by_css_selector(
"label[for=ui-id-1]").click()
self.driver.find_element_by_css_selector(
c.DeleteEnvironments).click()
self.driver.find_element_by_xpath(c.ConfirmDeletion).click()
self.wait_for_alert_message()
self.check_element_not_on_page(by.By.LINK_TEXT,
'test_create_del_env_1')
self.check_element_not_on_page(by.By.LINK_TEXT,
'test_create_del_env_2')
def test_deploy_two_environments_at_once(self):
"""Test check ability to deploy multiple environments
Scenario:
1. Add two apps to different environments
2. Navigate to environment list
3. Check created environments
4. Deploy created environments at once
"""
self.add_app_to_env(self.mockapp_id)
self.add_app_to_env(self.mockapp_id)
self.go_to_submenu('Environments')
self.driver.find_element_by_css_selector(
"label[for=ui-id-1]").click()
self.driver.find_element_by_css_selector(
c.DeployEnvironments).click()
# check statuses of two environments
self.check_element_on_page(by.By.XPATH,
c.EnvStatus.format('quick-env-1', 'Ready'),
sec=90)
self.check_element_on_page(by.By.XPATH,
c.EnvStatus.format('quick-env-2', 'Ready'),
sec=90)
def test_abandon_two_environments_at_once(self):
"""Test check ability to abandon multiple environments
Scenario:
1. Add two apps to different environments
2. Navigate to environment list
3. Check created environments
4. Deploy created environments at once
5. Abandon environments before they are deployed
"""
self.add_app_to_env(self.mockapp_id)
self.add_app_to_env(self.mockapp_id)
self.go_to_submenu('Environments')
self.driver.find_element_by_css_selector(
"label[for=ui-id-1]").click()
self.driver.find_element_by_css_selector(
c.DeployEnvironments).click()
self.go_to_submenu('Environments')
self.driver.find_element_by_css_selector(
"label[for=ui-id-1]").click()
self.driver.find_element_by_css_selector(
c.AbandonEnvironments).click()
self.driver.find_element_by_xpath(c.ConfirmAbandon).click()
self.wait_for_alert_message()
self.check_element_not_on_page(by.By.LINK_TEXT, 'quick-env-1')
self.check_element_not_on_page(by.By.LINK_TEXT, 'quick-env-2')

View File

@ -0,0 +1,6 @@
---
features:
- Added possibility for user to select several environments and execute
one of the three actions to all of them at once (delete, abandon or
deploy).