test_testsuite_class.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. #!/usr/bin/env python3
  2. # Copyright (c) 2020 Intel Corporation
  3. #
  4. # SPDX-License-Identifier: Apache-2.0
  5. '''
  6. This test file contains testcases for Testsuite class of twister
  7. '''
  8. import sys
  9. import os
  10. import csv
  11. import pytest
  12. from mock import call, patch, MagicMock
  13. ZEPHYR_BASE = os.getenv("ZEPHYR_BASE")
  14. sys.path.insert(0, os.path.join(ZEPHYR_BASE, "scripts/pylib/twister"))
  15. from twisterlib import TestCase, TestSuite, TestInstance, Platform
  16. def test_testsuite_add_testcases(class_testsuite):
  17. """ Testing add_testcase function of Testsuite class in twister """
  18. # Test 1: Check the list of testcases after calling add testcases function is as expected
  19. class_testsuite.SAMPLE_FILENAME = 'test_sample_app.yaml'
  20. class_testsuite.TESTCASE_FILENAME = 'test_data.yaml'
  21. class_testsuite.add_testcases()
  22. tests_rel_dir = 'scripts/tests/twister/test_data/testcases/tests/'
  23. expected_testcases = ['test_b.check_1',
  24. 'test_b.check_2',
  25. 'test_c.check_1',
  26. 'test_c.check_2',
  27. 'test_a.check_1',
  28. 'test_a.check_2',
  29. 'sample_test.app']
  30. testcase_list = []
  31. for key in sorted(class_testsuite.testcases.keys()):
  32. testcase_list.append(os.path.basename(os.path.normpath(key)))
  33. assert sorted(testcase_list) == sorted(expected_testcases)
  34. # Test 2 : Assert Testcase name is expected & all testcases values are testcase class objects
  35. testcase = class_testsuite.testcases.get(tests_rel_dir + 'test_a/test_a.check_1')
  36. assert testcase.name == tests_rel_dir + 'test_a/test_a.check_1'
  37. assert all(isinstance(n, TestCase) for n in class_testsuite.testcases.values())
  38. @pytest.mark.parametrize("board_root_dir", [("board_config_file_not_exist"), ("board_config")])
  39. def test_add_configurations(test_data, class_testsuite, board_root_dir):
  40. """ Testing add_configurations function of TestSuite class in Twister
  41. Test : Asserting on default platforms list
  42. """
  43. class_testsuite.board_roots = os.path.abspath(test_data + board_root_dir)
  44. suite = TestSuite(class_testsuite.board_roots, class_testsuite.roots, class_testsuite.outdir)
  45. if board_root_dir == "board_config":
  46. suite.add_configurations()
  47. assert sorted(suite.default_platforms) == sorted(['demo_board_1', 'demo_board_3'])
  48. elif board_root_dir == "board_config_file_not_exist":
  49. suite.add_configurations()
  50. assert sorted(suite.default_platforms) != sorted(['demo_board_1'])
  51. def test_get_all_testcases(class_testsuite, all_testcases_dict):
  52. """ Testing get_all_testcases function of TestSuite class in Twister """
  53. class_testsuite.testcases = all_testcases_dict
  54. expected_tests = ['sample_test.app', 'test_a.check_1.1a', 'test_a.check_1.1c',
  55. 'test_a.check_1.2a', 'test_a.check_1.2b', 'test_a.check_1.Unit_1c', 'test_a.check_1.unit_1a', 'test_a.check_1.unit_1b', 'test_a.check_2.1a', 'test_a.check_2.1c', 'test_a.check_2.2a', 'test_a.check_2.2b', 'test_a.check_2.Unit_1c', 'test_a.check_2.unit_1a', 'test_a.check_2.unit_1b', 'test_b.check_1', 'test_b.check_2', 'test_c.check_1', 'test_c.check_2']
  56. assert len(class_testsuite.get_all_tests()) == 19
  57. assert sorted(class_testsuite.get_all_tests()) == sorted(expected_tests)
  58. def test_get_platforms(class_testsuite, platforms_list):
  59. """ Testing get_platforms function of TestSuite class in Twister """
  60. class_testsuite.platforms = platforms_list
  61. platform = class_testsuite.get_platform("demo_board_1")
  62. assert isinstance(platform, Platform)
  63. assert platform.name == "demo_board_1"
  64. def test_load_from_file(test_data, class_testsuite,
  65. platforms_list, all_testcases_dict, caplog, tmpdir_factory):
  66. """ Testing load_from_file function of TestSuite class in Twister """
  67. # Scenario 1 : Validating the error raised if file to load from doesn't exist
  68. with pytest.raises(SystemExit):
  69. class_testsuite.load_from_file(test_data +"twister_test.csv")
  70. assert "Couldn't find input file with list of tests." in caplog.text
  71. # Scenario 2: Testing if the 'instances' dictionary in Testsuite class contains
  72. # the expected values after execution of load_from_file function
  73. # Note: tmp_dir is the temporary directory created so that the contents
  74. # get deleted after invocation of this testcase.
  75. tmp_dir = tmpdir_factory.mktemp("tmp")
  76. class_testsuite.outdir = tmp_dir
  77. class_testsuite.platforms = platforms_list
  78. class_testsuite.testcases = all_testcases_dict
  79. instance_name_list = []
  80. failed_platform_list = []
  81. with open(os.path.join(test_data, "twister.csv"), "r") as filepath:
  82. for row in csv.DictReader(filepath):
  83. testcase_root = os.path.join(ZEPHYR_BASE,
  84. "scripts/tests/twister/test_data/testcases")
  85. workdir = row['test'].split('/')[-3] + "/" + row['test'].split('/')[-2]
  86. test_name = os.path.basename(os.path.normpath(row['test']))
  87. testcase = TestCase(testcase_root, workdir, test_name)
  88. testcase.build_only = False
  89. instance_name = row["platform"] + "/" + row["test"]
  90. instance_name_list.append(instance_name)
  91. class_testsuite.load_from_file(test_data + "twister.csv")
  92. assert list(class_testsuite.instances.keys()) == instance_name_list
  93. #Scenario 3 : Assert the number of times mock method (get_platform) is called,
  94. # equals to the number of testcases failed
  95. failed_platform_list = [row["platform"]
  96. for row in csv.DictReader(filepath)
  97. if row["status"] == "failed"]
  98. for row in failed_platform_list:
  99. with patch.object(TestSuite, 'get_platform') as mock_method:
  100. class_testsuite.load_from_file(class_testsuite.outdir + "twister.csv",
  101. filter_status=["Skipped", "Passed"])
  102. calls = [call(row)]
  103. mock_method.assert_has_calls(calls, any_order=True)
  104. assert mock_method.call_count == len(failed_platform_list)
  105. # Scenario 4 : Assert add_instances function is called from load_from_file function
  106. class_testsuite.add_instances = MagicMock(side_effect=class_testsuite.add_instances)
  107. class_testsuite.load_from_file(test_data + "twister.csv")
  108. class_testsuite.add_instances.assert_called()
  109. # Scenario 5 : Validate if the Keyerror is raised in case if a header expected is missing
  110. with pytest.raises(SystemExit):
  111. class_testsuite.load_from_file(test_data + "twister_keyerror.csv")
  112. assert "Key error while parsing tests file.('status')" in caplog.text
  113. TESTDATA_PART1 = [
  114. ("toolchain_allow", ['gcc'], None, None, "Not in testcase toolchain allow list"),
  115. ("platform_allow", ['demo_board_1'], None, None, "Not in testcase platform allow list"),
  116. ("toolchain_exclude", ['zephyr'], None, None, "In test case toolchain exclude"),
  117. ("platform_exclude", ['demo_board_2'], None, None, "In test case platform exclude"),
  118. ("arch_exclude", ['x86_demo'], None, None, "In test case arch exclude"),
  119. ("arch_allow", ['arm'], None, None, "Not in test case arch allow list"),
  120. ("skip", True, None, None, "Skip filter"),
  121. ("tags", set(['sensor', 'bluetooth']), "ignore_tags", ['bluetooth'], "Excluded tags per platform (exclude_tags)"),
  122. ("min_flash", "2024", "flash", "1024", "Not enough FLASH"),
  123. ("min_ram", "500", "ram", "256", "Not enough RAM"),
  124. ("None", "None", "env", ['BSIM_OUT_PATH', 'demo_env'], "Environment (BSIM_OUT_PATH, demo_env) not satisfied"),
  125. ("build_on_all", True, None, None, "Platform is excluded on command line."),
  126. (None, None, "supported_toolchains", ['gcc'], "Not supported by the toolchain"),
  127. ]
  128. @pytest.mark.parametrize("tc_attribute, tc_value, plat_attribute, plat_value, expected_discards",
  129. TESTDATA_PART1)
  130. def test_apply_filters_part1(class_testsuite, all_testcases_dict, platforms_list,
  131. tc_attribute, tc_value, plat_attribute, plat_value, expected_discards):
  132. """ Testing apply_filters function of TestSuite class in Twister
  133. Part 1: Response of apply_filters function (discard dictionary) have
  134. appropriate values according to the filters
  135. """
  136. if tc_attribute is None and plat_attribute is None:
  137. discards = class_testsuite.apply_filters()
  138. assert not discards
  139. class_testsuite.platforms = platforms_list
  140. class_testsuite.testcases = all_testcases_dict
  141. for plat in class_testsuite.platforms:
  142. if plat_attribute == "ignore_tags":
  143. plat.ignore_tags = plat_value
  144. if plat_attribute == "flash":
  145. plat.flash = plat_value
  146. if plat_attribute == "ram":
  147. plat.ram = plat_value
  148. if plat_attribute == "env":
  149. plat.env = plat_value
  150. plat.env_satisfied = False
  151. if plat_attribute == "supported_toolchains":
  152. plat.supported_toolchains = plat_value
  153. for _, testcase in class_testsuite.testcases.items():
  154. if tc_attribute == "toolchain_allow":
  155. testcase.toolchain_allow = tc_value
  156. if tc_attribute == "platform_allow":
  157. testcase.platform_allow = tc_value
  158. if tc_attribute == "toolchain_exclude":
  159. testcase.toolchain_exclude = tc_value
  160. if tc_attribute == "platform_exclude":
  161. testcase.platform_exclude = tc_value
  162. if tc_attribute == "arch_exclude":
  163. testcase.arch_exclude = tc_value
  164. if tc_attribute == "arch_allow":
  165. testcase.arch_allow = tc_value
  166. if tc_attribute == "skip":
  167. testcase.skip = tc_value
  168. if tc_attribute == "tags":
  169. testcase.tags = tc_value
  170. if tc_attribute == "min_flash":
  171. testcase.min_flash = tc_value
  172. if tc_attribute == "min_ram":
  173. testcase.min_ram = tc_value
  174. if tc_attribute == "build_on_all":
  175. for _, testcase in class_testsuite.testcases.items():
  176. testcase.build_on_all = tc_value
  177. discards = class_testsuite.apply_filters(exclude_platform=['demo_board_1'])
  178. elif plat_attribute == "supported_toolchains":
  179. discards = class_testsuite.apply_filters(force_toolchain=False,
  180. exclude_platform=['demo_board_1'],
  181. platform=['demo_board_2'])
  182. elif tc_attribute is None and plat_attribute is None:
  183. discards = class_testsuite.apply_filters()
  184. else:
  185. discards = class_testsuite.apply_filters(exclude_platform=['demo_board_1'],
  186. platform=['demo_board_2'])
  187. for x in [expected_discards]:
  188. assert x in discards.values()
  189. TESTDATA_PART2 = [
  190. ("runnable", "True", "Not runnable on device"),
  191. ("exclude_tag", ['test_a'], "Command line testcase exclude filter"),
  192. ("run_individual_tests", ['scripts/tests/twister/test_data/testcases/tests/test_a/test_a.check_1'], "Testcase name filter"),
  193. ("arch", ['arm_test'], "Command line testcase arch filter"),
  194. ("tag", ['test_d'], "Command line testcase tag filter")
  195. ]
  196. @pytest.mark.parametrize("extra_filter, extra_filter_value, expected_discards", TESTDATA_PART2)
  197. def test_apply_filters_part2(class_testsuite, all_testcases_dict,
  198. platforms_list, extra_filter, extra_filter_value, expected_discards):
  199. """ Testing apply_filters function of TestSuite class in Twister
  200. Part 2 : Response of apply_filters function (discard dictionary) have
  201. appropriate values according to the filters
  202. """
  203. class_testsuite.platforms = platforms_list
  204. class_testsuite.testcases = all_testcases_dict
  205. kwargs = {
  206. extra_filter : extra_filter_value,
  207. "exclude_platform" : [
  208. 'demo_board_1'
  209. ],
  210. "platform" : [
  211. 'demo_board_2'
  212. ]
  213. }
  214. discards = class_testsuite.apply_filters(**kwargs)
  215. assert discards
  216. for d in discards.values():
  217. assert d == expected_discards
  218. TESTDATA_PART3 = [
  219. (20, 20, -1, 0),
  220. (-2, -1, 10, 20),
  221. (0, 0, 0, 0)
  222. ]
  223. @pytest.mark.parametrize("tc_min_flash, plat_flash, tc_min_ram, plat_ram",
  224. TESTDATA_PART3)
  225. def test_apply_filters_part3(class_testsuite, all_testcases_dict, platforms_list,
  226. tc_min_flash, plat_flash, tc_min_ram, plat_ram):
  227. """ Testing apply_filters function of TestSuite class in Twister
  228. Part 3 : Testing edge cases for ram and flash values of platforms & testcases
  229. """
  230. class_testsuite.platforms = platforms_list
  231. class_testsuite.testcases = all_testcases_dict
  232. for plat in class_testsuite.platforms:
  233. plat.flash = plat_flash
  234. plat.ram = plat_ram
  235. for _, testcase in class_testsuite.testcases.items():
  236. testcase.min_ram = tc_min_ram
  237. testcase.min_flash = tc_min_flash
  238. discards = class_testsuite.apply_filters(exclude_platform=['demo_board_1'],
  239. platform=['demo_board_2'])
  240. assert not discards
  241. def test_add_instances(test_data, class_testsuite, all_testcases_dict, platforms_list):
  242. """ Testing add_instances() function of TestSuite class in Twister
  243. Test 1: instances dictionary keys have expected values (Platform Name + Testcase Name)
  244. Test 2: Values of 'instances' dictionary in Testsuite class are an
  245. instance of 'TestInstance' class
  246. Test 3: Values of 'instances' dictionary have expected values.
  247. """
  248. class_testsuite.outdir = test_data
  249. class_testsuite.platforms = platforms_list
  250. platform = class_testsuite.get_platform("demo_board_2")
  251. instance_list = []
  252. for _, testcase in all_testcases_dict.items():
  253. instance = TestInstance(testcase, platform, class_testsuite.outdir)
  254. instance_list.append(instance)
  255. class_testsuite.add_instances(instance_list)
  256. assert list(class_testsuite.instances.keys()) == \
  257. [platform.name + '/' + s for s in list(all_testcases_dict.keys())]
  258. assert all(isinstance(n, TestInstance) for n in list(class_testsuite.instances.values()))
  259. assert list(class_testsuite.instances.values()) == instance_list