mirror of
https://git.yoctoproject.org/poky
synced 2026-06-01 13:09:50 +00:00
bitbake: toaster: update selenium version and code syntax
Updated selenium version to latest 4.13.0, changed selenum specific version syntax elements to accomplish test success (Bitbake rev: 868c88a249ef4b9fe5a891e76e25e054e4fcd994) Signed-off-by: Marlon Rodriguez Garcia <marlon.rodriguez-garcia@savoirfairelinux.com> Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
This commit is contained in:
committed by
Richard Purdie
parent
08421d8985
commit
3ac4694fc3
@@ -1733,7 +1733,7 @@ class CustomImageRecipe(Recipe):
|
|||||||
packages_conf += "\""
|
packages_conf += "\""
|
||||||
|
|
||||||
base_recipe_path = self.get_base_recipe_file()
|
base_recipe_path = self.get_base_recipe_file()
|
||||||
if base_recipe_path:
|
if base_recipe_path and os.path.isfile(base_recipe_path):
|
||||||
base_recipe = open(base_recipe_path, 'r').read()
|
base_recipe = open(base_recipe_path, 'r').read()
|
||||||
else:
|
else:
|
||||||
# Pass back None to trigger error message to user
|
# Pass back None to trigger error message to user
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ import unittest
|
|||||||
|
|
||||||
from selenium import webdriver
|
from selenium import webdriver
|
||||||
from selenium.webdriver.support.ui import WebDriverWait
|
from selenium.webdriver.support.ui import WebDriverWait
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
from selenium.webdriver.common.desired_capabilities import DesiredCapabilities
|
from selenium.webdriver.common.desired_capabilities import DesiredCapabilities
|
||||||
from selenium.common.exceptions import NoSuchElementException, \
|
from selenium.common.exceptions import NoSuchElementException, \
|
||||||
StaleElementReferenceException, TimeoutException
|
StaleElementReferenceException, TimeoutException
|
||||||
@@ -32,9 +33,7 @@ def create_selenium_driver(cls,browser='chrome'):
|
|||||||
browser = env_browser
|
browser = env_browser
|
||||||
|
|
||||||
if browser == 'chrome':
|
if browser == 'chrome':
|
||||||
return webdriver.Chrome(
|
return webdriver.Chrome()
|
||||||
service_args=["--verbose", "--log-path=selenium.log"]
|
|
||||||
)
|
|
||||||
elif browser == 'firefox':
|
elif browser == 'firefox':
|
||||||
return webdriver.Firefox()
|
return webdriver.Firefox()
|
||||||
elif browser == 'marionette':
|
elif browser == 'marionette':
|
||||||
@@ -153,11 +152,11 @@ class SeleniumTestCaseBase(unittest.TestCase):
|
|||||||
|
|
||||||
def find(self, selector):
|
def find(self, selector):
|
||||||
""" Find single element by CSS selector """
|
""" Find single element by CSS selector """
|
||||||
return self.driver.find_element_by_css_selector(selector)
|
return self.driver.find_element(By.CSS_SELECTOR, selector)
|
||||||
|
|
||||||
def find_all(self, selector):
|
def find_all(self, selector):
|
||||||
""" Find all elements matching CSS selector """
|
""" Find all elements matching CSS selector """
|
||||||
return self.driver.find_elements_by_css_selector(selector)
|
return self.driver.find_elements(By.CSS_SELECTOR, selector)
|
||||||
|
|
||||||
def element_exists(self, selector):
|
def element_exists(self, selector):
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
# SPDX-License-Identifier: GPL-2.0-only
|
# SPDX-License-Identifier: GPL-2.0-only
|
||||||
#
|
#
|
||||||
|
|
||||||
import re
|
import re, time
|
||||||
|
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
@@ -15,6 +15,8 @@ from tests.browser.selenium_helpers import SeleniumTestCase
|
|||||||
|
|
||||||
from orm.models import BitbakeVersion, Release, Project, Build, Target
|
from orm.models import BitbakeVersion, Release, Project, Build, Target
|
||||||
|
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
|
|
||||||
class TestAllBuildsPage(SeleniumTestCase):
|
class TestAllBuildsPage(SeleniumTestCase):
|
||||||
""" Tests for all builds page /builds/ """
|
""" Tests for all builds page /builds/ """
|
||||||
@@ -91,7 +93,7 @@ class TestAllBuildsPage(SeleniumTestCase):
|
|||||||
found_row = None
|
found_row = None
|
||||||
for row in rows:
|
for row in rows:
|
||||||
|
|
||||||
outcome_links = row.find_elements_by_css_selector(selector)
|
outcome_links = row.find_elements(By.CSS_SELECTOR, selector)
|
||||||
if len(outcome_links) == 1:
|
if len(outcome_links) == 1:
|
||||||
found_row = row
|
found_row = row
|
||||||
break
|
break
|
||||||
@@ -131,17 +133,19 @@ class TestAllBuildsPage(SeleniumTestCase):
|
|||||||
url = reverse('all-builds')
|
url = reverse('all-builds')
|
||||||
self.get(url)
|
self.get(url)
|
||||||
|
|
||||||
|
# should see a rebuild button for non-command-line builds
|
||||||
|
selector = 'div[data-latest-build-result="%s"] .rebuild-btn' % build1.id
|
||||||
|
time.sleep(2)
|
||||||
|
run_again_button = self.find_all(selector)
|
||||||
|
self.assertEqual(len(run_again_button), 1,
|
||||||
|
'should see a rebuild button for non-cli builds')
|
||||||
|
|
||||||
# shouldn't see a rebuild button for command-line builds
|
# shouldn't see a rebuild button for command-line builds
|
||||||
selector = 'div[data-latest-build-result="%s"] .rebuild-btn' % default_build.id
|
selector = 'div[data-latest-build-result="%s"] .rebuild-btn' % default_build.id
|
||||||
run_again_button = self.find_all(selector)
|
run_again_button = self.find_all(selector)
|
||||||
self.assertEqual(len(run_again_button), 0,
|
self.assertEqual(len(run_again_button), 0,
|
||||||
'should not see a rebuild button for cli builds')
|
'should not see a rebuild button for cli builds')
|
||||||
|
|
||||||
# should see a rebuild button for non-command-line builds
|
|
||||||
selector = 'div[data-latest-build-result="%s"] .rebuild-btn' % build1.id
|
|
||||||
run_again_button = self.find_all(selector)
|
|
||||||
self.assertEqual(len(run_again_button), 1,
|
|
||||||
'should see a rebuild button for non-cli builds')
|
|
||||||
|
|
||||||
def test_tooltips_on_project_name(self):
|
def test_tooltips_on_project_name(self):
|
||||||
"""
|
"""
|
||||||
@@ -198,24 +202,24 @@ class TestAllBuildsPage(SeleniumTestCase):
|
|||||||
|
|
||||||
# test recent builds area for successful build
|
# test recent builds area for successful build
|
||||||
element = self._get_build_time_element(build1)
|
element = self._get_build_time_element(build1)
|
||||||
links = element.find_elements_by_css_selector('a')
|
links = element.find_elements(By.CSS_SELECTOR, 'a')
|
||||||
msg = 'should be a link on the build time for a successful recent build'
|
msg = 'should be a link on the build time for a successful recent build'
|
||||||
self.assertEquals(len(links), 1, msg)
|
self.assertEquals(len(links), 1, msg)
|
||||||
|
|
||||||
# test recent builds area for failed build
|
# test recent builds area for failed build
|
||||||
element = self._get_build_time_element(build2)
|
element = self._get_build_time_element(build2)
|
||||||
links = element.find_elements_by_css_selector('a')
|
links = element.find_elements(By.CSS_SELECTOR, 'a')
|
||||||
msg = 'should not be a link on the build time for a failed recent build'
|
msg = 'should not be a link on the build time for a failed recent build'
|
||||||
self.assertEquals(len(links), 0, msg)
|
self.assertEquals(len(links), 0, msg)
|
||||||
|
|
||||||
# test the time column for successful build
|
# test the time column for successful build
|
||||||
build1_row = self._get_row_for_build(build1)
|
build1_row = self._get_row_for_build(build1)
|
||||||
links = build1_row.find_elements_by_css_selector('td.time a')
|
links = build1_row.find_elements(By.CSS_SELECTOR, 'td.time a')
|
||||||
msg = 'should be a link on the build time for a successful build'
|
msg = 'should be a link on the build time for a successful build'
|
||||||
self.assertEquals(len(links), 1, msg)
|
self.assertEquals(len(links), 1, msg)
|
||||||
|
|
||||||
# test the time column for failed build
|
# test the time column for failed build
|
||||||
build2_row = self._get_row_for_build(build2)
|
build2_row = self._get_row_for_build(build2)
|
||||||
links = build2_row.find_elements_by_css_selector('td.time a')
|
links = build2_row.find_elements(By.CSS_SELECTOR, 'td.time a')
|
||||||
msg = 'should not be a link on the build time for a failed build'
|
msg = 'should not be a link on the build time for a failed build'
|
||||||
self.assertEquals(len(links), 0, msg)
|
self.assertEquals(len(links), 0, msg)
|
||||||
|
|||||||
@@ -16,6 +16,8 @@ from tests.browser.selenium_helpers import SeleniumTestCase
|
|||||||
from orm.models import BitbakeVersion, Release, Project, Build
|
from orm.models import BitbakeVersion, Release, Project, Build
|
||||||
from orm.models import ProjectVariable
|
from orm.models import ProjectVariable
|
||||||
|
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
class TestAllProjectsPage(SeleniumTestCase):
|
class TestAllProjectsPage(SeleniumTestCase):
|
||||||
""" Browser tests for projects page /projects/ """
|
""" Browser tests for projects page /projects/ """
|
||||||
|
|
||||||
@@ -117,7 +119,7 @@ class TestAllProjectsPage(SeleniumTestCase):
|
|||||||
|
|
||||||
# check the release text for the default project
|
# check the release text for the default project
|
||||||
selector = 'span[data-project-field="release"] span.text-muted'
|
selector = 'span[data-project-field="release"] span.text-muted'
|
||||||
element = default_project_row.find_element_by_css_selector(selector)
|
element = default_project_row.find_element(By.CSS_SELECTOR, selector)
|
||||||
text = element.text.strip()
|
text = element.text.strip()
|
||||||
self.assertEqual(text, 'Not applicable',
|
self.assertEqual(text, 'Not applicable',
|
||||||
'release should be "not applicable" for default project')
|
'release should be "not applicable" for default project')
|
||||||
@@ -127,7 +129,7 @@ class TestAllProjectsPage(SeleniumTestCase):
|
|||||||
|
|
||||||
# check the link in the release cell for the other project
|
# check the link in the release cell for the other project
|
||||||
selector = 'span[data-project-field="release"]'
|
selector = 'span[data-project-field="release"]'
|
||||||
element = other_project_row.find_element_by_css_selector(selector)
|
element = other_project_row.find_element(By.CSS_SELECTOR, selector)
|
||||||
text = element.text.strip()
|
text = element.text.strip()
|
||||||
self.assertEqual(text, self.release.name,
|
self.assertEqual(text, self.release.name,
|
||||||
'release name should be shown for non-default project')
|
'release name should be shown for non-default project')
|
||||||
@@ -152,7 +154,7 @@ class TestAllProjectsPage(SeleniumTestCase):
|
|||||||
|
|
||||||
# check the machine cell for the default project
|
# check the machine cell for the default project
|
||||||
selector = 'span[data-project-field="machine"] span.text-muted'
|
selector = 'span[data-project-field="machine"] span.text-muted'
|
||||||
element = default_project_row.find_element_by_css_selector(selector)
|
element = default_project_row.find_element(By.CSS_SELECTOR, selector)
|
||||||
text = element.text.strip()
|
text = element.text.strip()
|
||||||
self.assertEqual(text, 'Not applicable',
|
self.assertEqual(text, 'Not applicable',
|
||||||
'machine should be not applicable for default project')
|
'machine should be not applicable for default project')
|
||||||
@@ -162,7 +164,7 @@ class TestAllProjectsPage(SeleniumTestCase):
|
|||||||
|
|
||||||
# check the link in the machine cell for the other project
|
# check the link in the machine cell for the other project
|
||||||
selector = 'span[data-project-field="machine"]'
|
selector = 'span[data-project-field="machine"]'
|
||||||
element = other_project_row.find_element_by_css_selector(selector)
|
element = other_project_row.find_element(By.CSS_SELECTOR, selector)
|
||||||
text = element.text.strip()
|
text = element.text.strip()
|
||||||
self.assertEqual(text, self.MACHINE_NAME,
|
self.assertEqual(text, self.MACHINE_NAME,
|
||||||
'machine name should be shown for non-default project')
|
'machine name should be shown for non-default project')
|
||||||
@@ -187,7 +189,7 @@ class TestAllProjectsPage(SeleniumTestCase):
|
|||||||
|
|
||||||
# check the link on the name field
|
# check the link on the name field
|
||||||
selector = 'span[data-project-field="name"] a'
|
selector = 'span[data-project-field="name"] a'
|
||||||
element = default_project_row.find_element_by_css_selector(selector)
|
element = default_project_row.find_element(By.CSS_SELECTOR, selector)
|
||||||
link_url = element.get_attribute('href').strip()
|
link_url = element.get_attribute('href').strip()
|
||||||
expected_url = reverse('projectbuilds', args=(self.default_project.id,))
|
expected_url = reverse('projectbuilds', args=(self.default_project.id,))
|
||||||
msg = 'link on default project name should point to builds but was %s' % link_url
|
msg = 'link on default project name should point to builds but was %s' % link_url
|
||||||
@@ -198,7 +200,7 @@ class TestAllProjectsPage(SeleniumTestCase):
|
|||||||
|
|
||||||
# check the link for the other project
|
# check the link for the other project
|
||||||
selector = 'span[data-project-field="name"] a'
|
selector = 'span[data-project-field="name"] a'
|
||||||
element = other_project_row.find_element_by_css_selector(selector)
|
element = other_project_row.find_element(By.CSS_SELECTOR, selector)
|
||||||
link_url = element.get_attribute('href').strip()
|
link_url = element.get_attribute('href').strip()
|
||||||
expected_url = reverse('project', args=(self.project.id,))
|
expected_url = reverse('project', args=(self.project.id,))
|
||||||
msg = 'link on project name should point to configuration but was %s' % link_url
|
msg = 'link on project name should point to configuration but was %s' % link_url
|
||||||
|
|||||||
@@ -15,6 +15,8 @@ from tests.browser.selenium_helpers import SeleniumTestCase
|
|||||||
from orm.models import Project, Release, BitbakeVersion, Build, LogMessage
|
from orm.models import Project, Release, BitbakeVersion, Build, LogMessage
|
||||||
from orm.models import Layer, Layer_Version, Recipe, CustomImageRecipe, Variable
|
from orm.models import Layer, Layer_Version, Recipe, CustomImageRecipe, Variable
|
||||||
|
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
class TestBuildDashboardPage(SeleniumTestCase):
|
class TestBuildDashboardPage(SeleniumTestCase):
|
||||||
""" Tests for the build dashboard /build/X """
|
""" Tests for the build dashboard /build/X """
|
||||||
|
|
||||||
@@ -183,7 +185,7 @@ class TestBuildDashboardPage(SeleniumTestCase):
|
|||||||
|
|
||||||
found = False
|
found = False
|
||||||
for element in message_elements:
|
for element in message_elements:
|
||||||
log_message_text = element.find_element_by_tag_name('pre').text.strip()
|
log_message_text = element.find_element(By.TAG_NAME, 'pre').text.strip()
|
||||||
text_matches = (log_message_text == expected_text)
|
text_matches = (log_message_text == expected_text)
|
||||||
|
|
||||||
log_message_pk = element.get_attribute('data-log-message-id')
|
log_message_pk = element.get_attribute('data-log-message-id')
|
||||||
@@ -213,7 +215,7 @@ class TestBuildDashboardPage(SeleniumTestCase):
|
|||||||
the WebElement modal match the list of text values in expected
|
the WebElement modal match the list of text values in expected
|
||||||
"""
|
"""
|
||||||
# labels containing the radio buttons we're testing for
|
# labels containing the radio buttons we're testing for
|
||||||
labels = modal.find_elements_by_css_selector(".radio")
|
labels = modal.find_elements(By.CSS_SELECTOR,".radio")
|
||||||
|
|
||||||
labels_text = [lab.text for lab in labels]
|
labels_text = [lab.text for lab in labels]
|
||||||
self.assertEqual(len(labels_text), len(expected))
|
self.assertEqual(len(labels_text), len(expected))
|
||||||
@@ -248,7 +250,7 @@ class TestBuildDashboardPage(SeleniumTestCase):
|
|||||||
selector = '[data-role="edit-custom-image-trigger"]'
|
selector = '[data-role="edit-custom-image-trigger"]'
|
||||||
self.click(selector)
|
self.click(selector)
|
||||||
|
|
||||||
modal = self.driver.find_element_by_id('edit-custom-image-modal')
|
modal = self.driver.find_element(By.ID, 'edit-custom-image-modal')
|
||||||
self.wait_until_visible("#edit-custom-image-modal")
|
self.wait_until_visible("#edit-custom-image-modal")
|
||||||
|
|
||||||
# recipes we expect to see in the edit custom image modal
|
# recipes we expect to see in the edit custom image modal
|
||||||
@@ -270,7 +272,7 @@ class TestBuildDashboardPage(SeleniumTestCase):
|
|||||||
selector = '[data-role="new-custom-image-trigger"]'
|
selector = '[data-role="new-custom-image-trigger"]'
|
||||||
self.click(selector)
|
self.click(selector)
|
||||||
|
|
||||||
modal = self.driver.find_element_by_id('new-custom-image-modal')
|
modal = self.driver.find_element(By.ID,'new-custom-image-modal')
|
||||||
self.wait_until_visible("#new-custom-image-modal")
|
self.wait_until_visible("#new-custom-image-modal")
|
||||||
|
|
||||||
# recipes we expect to see in the new custom image modal
|
# recipes we expect to see in the new custom image modal
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
#
|
#
|
||||||
# Copyright (C) 2013-2016 Intel Corporation
|
# Copyright (C) 2013-2016 Intel Corporation
|
||||||
#
|
#
|
||||||
|
import time
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from tests.browser.selenium_helpers import SeleniumTestCase
|
from tests.browser.selenium_helpers import SeleniumTestCase
|
||||||
@@ -14,6 +14,8 @@ from tests.browser.selenium_helpers_base import Wait
|
|||||||
from orm.models import Project, Build, Task, Recipe, Layer, Layer_Version
|
from orm.models import Project, Build, Task, Recipe, Layer, Layer_Version
|
||||||
from bldcontrol.models import BuildRequest
|
from bldcontrol.models import BuildRequest
|
||||||
|
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
class TestMostRecentBuildsStates(SeleniumTestCase):
|
class TestMostRecentBuildsStates(SeleniumTestCase):
|
||||||
""" Test states update correctly in most recent builds area """
|
""" Test states update correctly in most recent builds area """
|
||||||
|
|
||||||
@@ -62,7 +64,7 @@ class TestMostRecentBuildsStates(SeleniumTestCase):
|
|||||||
element = self.wait_until_visible(selector)
|
element = self.wait_until_visible(selector)
|
||||||
|
|
||||||
bar_selector = '#recipes-parsed-percentage-bar-%s' % build.id
|
bar_selector = '#recipes-parsed-percentage-bar-%s' % build.id
|
||||||
bar_element = element.find_element_by_css_selector(bar_selector)
|
bar_element = element.find_element(By.CSS_SELECTOR, bar_selector)
|
||||||
self.assertEqual(bar_element.value_of_css_property('width'), '0px',
|
self.assertEqual(bar_element.value_of_css_property('width'), '0px',
|
||||||
'recipe parse progress should be at 0')
|
'recipe parse progress should be at 0')
|
||||||
|
|
||||||
@@ -73,7 +75,7 @@ class TestMostRecentBuildsStates(SeleniumTestCase):
|
|||||||
self.get(url)
|
self.get(url)
|
||||||
|
|
||||||
element = self.wait_until_visible(selector)
|
element = self.wait_until_visible(selector)
|
||||||
bar_element = element.find_element_by_css_selector(bar_selector)
|
bar_element = element.find_element(By.CSS_SELECTOR, bar_selector)
|
||||||
recipe_bar_updated = lambda driver: \
|
recipe_bar_updated = lambda driver: \
|
||||||
bar_element.get_attribute('style') == 'width: 50%;'
|
bar_element.get_attribute('style') == 'width: 50%;'
|
||||||
msg = 'recipe parse progress bar should update to 50%'
|
msg = 'recipe parse progress bar should update to 50%'
|
||||||
@@ -107,7 +109,7 @@ class TestMostRecentBuildsStates(SeleniumTestCase):
|
|||||||
element = self.wait_until_visible(selector)
|
element = self.wait_until_visible(selector)
|
||||||
|
|
||||||
bar_selector = '#build-pc-done-bar-%s' % build.id
|
bar_selector = '#build-pc-done-bar-%s' % build.id
|
||||||
bar_element = element.find_element_by_css_selector(bar_selector)
|
bar_element = element.find_element(By.CSS_SELECTOR, bar_selector)
|
||||||
|
|
||||||
task_bar_updated = lambda driver: \
|
task_bar_updated = lambda driver: \
|
||||||
bar_element.get_attribute('style') == 'width: 50%;'
|
bar_element.get_attribute('style') == 'width: 50%;'
|
||||||
@@ -121,7 +123,7 @@ class TestMostRecentBuildsStates(SeleniumTestCase):
|
|||||||
self.get(url)
|
self.get(url)
|
||||||
|
|
||||||
element = self.wait_until_visible(selector)
|
element = self.wait_until_visible(selector)
|
||||||
bar_element = element.find_element_by_css_selector(bar_selector)
|
bar_element = element.find_element(By.CSS_SELECTOR, bar_selector)
|
||||||
task_bar_updated = lambda driver: \
|
task_bar_updated = lambda driver: \
|
||||||
bar_element.get_attribute('style') == 'width: 100%;'
|
bar_element.get_attribute('style') == 'width: 100%;'
|
||||||
msg = 'tasks progress bar should update to 100%'
|
msg = 'tasks progress bar should update to 100%'
|
||||||
|
|||||||
@@ -6,6 +6,7 @@
|
|||||||
#
|
#
|
||||||
# SPDX-License-Identifier: GPL-2.0-only
|
# SPDX-License-Identifier: GPL-2.0-only
|
||||||
#
|
#
|
||||||
|
from bldcontrol.models import BuildEnvironment
|
||||||
|
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from tests.browser.selenium_helpers import SeleniumTestCase
|
from tests.browser.selenium_helpers import SeleniumTestCase
|
||||||
@@ -18,6 +19,9 @@ class TestNewCustomImagePage(SeleniumTestCase):
|
|||||||
CUSTOM_IMAGE_NAME = 'roopa-doopa'
|
CUSTOM_IMAGE_NAME = 'roopa-doopa'
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
|
BuildEnvironment.objects.get_or_create(
|
||||||
|
betype=BuildEnvironment.TYPE_LOCAL,
|
||||||
|
)
|
||||||
release = Release.objects.create(
|
release = Release.objects.create(
|
||||||
name='baz',
|
name='baz',
|
||||||
bitbake_version=BitbakeVersion.objects.create(name='v1')
|
bitbake_version=BitbakeVersion.objects.create(name='v1')
|
||||||
|
|||||||
@@ -6,11 +6,13 @@
|
|||||||
#
|
#
|
||||||
# SPDX-License-Identifier: GPL-2.0-only
|
# SPDX-License-Identifier: GPL-2.0-only
|
||||||
#
|
#
|
||||||
|
import time
|
||||||
|
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from tests.browser.selenium_helpers import SeleniumTestCase
|
from tests.browser.selenium_helpers import SeleniumTestCase
|
||||||
from selenium.webdriver.support.ui import Select
|
from selenium.webdriver.support.ui import Select
|
||||||
from selenium.common.exceptions import InvalidElementStateException
|
from selenium.common.exceptions import InvalidElementStateException
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
from orm.models import Project, Release, BitbakeVersion
|
from orm.models import Project, Release, BitbakeVersion
|
||||||
|
|
||||||
@@ -47,13 +49,14 @@ class TestNewProjectPage(SeleniumTestCase):
|
|||||||
|
|
||||||
url = reverse('newproject')
|
url = reverse('newproject')
|
||||||
self.get(url)
|
self.get(url)
|
||||||
|
|
||||||
self.enter_text('#new-project-name', project_name)
|
self.enter_text('#new-project-name', project_name)
|
||||||
|
|
||||||
select = Select(self.find('#projectversion'))
|
select = Select(self.find('#projectversion'))
|
||||||
select.select_by_value(str(self.release.pk))
|
select.select_by_value(str(self.release.pk))
|
||||||
|
|
||||||
|
time.sleep(1)
|
||||||
self.click("#create-project-button")
|
self.click("#create-project-button")
|
||||||
|
time.sleep(2)
|
||||||
|
|
||||||
# We should get redirected to the new project's page with the
|
# We should get redirected to the new project's page with the
|
||||||
# notification at the top
|
# notification at the top
|
||||||
@@ -84,6 +87,12 @@ class TestNewProjectPage(SeleniumTestCase):
|
|||||||
select = Select(self.find('#projectversion'))
|
select = Select(self.find('#projectversion'))
|
||||||
select.select_by_value(str(self.release.pk))
|
select.select_by_value(str(self.release.pk))
|
||||||
|
|
||||||
|
radio = self.driver.find_element(By.ID, 'type-new')
|
||||||
|
radio.click()
|
||||||
|
|
||||||
|
self.click("#create-project-button")
|
||||||
|
time.sleep(2)
|
||||||
|
|
||||||
element = self.wait_until_visible('#hint-error-project-name')
|
element = self.wait_until_visible('#hint-error-project-name')
|
||||||
|
|
||||||
self.assertTrue(("Project names must be unique" in element.text),
|
self.assertTrue(("Project names must be unique" in element.text),
|
||||||
@@ -96,6 +105,7 @@ class TestNewProjectPage(SeleniumTestCase):
|
|||||||
except InvalidElementStateException:
|
except InvalidElementStateException:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
time.sleep(2)
|
||||||
self.assertTrue(
|
self.assertTrue(
|
||||||
(Project.objects.filter(name=project_name).count() == 1),
|
(Project.objects.filter(name=project_name).count() == 1),
|
||||||
"New project not found in database")
|
"New project not found in database")
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ from django.urls import reverse
|
|||||||
from tests.browser.selenium_helpers import SeleniumTestCase
|
from tests.browser.selenium_helpers import SeleniumTestCase
|
||||||
|
|
||||||
from orm.models import BitbakeVersion, Release, Project, ProjectVariable
|
from orm.models import BitbakeVersion, Release, Project, ProjectVariable
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
class TestProjectConfigsPage(SeleniumTestCase):
|
class TestProjectConfigsPage(SeleniumTestCase):
|
||||||
""" Test data at /project/X/builds is displayed correctly """
|
""" Test data at /project/X/builds is displayed correctly """
|
||||||
@@ -66,7 +67,7 @@ class TestProjectConfigsPage(SeleniumTestCase):
|
|||||||
|
|
||||||
self.enter_text('#new-imagefs_types', imagefs_type)
|
self.enter_text('#new-imagefs_types', imagefs_type)
|
||||||
|
|
||||||
checkboxes = self.driver.find_elements_by_xpath("//input[@class='fs-checkbox-fstypes']")
|
checkboxes = self.driver.find_elements(By.XPATH, "//input[@class='fs-checkbox-fstypes']")
|
||||||
|
|
||||||
for checkbox in checkboxes:
|
for checkbox in checkboxes:
|
||||||
if checkbox.get_attribute("value") == "btrfs":
|
if checkbox.get_attribute("value") == "btrfs":
|
||||||
@@ -95,7 +96,7 @@ class TestProjectConfigsPage(SeleniumTestCase):
|
|||||||
for checkbox in checkboxes:
|
for checkbox in checkboxes:
|
||||||
if checkbox.get_attribute("value") == "cpio":
|
if checkbox.get_attribute("value") == "cpio":
|
||||||
checkbox.click()
|
checkbox.click()
|
||||||
element = self.driver.find_element_by_id('new-imagefs_types')
|
element = self.driver.find_element(By.ID, 'new-imagefs_types')
|
||||||
|
|
||||||
self.wait_until_visible('#new-imagefs_types')
|
self.wait_until_visible('#new-imagefs_types')
|
||||||
|
|
||||||
@@ -129,7 +130,7 @@ class TestProjectConfigsPage(SeleniumTestCase):
|
|||||||
self.assertTrue((self.INVALID_PATH_START_TEXT in element.text), msg)
|
self.assertTrue((self.INVALID_PATH_START_TEXT in element.text), msg)
|
||||||
|
|
||||||
# downloads dir path has a space
|
# downloads dir path has a space
|
||||||
self.driver.find_element_by_id('new-dl_dir').clear()
|
self.driver.find_element(By.ID, 'new-dl_dir').clear()
|
||||||
self.enter_text('#new-dl_dir', '/foo/bar a')
|
self.enter_text('#new-dl_dir', '/foo/bar a')
|
||||||
|
|
||||||
element = self.wait_until_visible('#hintError-dl_dir')
|
element = self.wait_until_visible('#hintError-dl_dir')
|
||||||
@@ -137,7 +138,7 @@ class TestProjectConfigsPage(SeleniumTestCase):
|
|||||||
self.assertTrue((self.INVALID_PATH_CHAR_TEXT in element.text), msg)
|
self.assertTrue((self.INVALID_PATH_CHAR_TEXT in element.text), msg)
|
||||||
|
|
||||||
# downloads dir path starts with ${...} but has a space
|
# downloads dir path starts with ${...} but has a space
|
||||||
self.driver.find_element_by_id('new-dl_dir').clear()
|
self.driver.find_element(By.ID,'new-dl_dir').clear()
|
||||||
self.enter_text('#new-dl_dir', '${TOPDIR}/down foo')
|
self.enter_text('#new-dl_dir', '${TOPDIR}/down foo')
|
||||||
|
|
||||||
element = self.wait_until_visible('#hintError-dl_dir')
|
element = self.wait_until_visible('#hintError-dl_dir')
|
||||||
@@ -145,18 +146,18 @@ class TestProjectConfigsPage(SeleniumTestCase):
|
|||||||
self.assertTrue((self.INVALID_PATH_CHAR_TEXT in element.text), msg)
|
self.assertTrue((self.INVALID_PATH_CHAR_TEXT in element.text), msg)
|
||||||
|
|
||||||
# downloads dir path starts with /
|
# downloads dir path starts with /
|
||||||
self.driver.find_element_by_id('new-dl_dir').clear()
|
self.driver.find_element(By.ID,'new-dl_dir').clear()
|
||||||
self.enter_text('#new-dl_dir', '/bar/foo')
|
self.enter_text('#new-dl_dir', '/bar/foo')
|
||||||
|
|
||||||
hidden_element = self.driver.find_element_by_id('hintError-dl_dir')
|
hidden_element = self.driver.find_element(By.ID,'hintError-dl_dir')
|
||||||
self.assertEqual(hidden_element.is_displayed(), False,
|
self.assertEqual(hidden_element.is_displayed(), False,
|
||||||
'downloads directory path valid but treated as invalid')
|
'downloads directory path valid but treated as invalid')
|
||||||
|
|
||||||
# downloads dir path starts with ${...}
|
# downloads dir path starts with ${...}
|
||||||
self.driver.find_element_by_id('new-dl_dir').clear()
|
self.driver.find_element(By.ID,'new-dl_dir').clear()
|
||||||
self.enter_text('#new-dl_dir', '${TOPDIR}/down')
|
self.enter_text('#new-dl_dir', '${TOPDIR}/down')
|
||||||
|
|
||||||
hidden_element = self.driver.find_element_by_id('hintError-dl_dir')
|
hidden_element = self.driver.find_element(By.ID,'hintError-dl_dir')
|
||||||
self.assertEqual(hidden_element.is_displayed(), False,
|
self.assertEqual(hidden_element.is_displayed(), False,
|
||||||
'downloads directory path valid but treated as invalid')
|
'downloads directory path valid but treated as invalid')
|
||||||
|
|
||||||
@@ -184,7 +185,7 @@ class TestProjectConfigsPage(SeleniumTestCase):
|
|||||||
self.assertTrue((self.INVALID_PATH_START_TEXT in element.text), msg)
|
self.assertTrue((self.INVALID_PATH_START_TEXT in element.text), msg)
|
||||||
|
|
||||||
# path has a space
|
# path has a space
|
||||||
self.driver.find_element_by_id('new-sstate_dir').clear()
|
self.driver.find_element(By.ID, 'new-sstate_dir').clear()
|
||||||
self.enter_text('#new-sstate_dir', '/foo/bar a')
|
self.enter_text('#new-sstate_dir', '/foo/bar a')
|
||||||
|
|
||||||
element = self.wait_until_visible('#hintError-sstate_dir')
|
element = self.wait_until_visible('#hintError-sstate_dir')
|
||||||
@@ -192,7 +193,7 @@ class TestProjectConfigsPage(SeleniumTestCase):
|
|||||||
self.assertTrue((self.INVALID_PATH_CHAR_TEXT in element.text), msg)
|
self.assertTrue((self.INVALID_PATH_CHAR_TEXT in element.text), msg)
|
||||||
|
|
||||||
# path starts with ${...} but has a space
|
# path starts with ${...} but has a space
|
||||||
self.driver.find_element_by_id('new-sstate_dir').clear()
|
self.driver.find_element(By.ID,'new-sstate_dir').clear()
|
||||||
self.enter_text('#new-sstate_dir', '${TOPDIR}/down foo')
|
self.enter_text('#new-sstate_dir', '${TOPDIR}/down foo')
|
||||||
|
|
||||||
element = self.wait_until_visible('#hintError-sstate_dir')
|
element = self.wait_until_visible('#hintError-sstate_dir')
|
||||||
@@ -200,18 +201,18 @@ class TestProjectConfigsPage(SeleniumTestCase):
|
|||||||
self.assertTrue((self.INVALID_PATH_CHAR_TEXT in element.text), msg)
|
self.assertTrue((self.INVALID_PATH_CHAR_TEXT in element.text), msg)
|
||||||
|
|
||||||
# path starts with /
|
# path starts with /
|
||||||
self.driver.find_element_by_id('new-sstate_dir').clear()
|
self.driver.find_element(By.ID,'new-sstate_dir').clear()
|
||||||
self.enter_text('#new-sstate_dir', '/bar/foo')
|
self.enter_text('#new-sstate_dir', '/bar/foo')
|
||||||
|
|
||||||
hidden_element = self.driver.find_element_by_id('hintError-sstate_dir')
|
hidden_element = self.driver.find_element(By.ID, 'hintError-sstate_dir')
|
||||||
self.assertEqual(hidden_element.is_displayed(), False,
|
self.assertEqual(hidden_element.is_displayed(), False,
|
||||||
'sstate directory path valid but treated as invalid')
|
'sstate directory path valid but treated as invalid')
|
||||||
|
|
||||||
# paths starts with ${...}
|
# paths starts with ${...}
|
||||||
self.driver.find_element_by_id('new-sstate_dir').clear()
|
self.driver.find_element(By.ID, 'new-sstate_dir').clear()
|
||||||
self.enter_text('#new-sstate_dir', '${TOPDIR}/down')
|
self.enter_text('#new-sstate_dir', '${TOPDIR}/down')
|
||||||
|
|
||||||
hidden_element = self.driver.find_element_by_id('hintError-sstate_dir')
|
hidden_element = self.driver.find_element(By.ID, 'hintError-sstate_dir')
|
||||||
self.assertEqual(hidden_element.is_displayed(), False,
|
self.assertEqual(hidden_element.is_displayed(), False,
|
||||||
'sstate directory path valid but treated as invalid')
|
'sstate directory path valid but treated as invalid')
|
||||||
|
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ from django.urls import reverse
|
|||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from tests.browser.selenium_helpers import SeleniumTestCase
|
from tests.browser.selenium_helpers import SeleniumTestCase
|
||||||
from orm.models import BitbakeVersion, Release, Project, Build
|
from orm.models import BitbakeVersion, Release, Project, Build
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
class TestToasterTableUI(SeleniumTestCase):
|
class TestToasterTableUI(SeleniumTestCase):
|
||||||
"""
|
"""
|
||||||
@@ -33,7 +34,7 @@ class TestToasterTableUI(SeleniumTestCase):
|
|||||||
table: WebElement for a ToasterTable
|
table: WebElement for a ToasterTable
|
||||||
"""
|
"""
|
||||||
selector = 'thead a.sorted'
|
selector = 'thead a.sorted'
|
||||||
heading = table.find_element_by_css_selector(selector)
|
heading = table.find_element(By.CSS_SELECTOR, selector)
|
||||||
return heading.get_attribute('innerHTML').strip()
|
return heading.get_attribute('innerHTML').strip()
|
||||||
|
|
||||||
def _get_datetime_from_cell(self, row, selector):
|
def _get_datetime_from_cell(self, row, selector):
|
||||||
@@ -45,7 +46,7 @@ class TestToasterTableUI(SeleniumTestCase):
|
|||||||
selector: CSS selector to use to find the cell containing the date time
|
selector: CSS selector to use to find the cell containing the date time
|
||||||
string
|
string
|
||||||
"""
|
"""
|
||||||
cell = row.find_element_by_css_selector(selector)
|
cell = row.find_element(By.CSS_SELECTOR, selector)
|
||||||
cell_text = cell.get_attribute('innerHTML').strip()
|
cell_text = cell.get_attribute('innerHTML').strip()
|
||||||
return datetime.strptime(cell_text, '%d/%m/%y %H:%M')
|
return datetime.strptime(cell_text, '%d/%m/%y %H:%M')
|
||||||
|
|
||||||
@@ -105,7 +106,7 @@ class TestToasterTableUI(SeleniumTestCase):
|
|||||||
self.click('#checkbox-started_on')
|
self.click('#checkbox-started_on')
|
||||||
|
|
||||||
# sort by started_on column
|
# sort by started_on column
|
||||||
links = table.find_elements_by_css_selector('th.started_on a')
|
links = table.find_elements(By.CSS_SELECTOR, 'th.started_on a')
|
||||||
for link in links:
|
for link in links:
|
||||||
if link.get_attribute('innerHTML').strip() == 'Started on':
|
if link.get_attribute('innerHTML').strip() == 'Started on':
|
||||||
link.click()
|
link.click()
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ class BuildCoreImageMinimal(BuildTest):
|
|||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
self.completed_build = self.build("core-image-minimal")
|
self.completed_build = self.build("core-image-minimal")
|
||||||
|
self.built = self.target_already_built("core-image-minimal")
|
||||||
|
|
||||||
# Check if build name is unique - tc_id=795
|
# Check if build name is unique - tc_id=795
|
||||||
def test_Build_Unique_Name(self):
|
def test_Build_Unique_Name(self):
|
||||||
|
|||||||
@@ -16,6 +16,9 @@ import re
|
|||||||
|
|
||||||
from tests.browser.selenium_helpers_base import SeleniumTestCaseBase
|
from tests.browser.selenium_helpers_base import SeleniumTestCaseBase
|
||||||
from tests.builds.buildtest import load_build_environment
|
from tests.builds.buildtest import load_build_environment
|
||||||
|
from bldcontrol.models import BuildEnvironment
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
from selenium.common.exceptions import NoSuchElementException
|
||||||
|
|
||||||
logger = logging.getLogger("toaster")
|
logger = logging.getLogger("toaster")
|
||||||
|
|
||||||
@@ -30,6 +33,8 @@ class SeleniumFunctionalTestCase(SeleniumTestCaseBase):
|
|||||||
raise RuntimeError("Please initialise django with the tests settings: " \
|
raise RuntimeError("Please initialise django with the tests settings: " \
|
||||||
"DJANGO_SETTINGS_MODULE='toastermain.settings_test'")
|
"DJANGO_SETTINGS_MODULE='toastermain.settings_test'")
|
||||||
|
|
||||||
|
if BuildEnvironment.objects.count() == 0:
|
||||||
|
BuildEnvironment.objects.create(betype=BuildEnvironment.TYPE_LOCAL)
|
||||||
load_build_environment()
|
load_build_environment()
|
||||||
|
|
||||||
# start toaster
|
# start toaster
|
||||||
@@ -74,8 +79,8 @@ class SeleniumFunctionalTestCase(SeleniumTestCaseBase):
|
|||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
table_element = self.get_table_element(table_id)
|
table_element = self.get_table_element(table_id)
|
||||||
element = table_element.find_element_by_link_text(link_text)
|
element = table_element.find_element(By.LINK_TEXT, link_text)
|
||||||
except self.NoSuchElementException:
|
except NoSuchElementException:
|
||||||
print('no element found')
|
print('no element found')
|
||||||
raise
|
raise
|
||||||
return element
|
return element
|
||||||
@@ -85,8 +90,8 @@ class SeleniumFunctionalTestCase(SeleniumTestCaseBase):
|
|||||||
#return whole-table element
|
#return whole-table element
|
||||||
element_xpath = "//*[@id='" + table_id + "']"
|
element_xpath = "//*[@id='" + table_id + "']"
|
||||||
try:
|
try:
|
||||||
element = self.driver.find_element_by_xpath(element_xpath)
|
element = self.driver.find_element(By.XPATH, element_xpath)
|
||||||
except self.NoSuchElementException:
|
except NoSuchElementException:
|
||||||
raise
|
raise
|
||||||
return element
|
return element
|
||||||
row = coordinate[0]
|
row = coordinate[0]
|
||||||
@@ -95,8 +100,8 @@ class SeleniumFunctionalTestCase(SeleniumTestCaseBase):
|
|||||||
#return whole-row element
|
#return whole-row element
|
||||||
element_xpath = "//*[@id='" + table_id + "']/tbody/tr[" + str(row) + "]"
|
element_xpath = "//*[@id='" + table_id + "']/tbody/tr[" + str(row) + "]"
|
||||||
try:
|
try:
|
||||||
element = self.driver.find_element_by_xpath(element_xpath)
|
element = self.driver.find_element(By.XPATH, element_xpath)
|
||||||
except self.NoSuchElementException:
|
except NoSuchElementException:
|
||||||
return False
|
return False
|
||||||
return element
|
return element
|
||||||
#now we are looking for an element with specified X and Y
|
#now we are looking for an element with specified X and Y
|
||||||
@@ -104,7 +109,7 @@ class SeleniumFunctionalTestCase(SeleniumTestCaseBase):
|
|||||||
|
|
||||||
element_xpath = "//*[@id='" + table_id + "']/tbody/tr[" + str(row) + "]/td[" + str(column) + "]"
|
element_xpath = "//*[@id='" + table_id + "']/tbody/tr[" + str(row) + "]/td[" + str(column) + "]"
|
||||||
try:
|
try:
|
||||||
element = self.driver.find_element_by_xpath(element_xpath)
|
element = self.driver.find_element(By.XPATH, element_xpath)
|
||||||
except self.NoSuchElementException:
|
except NoSuchElementException:
|
||||||
return False
|
return False
|
||||||
return element
|
return element
|
||||||
|
|||||||
@@ -10,6 +10,7 @@
|
|||||||
import re
|
import re
|
||||||
from tests.functional.functional_helpers import SeleniumFunctionalTestCase
|
from tests.functional.functional_helpers import SeleniumFunctionalTestCase
|
||||||
from orm.models import Project
|
from orm.models import Project
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
|
|
||||||
class FuntionalTestBasic(SeleniumFunctionalTestCase):
|
class FuntionalTestBasic(SeleniumFunctionalTestCase):
|
||||||
|
|
||||||
@@ -17,10 +18,10 @@ class FuntionalTestBasic(SeleniumFunctionalTestCase):
|
|||||||
def test_create_slenium_project(self):
|
def test_create_slenium_project(self):
|
||||||
project_name = 'selenium-project'
|
project_name = 'selenium-project'
|
||||||
self.get('')
|
self.get('')
|
||||||
self.driver.find_element_by_link_text("To start building, create your first Toaster project").click()
|
self.driver.find_element(By.LINK_TEXT, "To start building, create your first Toaster project").click()
|
||||||
self.driver.find_element_by_id("new-project-name").send_keys(project_name)
|
self.driver.find_element(By.ID, "new-project-name").send_keys(project_name)
|
||||||
self.driver.find_element_by_id('projectversion').click()
|
self.driver.find_element(By.ID, 'projectversion').click()
|
||||||
self.driver.find_element_by_id("create-project-button").click()
|
self.driver.find_element(By.ID, "create-project-button").click()
|
||||||
element = self.wait_until_visible('#project-created-notification')
|
element = self.wait_until_visible('#project-created-notification')
|
||||||
self.assertTrue(self.element_exists('#project-created-notification'),'Project creation notification not shown')
|
self.assertTrue(self.element_exists('#project-created-notification'),'Project creation notification not shown')
|
||||||
self.assertTrue(project_name in element.text,
|
self.assertTrue(project_name in element.text,
|
||||||
@@ -35,77 +36,77 @@ class FuntionalTestBasic(SeleniumFunctionalTestCase):
|
|||||||
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
||||||
self.assertTrue(self.element_exists('#config-nav'),'Configuration Tab does not exist')
|
self.assertTrue(self.element_exists('#config-nav'),'Configuration Tab does not exist')
|
||||||
project_URL=self.get_URL()
|
project_URL=self.get_URL()
|
||||||
self.driver.find_element_by_xpath('//a[@href="'+project_URL+'"]').click()
|
self.driver.find_element(By.XPATH, '//a[@href="'+project_URL+'"]').click()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'customimages/"'+"]").click()
|
self.driver.find_element(By.XPATH, "//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'customimages/"'+"]").click()
|
||||||
self.assertTrue(re.search("Custom images",self.driver.find_element_by_xpath("//div[@class='col-md-10']").text),'Custom images information is not loading properly')
|
self.assertTrue(re.search("Custom images",self.driver.find_element(By.XPATH, "//div[@class='col-md-10']").text),'Custom images information is not loading properly')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Custom images tab available')
|
self.fail(msg='No Custom images tab available')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'images/"'+"]").click()
|
self.driver.find_element(By.XPATH, "//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'images/"'+"]").click()
|
||||||
self.assertTrue(re.search("Compatible image recipes",self.driver.find_element_by_xpath("//div[@class='col-md-10']").text),'The Compatible image recipes information is not loading properly')
|
self.assertTrue(re.search("Compatible image recipes",self.driver.find_element(By.XPATH, "//div[@class='col-md-10']").text),'The Compatible image recipes information is not loading properly')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Compatible image tab available')
|
self.fail(msg='No Compatible image tab available')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'softwarerecipes/"'+"]").click()
|
self.driver.find_element(By.XPATH, "//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'softwarerecipes/"'+"]").click()
|
||||||
self.assertTrue(re.search("Compatible software recipes",self.driver.find_element_by_xpath("//div[@class='col-md-10']").text),'The Compatible software recipe information is not loading properly')
|
self.assertTrue(re.search("Compatible software recipes",self.driver.find_element(By.XPATH, "//div[@class='col-md-10']").text),'The Compatible software recipe information is not loading properly')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Compatible software recipe tab available')
|
self.fail(msg='No Compatible software recipe tab available')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'machines/"'+"]").click()
|
self.driver.find_element(By.XPATH, "//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'machines/"'+"]").click()
|
||||||
self.assertTrue(re.search("Compatible machines",self.driver.find_element_by_xpath("//div[@class='col-md-10']").text),'The Compatible machine information is not loading properly')
|
self.assertTrue(re.search("Compatible machines",self.driver.find_element(By.XPATH, "//div[@class='col-md-10']").text),'The Compatible machine information is not loading properly')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Compatible machines tab available')
|
self.fail(msg='No Compatible machines tab available')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'layers/"'+"]").click()
|
self.driver.find_element(By.XPATH, "//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'layers/"'+"]").click()
|
||||||
self.assertTrue(re.search("Compatible layers",self.driver.find_element_by_xpath("//div[@class='col-md-10']").text),'The Compatible layer information is not loading properly')
|
self.assertTrue(re.search("Compatible layers",self.driver.find_element(By.XPATH, "//div[@class='col-md-10']").text),'The Compatible layer information is not loading properly')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Compatible layers tab available')
|
self.fail(msg='No Compatible layers tab available')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'configuration"'+"]").click()
|
self.driver.find_element(By.XPATH, "//*[@id='config-nav']/ul/li/a[@href="+'"'+project_URL+'configuration"'+"]").click()
|
||||||
self.assertTrue(re.search("Bitbake variables",self.driver.find_element_by_xpath("//div[@class='col-md-10']").text),'The Bitbake variables information is not loading properly')
|
self.assertTrue(re.search("Bitbake variables",self.driver.find_element(By.XPATH, "//div[@class='col-md-10']").text),'The Bitbake variables information is not loading properly')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Bitbake variables tab available')
|
self.fail(msg='No Bitbake variables tab available')
|
||||||
|
|
||||||
# testcase (1516)
|
# testcase (1516)
|
||||||
def test_review_configuration_information(self):
|
def test_review_configuration_information(self):
|
||||||
self.get('')
|
self.get('')
|
||||||
self.driver.find_element_by_xpath("//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
||||||
self.wait_until_visible('#projectstable')
|
self.wait_until_visible('#projectstable')
|
||||||
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
||||||
project_URL=self.get_URL()
|
project_URL=self.get_URL()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.assertTrue(self.element_exists('#machine-section'),'Machine section for the project configuration page does not exist')
|
self.assertTrue(self.element_exists('#machine-section'),'Machine section for the project configuration page does not exist')
|
||||||
self.assertTrue(re.search("qemux86",self.driver.find_element_by_xpath("//span[@id='project-machine-name']").text),'The machine type is not assigned')
|
self.assertTrue(re.search("qemux86",self.driver.find_element(By.XPATH, "//span[@id='project-machine-name']").text),'The machine type is not assigned')
|
||||||
self.driver.find_element_by_xpath("//span[@id='change-machine-toggle']").click()
|
self.driver.find_element(By.XPATH, "//span[@id='change-machine-toggle']").click()
|
||||||
self.wait_until_visible('#select-machine-form')
|
self.wait_until_visible('#select-machine-form')
|
||||||
self.wait_until_visible('#cancel-machine-change')
|
self.wait_until_visible('#cancel-machine-change')
|
||||||
self.driver.find_element_by_xpath("//form[@id='select-machine-form']/a[@id='cancel-machine-change']").click()
|
self.driver.find_element(By.XPATH, "//form[@id='select-machine-form']/a[@id='cancel-machine-change']").click()
|
||||||
except:
|
except:
|
||||||
self.fail(msg='The machine information is wrong in the configuration page')
|
self.fail(msg='The machine information is wrong in the configuration page')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_id('no-most-built')
|
self.driver.find_element(By.ID, 'no-most-built')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Most built information in project detail page')
|
self.fail(msg='No Most built information in project detail page')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.assertTrue(re.search("Yocto Project master",self.driver.find_element_by_xpath("//span[@id='project-release-title']").text),'The project release is not defined')
|
self.assertTrue(re.search("Yocto Project master",self.driver.find_element(By.XPATH, "//span[@id='project-release-title']").text),'The project release is not defined')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No project release title information in project detail page')
|
self.fail(msg='No project release title information in project detail page')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//div[@id='layer-container']")
|
self.driver.find_element(By.XPATH, "//div[@id='layer-container']")
|
||||||
self.assertTrue(re.search("3",self.driver.find_element_by_id("project-layers-count").text),'There should be 3 layers listed in the layer count')
|
self.assertTrue(re.search("3",self.driver.find_element(By.ID, "project-layers-count").text),'There should be 3 layers listed in the layer count')
|
||||||
layer_list = self.driver.find_element_by_id("layers-in-project-list")
|
layer_list = self.driver.find_element(By.ID, "layers-in-project-list")
|
||||||
layers = layer_list.find_elements_by_tag_name("li")
|
layers = layer_list.find_elements(By.TAG_NAME, "li")
|
||||||
for layer in layers:
|
for layer in layers:
|
||||||
if re.match ("openembedded-core",layer.text):
|
if re.match ("openembedded-core",layer.text):
|
||||||
print ("openembedded-core layer is a default layer in the project configuration")
|
print ("openembedded-core layer is a default layer in the project configuration")
|
||||||
@@ -121,60 +122,61 @@ class FuntionalTestBasic(SeleniumFunctionalTestCase):
|
|||||||
# testcase (1517)
|
# testcase (1517)
|
||||||
def test_verify_machine_information(self):
|
def test_verify_machine_information(self):
|
||||||
self.get('')
|
self.get('')
|
||||||
self.driver.find_element_by_xpath("//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
||||||
self.wait_until_visible('#projectstable')
|
self.wait_until_visible('#projectstable')
|
||||||
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.assertTrue(self.element_exists('#machine-section'),'Machine section for the project configuration page does not exist')
|
self.assertTrue(self.element_exists('#machine-section'),'Machine section for the project configuration page does not exist')
|
||||||
self.assertTrue(re.search("qemux86",self.driver.find_element_by_id("project-machine-name").text),'The machine type is not assigned')
|
self.assertTrue(re.search("qemux86",self.driver.find_element(By.ID, "project-machine-name").text),'The machine type is not assigned')
|
||||||
self.driver.find_element_by_id("change-machine-toggle").click()
|
self.driver.find_element(By.ID, "change-machine-toggle").click()
|
||||||
self.wait_until_visible('#select-machine-form')
|
self.wait_until_visible('#select-machine-form')
|
||||||
self.wait_until_visible('#cancel-machine-change')
|
self.wait_until_visible('#cancel-machine-change')
|
||||||
self.driver.find_element_by_id("cancel-machine-change").click()
|
self.driver.find_element(By.ID, "cancel-machine-change").click()
|
||||||
except:
|
except:
|
||||||
self.fail(msg='The machine information is wrong in the configuration page')
|
self.fail(msg='The machine information is wrong in the configuration page')
|
||||||
|
|
||||||
# testcase (1518)
|
# testcase (1518)
|
||||||
def test_verify_most_built_recipes_information(self):
|
def test_verify_most_built_recipes_information(self):
|
||||||
self.get('')
|
self.get('')
|
||||||
self.driver.find_element_by_xpath("//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
||||||
|
|
||||||
self.wait_until_visible('#projectstable')
|
self.wait_until_visible('#projectstable')
|
||||||
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
||||||
project_URL=self.get_URL()
|
project_URL=self.get_URL()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.assertTrue(re.search("You haven't built any recipes yet",self.driver.find_element_by_id("no-most-built").text),'Default message of no builds is not present')
|
self.assertTrue(re.search("You haven't built any recipes yet",self.driver.find_element(By.ID, "no-most-built").text),'Default message of no builds is not present')
|
||||||
self.driver.find_element_by_xpath("//div[@id='no-most-built']/p/a[@href="+'"'+project_URL+'images/"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='no-most-built']/p/a[@href="+'"'+project_URL+'images/"'+"]").click()
|
||||||
self.assertTrue(re.search("Compatible image recipes",self.driver.find_element_by_xpath("//div[@class='col-md-10']").text),'The Choose a recipe to build link is not working properly')
|
self.assertTrue(re.search("Compatible image recipes",self.driver.find_element(By.XPATH, "//div[@class='col-md-10']").text),'The Choose a recipe to build link is not working properly')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Most built information in project detail page')
|
self.fail(msg='No Most built information in project detail page')
|
||||||
|
|
||||||
# testcase (1519)
|
# testcase (1519)
|
||||||
def test_verify_project_release_information(self):
|
def test_verify_project_release_information(self):
|
||||||
self.get('')
|
self.get('')
|
||||||
self.driver.find_element_by_xpath("//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
||||||
self.wait_until_visible('#projectstable')
|
self.wait_until_visible('#projectstable')
|
||||||
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.assertTrue(re.search("Yocto Project master",self.driver.find_element_by_id("project-release-title").text),'The project release is not defined')
|
self.assertTrue(re.search("Yocto Project master",self.driver.find_element(By.ID, "project-release-title").text),'The project release is not defined')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No project release title information in project detail page')
|
self.fail(msg='No project release title information in project detail page')
|
||||||
|
|
||||||
# testcase (1520)
|
# testcase (1520)
|
||||||
def test_verify_layer_information(self):
|
def test_verify_layer_information(self):
|
||||||
self.get('')
|
self.get('')
|
||||||
self.driver.find_element_by_xpath("//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
||||||
self.wait_until_visible('#projectstable')
|
self.wait_until_visible('#projectstable')
|
||||||
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
||||||
project_URL=self.get_URL()
|
project_URL=self.get_URL()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//div[@id='layer-container']")
|
self.driver.find_element(By.XPATH, "//div[@id='layer-container']")
|
||||||
self.assertTrue(re.search("3",self.driver.find_element_by_id("project-layers-count").text),'There should be 3 layers listed in the layer count')
|
self.assertTrue(re.search("3",self.driver.find_element(By.ID, "project-layers-count").text),'There should be 3 layers listed in the layer count')
|
||||||
layer_list = self.driver.find_element_by_id("layers-in-project-list")
|
layer_list = self.driver.find_element(By.ID, "layers-in-project-list")
|
||||||
layers = layer_list.find_elements_by_tag_name("li")
|
layers = layer_list.find_element(By.TAG_NAME, "li")
|
||||||
|
|
||||||
for layer in layers:
|
for layer in layers:
|
||||||
if re.match ("openembedded-core",layer.text):
|
if re.match ("openembedded-core",layer.text):
|
||||||
@@ -186,43 +188,43 @@ class FuntionalTestBasic(SeleniumFunctionalTestCase):
|
|||||||
else:
|
else:
|
||||||
self.fail(msg='default layers are missing from the project configuration')
|
self.fail(msg='default layers are missing from the project configuration')
|
||||||
|
|
||||||
self.driver.find_element_by_xpath("//input[@id='layer-add-input']")
|
self.driver.find_element(By.XPATH, "//input[@id='layer-add-input']")
|
||||||
self.driver.find_element_by_xpath("//button[@id='add-layer-btn']")
|
self.driver.find_element(By.XPATH, "//button[@id='add-layer-btn']")
|
||||||
self.driver.find_element_by_xpath("//div[@id='layer-container']/form[@class='form-inline']/p/a[@id='view-compatible-layers']")
|
self.driver.find_element(By.XPATH, "//div[@id='layer-container']/form[@class='form-inline']/p/a[@id='view-compatible-layers']")
|
||||||
self.driver.find_element_by_xpath("//div[@id='layer-container']/form[@class='form-inline']/p/a[@href="+'"'+project_URL+'importlayer"'+"]")
|
self.driver.find_element(By.XPATH, "//div[@id='layer-container']/form[@class='form-inline']/p/a[@href="+'"'+project_URL+'importlayer"'+"]")
|
||||||
except:
|
except:
|
||||||
self.fail(msg='No Layer information in project detail page')
|
self.fail(msg='No Layer information in project detail page')
|
||||||
|
|
||||||
# testcase (1521)
|
# testcase (1521)
|
||||||
def test_verify_project_detail_links(self):
|
def test_verify_project_detail_links(self):
|
||||||
self.get('')
|
self.get('')
|
||||||
self.driver.find_element_by_xpath("//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='global-nav']/ul/li/a[@href="+'"'+'/toastergui/projects/'+'"'+"]").click()
|
||||||
self.wait_until_visible('#projectstable')
|
self.wait_until_visible('#projectstable')
|
||||||
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
self.find_element_by_link_text_in_table('projectstable', 'selenium-project').click()
|
||||||
project_URL=self.get_URL()
|
project_URL=self.get_URL()
|
||||||
|
|
||||||
self.driver.find_element_by_xpath("//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li[@id='topbar-configuration-tab']/a[@href="+'"'+project_URL+'"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li[@id='topbar-configuration-tab']/a[@href="+'"'+project_URL+'"'+"]").click()
|
||||||
self.assertTrue(re.search("Configuration",self.driver.find_element_by_xpath("//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li[@id='topbar-configuration-tab']/a[@href="+'"'+project_URL+'"'+"]").text), 'Configuration tab in project topbar is misspelled')
|
self.assertTrue(re.search("Configuration",self.driver.find_element(By.XPATH, "//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li[@id='topbar-configuration-tab']/a[@href="+'"'+project_URL+'"'+"]").text), 'Configuration tab in project topbar is misspelled')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'builds/"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'builds/"'+"]").click()
|
||||||
self.assertTrue(re.search("Builds",self.driver.find_element_by_xpath("//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'builds/"'+"]").text), 'Builds tab in project topbar is misspelled')
|
self.assertTrue(re.search("Builds",self.driver.find_element(By.XPATH, "//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'builds/"'+"]").text), 'Builds tab in project topbar is misspelled')
|
||||||
self.driver.find_element_by_xpath("//div[@id='empty-state-projectbuildstable']")
|
self.driver.find_element(By.XPATH, "//div[@id='empty-state-projectbuildstable']")
|
||||||
except:
|
except:
|
||||||
self.fail(msg='Builds tab information is not present')
|
self.fail(msg='Builds tab information is not present')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'importlayer"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'importlayer"'+"]").click()
|
||||||
self.assertTrue(re.search("Import layer",self.driver.find_element_by_xpath("//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'importlayer"'+"]").text), 'Import layer tab in project topbar is misspelled')
|
self.assertTrue(re.search("Import layer",self.driver.find_element(By.XPATH, "//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'importlayer"'+"]").text), 'Import layer tab in project topbar is misspelled')
|
||||||
self.driver.find_element_by_xpath("//fieldset[@id='repo-select']")
|
self.driver.find_element(By.XPATH, "//fieldset[@id='repo-select']")
|
||||||
self.driver.find_element_by_xpath("//fieldset[@id='git-repo']")
|
self.driver.find_element(By.XPATH, "//fieldset[@id='git-repo']")
|
||||||
except:
|
except:
|
||||||
self.fail(msg='Import layer tab not loading properly')
|
self.fail(msg='Import layer tab not loading properly')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.driver.find_element_by_xpath("//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'newcustomimage/"'+"]").click()
|
self.driver.find_element(By.XPATH, "//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'newcustomimage/"'+"]").click()
|
||||||
self.assertTrue(re.search("New custom image",self.driver.find_element_by_xpath("//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'newcustomimage/"'+"]").text), 'New custom image tab in project topbar is misspelled')
|
self.assertTrue(re.search("New custom image",self.driver.find_element(By.XPATH, "//div[@id='project-topbar']/ul[@class='nav nav-tabs']/li/a[@href="+'"'+project_URL+'newcustomimage/"'+"]").text), 'New custom image tab in project topbar is misspelled')
|
||||||
self.assertTrue(re.search("Select the image recipe you want to customise",self.driver.find_element_by_xpath("//div[@class='col-md-12']/h2").text),'The new custom image tab is not loading correctly')
|
self.assertTrue(re.search("Select the image recipe you want to customise",self.driver.find_element(By.XPATH, "//div[@class='col-md-12']/h2").text),'The new custom image tab is not loading correctly')
|
||||||
except:
|
except:
|
||||||
self.fail(msg='New custom image tab not loading properly')
|
self.fail(msg='New custom image tab not loading properly')
|
||||||
|
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
selenium==2.49.2
|
selenium>=4.13.0
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ from orm.models import Layer_Version, Recipe
|
|||||||
from orm.models import CustomImageRecipe
|
from orm.models import CustomImageRecipe
|
||||||
from orm.models import CustomImagePackage
|
from orm.models import CustomImagePackage
|
||||||
|
|
||||||
|
from bldcontrol.models import BuildEnvironment
|
||||||
import inspect
|
import inspect
|
||||||
import toastergui
|
import toastergui
|
||||||
|
|
||||||
@@ -45,6 +46,9 @@ class ViewTests(TestCase):
|
|||||||
self.cust_package = CustomImagePackage.objects.first()
|
self.cust_package = CustomImagePackage.objects.first()
|
||||||
self.package = Package.objects.first()
|
self.package = Package.objects.first()
|
||||||
self.lver = Layer_Version.objects.first()
|
self.lver = Layer_Version.objects.first()
|
||||||
|
if BuildEnvironment.objects.count() == 0:
|
||||||
|
BuildEnvironment.objects.create(betype=BuildEnvironment.TYPE_LOCAL)
|
||||||
|
|
||||||
|
|
||||||
def test_get_base_call_returns_html(self):
|
def test_get_base_call_returns_html(self):
|
||||||
"""Basic test for all-projects view"""
|
"""Basic test for all-projects view"""
|
||||||
|
|||||||
@@ -305,6 +305,7 @@ class ToasterTable(TemplateView):
|
|||||||
|
|
||||||
self.setup_columns(**kwargs)
|
self.setup_columns(**kwargs)
|
||||||
|
|
||||||
|
self.apply_orderby('pk')
|
||||||
if search:
|
if search:
|
||||||
self.apply_search(search)
|
self.apply_search(search)
|
||||||
if filters:
|
if filters:
|
||||||
|
|||||||
Reference in New Issue
Block a user