123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463 |
- #! /usr/bin/env python3
- #
- # BitBake Toaster Implementation
- #
- # Copyright (C) 2013-2016 Intel Corporation
- #
- # SPDX-License-Identifier: GPL-2.0-only
- #
- import re
- import time
- from django.urls import reverse
- from selenium.webdriver.support.select import Select
- from django.utils import timezone
- from bldcontrol.models import BuildRequest
- from tests.browser.selenium_helpers import SeleniumTestCase
- from orm.models import BitbakeVersion, Layer, Layer_Version, Recipe, Release, Project, Build, Target, Task
- from selenium.webdriver.common.by import By
- class TestAllBuildsPage(SeleniumTestCase):
- """ Tests for all builds page /builds/ """
- PROJECT_NAME = 'test project'
- CLI_BUILDS_PROJECT_NAME = 'command line builds'
- def setUp(self):
- bbv = BitbakeVersion.objects.create(name='bbv1', giturl='/tmp/',
- branch='master', dirpath='')
- release = Release.objects.create(name='release1',
- bitbake_version=bbv)
- self.project1 = Project.objects.create_project(name=self.PROJECT_NAME,
- release=release)
- self.default_project = Project.objects.create_project(
- name=self.CLI_BUILDS_PROJECT_NAME,
- release=release
- )
- self.default_project.is_default = True
- self.default_project.save()
- # parameters for builds to associate with the projects
- now = timezone.now()
- self.project1_build_success = {
- 'project': self.project1,
- 'started_on': now,
- 'completed_on': now,
- 'outcome': Build.SUCCEEDED
- }
- self.project1_build_failure = {
- 'project': self.project1,
- 'started_on': now,
- 'completed_on': now,
- 'outcome': Build.FAILED
- }
- self.default_project_build_success = {
- 'project': self.default_project,
- 'started_on': now,
- 'completed_on': now,
- 'outcome': Build.SUCCEEDED
- }
- def _get_build_time_element(self, build):
- """
- Return the HTML element containing the build time for a build
- in the recent builds area
- """
- selector = 'div[data-latest-build-result="%s"] ' \
- '[data-role="data-recent-build-buildtime-field"]' % build.id
- # because this loads via Ajax, wait for it to be visible
- self.wait_until_present(selector)
- build_time_spans = self.find_all(selector)
- self.assertEqual(len(build_time_spans), 1)
- return build_time_spans[0]
- def _get_row_for_build(self, build):
- """ Get the table row for the build from the all builds table """
- self.wait_until_present('#allbuildstable')
- rows = self.find_all('#allbuildstable tr')
- # look for the row with a download link on the recipe which matches the
- # build ID
- url = reverse('builddashboard', args=(build.id,))
- selector = 'td.target a[href="%s"]' % url
- found_row = None
- for row in rows:
- outcome_links = row.find_elements(By.CSS_SELECTOR, selector)
- if len(outcome_links) == 1:
- found_row = row
- break
- self.assertNotEqual(found_row, None)
- return found_row
- def _get_create_builds(self, **kwargs):
- """ Create a build and return the build object """
- build1 = Build.objects.create(**self.project1_build_success)
- build2 = Build.objects.create(**self.project1_build_failure)
- # add some targets to these builds so they have recipe links
- # (and so we can find the row in the ToasterTable corresponding to
- # a particular build)
- Target.objects.create(build=build1, target='foo')
- Target.objects.create(build=build2, target='bar')
- if kwargs:
- # Create kwargs.get('success') builds with success status with target
- # and kwargs.get('failure') builds with failure status with target
- for i in range(kwargs.get('success', 0)):
- now = timezone.now()
- self.project1_build_success['started_on'] = now
- self.project1_build_success[
- 'completed_on'] = now - timezone.timedelta(days=i)
- build = Build.objects.create(**self.project1_build_success)
- Target.objects.create(build=build,
- target=f'{i}_success_recipe',
- task=f'{i}_success_task')
- self._set_buildRequest_and_task_on_build(build)
- for i in range(kwargs.get('failure', 0)):
- now = timezone.now()
- self.project1_build_failure['started_on'] = now
- self.project1_build_failure[
- 'completed_on'] = now - timezone.timedelta(days=i)
- build = Build.objects.create(**self.project1_build_failure)
- Target.objects.create(build=build,
- target=f'{i}_fail_recipe',
- task=f'{i}_fail_task')
- self._set_buildRequest_and_task_on_build(build)
- return build1, build2
- def _create_recipe(self):
- """ Add a recipe to the database and return it """
- layer = Layer.objects.create()
- layer_version = Layer_Version.objects.create(layer=layer)
- return Recipe.objects.create(name='recipe_foo', layer_version=layer_version)
- def _set_buildRequest_and_task_on_build(self, build):
- """ Set buildRequest and task on build """
- build.recipes_parsed = 1
- build.save()
- buildRequest = BuildRequest.objects.create(
- build=build,
- project=self.project1,
- state=BuildRequest.REQ_COMPLETED)
- build.build_request = buildRequest
- recipe = self._create_recipe()
- task = Task.objects.create(build=build,
- recipe=recipe,
- task_name='task',
- outcome=Task.OUTCOME_SUCCESS)
- task.save()
- build.save()
- def test_show_tasks_with_suffix(self):
- """ Task should be shown as suffix on build name """
- build = Build.objects.create(**self.project1_build_success)
- target = 'bash'
- task = 'clean'
- Target.objects.create(build=build, target=target, task=task)
- url = reverse('all-builds')
- self.get(url)
- self.wait_until_present('td[class="target"]')
- cell = self.find('td[class="target"]')
- content = cell.get_attribute('innerHTML')
- expected_text = '%s:%s' % (target, task)
- self.assertTrue(re.search(expected_text, content),
- '"target" cell should contain text %s' % expected_text)
- def test_rebuild_buttons(self):
- """
- Test 'Rebuild' buttons in recent builds section
- 'Rebuild' button should not be shown for command-line builds,
- but should be shown for other builds
- """
- build1 = Build.objects.create(**self.project1_build_success)
- default_build = Build.objects.create(
- **self.default_project_build_success)
- url = reverse('all-builds')
- 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
- selector = 'div[data-latest-build-result="%s"] .rebuild-btn' % default_build.id
- run_again_button = self.find_all(selector)
- self.assertEqual(len(run_again_button), 0,
- 'should not see a rebuild button for cli builds')
- def test_tooltips_on_project_name(self):
- """
- Test tooltips shown next to project name in the main table
- A tooltip should be present next to the command line
- builds project name in the all builds page, but not for
- other projects
- """
- Build.objects.create(**self.project1_build_success)
- Build.objects.create(**self.default_project_build_success)
- url = reverse('all-builds')
- self.get(url)
- # get the project name cells from the table
- cells = self.find_all('#allbuildstable td[class="project"]')
- selector = 'span.get-help'
- for cell in cells:
- content = cell.get_attribute('innerHTML')
- help_icons = cell.find_elements_by_css_selector(selector)
- if re.search(self.PROJECT_NAME, content):
- # no help icon next to non-cli project name
- msg = 'should not be a help icon for non-cli builds name'
- self.assertEqual(len(help_icons), 0, msg)
- elif re.search(self.CLI_BUILDS_PROJECT_NAME, content):
- # help icon next to cli project name
- msg = 'should be a help icon for cli builds name'
- self.assertEqual(len(help_icons), 1, msg)
- else:
- msg = 'found unexpected project name cell in all builds table'
- self.fail(msg)
- def test_builds_time_links(self):
- """
- Successful builds should have links on the time column and in the
- recent builds area; failed builds should not have links on the time column,
- or in the recent builds area
- """
- build1, build2 = self._get_create_builds()
- url = reverse('all-builds')
- self.get(url)
- # test recent builds area for successful build
- element = self._get_build_time_element(build1)
- links = element.find_elements(By.CSS_SELECTOR, 'a')
- msg = 'should be a link on the build time for a successful recent build'
- self.assertEquals(len(links), 1, msg)
- # test recent builds area for failed build
- element = self._get_build_time_element(build2)
- links = element.find_elements(By.CSS_SELECTOR, 'a')
- msg = 'should not be a link on the build time for a failed recent build'
- self.assertEquals(len(links), 0, msg)
- # test the time column for successful build
- build1_row = self._get_row_for_build(build1)
- links = build1_row.find_elements(By.CSS_SELECTOR, 'td.time a')
- msg = 'should be a link on the build time for a successful build'
- self.assertEquals(len(links), 1, msg)
- # test the time column for failed build
- build2_row = self._get_row_for_build(build2)
- 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'
- self.assertEquals(len(links), 0, msg)
- def test_builds_table_search_box(self):
- """ Test the search box in the builds table on the all builds page """
- self._get_create_builds()
- url = reverse('all-builds')
- self.get(url)
- # Check search box is present and works
- self.wait_until_present('#allbuildstable tbody tr')
- search_box = self.find('#search-input-allbuildstable')
- self.assertTrue(search_box.is_displayed())
- # Check that we can search for a build by recipe name
- search_box.send_keys('foo')
- search_btn = self.find('#search-submit-allbuildstable')
- search_btn.click()
- self.wait_until_present('#allbuildstable tbody tr')
- rows = self.find_all('#allbuildstable tbody tr')
- self.assertTrue(len(rows) >= 1)
- def test_filtering_on_failure_tasks_column(self):
- """ Test the filtering on failure tasks column in the builds table on the all builds page """
- self._get_create_builds(success=10, failure=10)
- url = reverse('all-builds')
- self.get(url)
- # Check filtering on failure tasks column
- self.wait_until_present('#allbuildstable tbody tr')
- failed_tasks_filter = self.find('#failed_tasks_filter')
- failed_tasks_filter.click()
- # Check popup is visible
- time.sleep(1)
- self.wait_until_present('#filter-modal-allbuildstable')
- self.assertTrue(
- self.find('#filter-modal-allbuildstable').is_displayed())
- # Check that we can filter by failure tasks
- build_without_failure_tasks = self.find(
- '#failed_tasks_filter\\:without_failed_tasks')
- build_without_failure_tasks.click()
- # click on apply button
- self.find('#filter-modal-allbuildstable .btn-primary').click()
- self.wait_until_present('#allbuildstable tbody tr')
- # Check if filter is applied, by checking if failed_tasks_filter has btn-primary class
- self.assertTrue(self.find('#failed_tasks_filter').get_attribute(
- 'class').find('btn-primary') != -1)
- def test_filtering_on_completedOn_column(self):
- """ Test the filtering on completed_on column in the builds table on the all builds page """
- self._get_create_builds(success=10, failure=10)
- url = reverse('all-builds')
- self.get(url)
- # Check filtering on failure tasks column
- self.wait_until_present('#allbuildstable tbody tr')
- completed_on_filter = self.find('#completed_on_filter')
- completed_on_filter.click()
- # Check popup is visible
- time.sleep(1)
- self.wait_until_present('#filter-modal-allbuildstable')
- self.assertTrue(
- self.find('#filter-modal-allbuildstable').is_displayed())
- # Check that we can filter by failure tasks
- build_without_failure_tasks = self.find(
- '#completed_on_filter\\:date_range')
- build_without_failure_tasks.click()
- # click on apply button
- self.find('#filter-modal-allbuildstable .btn-primary').click()
- self.wait_until_present('#allbuildstable tbody tr')
- # Check if filter is applied, by checking if completed_on_filter has btn-primary class
- self.assertTrue(self.find('#completed_on_filter').get_attribute(
- 'class').find('btn-primary') != -1)
- # Filter by date range
- self.find('#completed_on_filter').click()
- self.wait_until_present('#filter-modal-allbuildstable')
- date_ranges = self.driver.find_elements(
- By.XPATH, '//input[@class="form-control hasDatepicker"]')
- today = timezone.now()
- yestersday = today - timezone.timedelta(days=1)
- time.sleep(1)
- date_ranges[0].send_keys(yestersday.strftime('%Y-%m-%d'))
- date_ranges[1].send_keys(today.strftime('%Y-%m-%d'))
- self.find('#filter-modal-allbuildstable .btn-primary').click()
- self.wait_until_present('#allbuildstable tbody tr')
- self.assertTrue(self.find('#completed_on_filter').get_attribute(
- 'class').find('btn-primary') != -1)
- # Check if filter is applied, number of builds displayed should be 6
- time.sleep(1)
- self.assertTrue(len(self.find_all('#allbuildstable tbody tr')) == 6)
- def test_builds_table_editColumn(self):
- """ Test the edit column feature in the builds table on the all builds page """
- self._get_create_builds(success=10, failure=10)
- def test_edit_column(check_box_id):
- # Check that we can hide/show table column
- check_box = self.find(f'#{check_box_id}')
- th_class = str(check_box_id).replace('checkbox-', '')
- if check_box.is_selected():
- # check if column is visible in table
- self.assertTrue(
- self.find(
- f'#allbuildstable thead th.{th_class}'
- ).is_displayed(),
- f"The {th_class} column is checked in EditColumn dropdown, but it's not visible in table"
- )
- check_box.click()
- # check if column is hidden in table
- self.assertFalse(
- self.find(
- f'#allbuildstable thead th.{th_class}'
- ).is_displayed(),
- f"The {th_class} column is unchecked in EditColumn dropdown, but it's visible in table"
- )
- else:
- # check if column is hidden in table
- self.assertFalse(
- self.find(
- f'#allbuildstable thead th.{th_class}'
- ).is_displayed(),
- f"The {th_class} column is unchecked in EditColumn dropdown, but it's visible in table"
- )
- check_box.click()
- # check if column is visible in table
- self.assertTrue(
- self.find(
- f'#allbuildstable thead th.{th_class}'
- ).is_displayed(),
- f"The {th_class} column is checked in EditColumn dropdown, but it's not visible in table"
- )
- url = reverse('all-builds')
- self.get(url)
- self.wait_until_present('#allbuildstable tbody tr')
- # Check edit column
- edit_column = self.find('#edit-columns-button')
- self.assertTrue(edit_column.is_displayed())
- edit_column.click()
- # Check dropdown is visible
- self.wait_until_visible('ul.dropdown-menu.editcol')
- # Check that we can hide the edit column
- test_edit_column('checkbox-errors_no')
- test_edit_column('checkbox-failed_tasks')
- test_edit_column('checkbox-image_files')
- test_edit_column('checkbox-project')
- test_edit_column('checkbox-started_on')
- test_edit_column('checkbox-time')
- test_edit_column('checkbox-warnings_no')
- def test_builds_table_show_rows(self):
- """ Test the show rows feature in the builds table on the all builds page """
- self._get_create_builds(success=100, failure=100)
- def test_show_rows(row_to_show, show_row_link):
- # Check that we can show rows == row_to_show
- show_row_link.select_by_value(str(row_to_show))
- self.wait_until_present('#allbuildstable tbody tr')
- time.sleep(1)
- self.assertTrue(
- len(self.find_all('#allbuildstable tbody tr')) == row_to_show
- )
- url = reverse('all-builds')
- self.get(url)
- self.wait_until_present('#allbuildstable tbody tr')
- show_rows = self.driver.find_elements(
- By.XPATH,
- '//select[@class="form-control pagesize-allbuildstable"]'
- )
- # Check show rows
- for show_row_link in show_rows:
- show_row_link = Select(show_row_link)
- test_show_rows(10, show_row_link)
- test_show_rows(25, show_row_link)
- test_show_rows(50, show_row_link)
- test_show_rows(100, show_row_link)
- test_show_rows(150, show_row_link)
|