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:
parent
4ebf7f56b6
commit
70a0cfbbea
@ -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):
|
||||||
|
@ -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">' +
|
||||||
|
@ -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"
|
||||||
|
@ -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')
|
||||||
|
@ -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).
|
Loading…
x
Reference in New Issue
Block a user