yocto-check-layer 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. #!/usr/bin/env python3
  2. # Yocto Project layer checking tool
  3. #
  4. # Copyright (C) 2017 Intel Corporation
  5. #
  6. # SPDX-License-Identifier: MIT
  7. #
  8. import os
  9. import sys
  10. import argparse
  11. import logging
  12. import time
  13. import signal
  14. import shutil
  15. import collections
  16. scripts_path = os.path.dirname(os.path.realpath(__file__))
  17. lib_path = scripts_path + '/lib'
  18. sys.path = sys.path + [lib_path]
  19. import scriptutils
  20. import scriptpath
  21. scriptpath.add_oe_lib_path()
  22. scriptpath.add_bitbake_lib_path()
  23. from checklayer import LayerType, detect_layers, add_layers, add_layer_dependencies, get_layer_dependencies, get_signatures, check_bblayers
  24. from oeqa.utils.commands import get_bb_vars
  25. PROGNAME = 'yocto-check-layer'
  26. CASES_PATHS = [os.path.join(os.path.abspath(os.path.dirname(__file__)),
  27. 'lib', 'checklayer', 'cases')]
  28. logger = scriptutils.logger_create(PROGNAME, stream=sys.stdout)
  29. def test_layer(td, layer, test_software_layer_signatures):
  30. from checklayer.context import CheckLayerTestContext
  31. logger.info("Starting to analyze: %s" % layer['name'])
  32. logger.info("----------------------------------------------------------------------")
  33. tc = CheckLayerTestContext(td=td, logger=logger, layer=layer, test_software_layer_signatures=test_software_layer_signatures)
  34. tc.loadTests(CASES_PATHS)
  35. return tc.runTests()
  36. def main():
  37. parser = argparse.ArgumentParser(
  38. description="Yocto Project layer checking tool",
  39. add_help=False)
  40. parser.add_argument('layers', metavar='LAYER_DIR', nargs='+',
  41. help='Layer to check')
  42. parser.add_argument('-o', '--output-log',
  43. help='File to output log (optional)', action='store')
  44. parser.add_argument('--dependency', nargs="+",
  45. help='Layers to process for dependencies', action='store')
  46. parser.add_argument('--no-auto-dependency', help='Disable automatic testing of dependencies',
  47. action='store_true')
  48. parser.add_argument('--machines', nargs="+",
  49. help='List of MACHINEs to be used during testing', action='store')
  50. parser.add_argument('--additional-layers', nargs="+",
  51. help='List of additional layers to add during testing', action='store')
  52. group = parser.add_mutually_exclusive_group()
  53. group.add_argument('--with-software-layer-signature-check', action='store_true', dest='test_software_layer_signatures',
  54. default=True,
  55. help='check that software layers do not change signatures (on by default)')
  56. group.add_argument('--without-software-layer-signature-check', action='store_false', dest='test_software_layer_signatures',
  57. help='disable signature checking for software layers')
  58. parser.add_argument('-n', '--no-auto', help='Disable auto layer discovery',
  59. action='store_true')
  60. parser.add_argument('-d', '--debug', help='Enable debug output',
  61. action='store_true')
  62. parser.add_argument('-q', '--quiet', help='Print only errors',
  63. action='store_true')
  64. parser.add_argument('-h', '--help', action='help',
  65. default=argparse.SUPPRESS,
  66. help='show this help message and exit')
  67. args = parser.parse_args()
  68. if args.output_log:
  69. fh = logging.FileHandler(args.output_log)
  70. fh.setFormatter(logging.Formatter("%(levelname)s: %(message)s"))
  71. logger.addHandler(fh)
  72. if args.debug:
  73. logger.setLevel(logging.DEBUG)
  74. elif args.quiet:
  75. logger.setLevel(logging.ERROR)
  76. if not 'BUILDDIR' in os.environ:
  77. logger.error("You must source the environment before running this script.")
  78. logger.error("$ source oe-init-build-env")
  79. return 1
  80. builddir = os.environ['BUILDDIR']
  81. bblayersconf = os.path.join(builddir, 'conf', 'bblayers.conf')
  82. layers = detect_layers(args.layers, args.no_auto)
  83. if not layers:
  84. logger.error("Failed to detect layers")
  85. return 1
  86. if args.additional_layers:
  87. additional_layers = detect_layers(args.additional_layers, args.no_auto)
  88. else:
  89. additional_layers = []
  90. if args.dependency:
  91. dep_layers = detect_layers(args.dependency, args.no_auto)
  92. dep_layers = dep_layers + layers
  93. else:
  94. dep_layers = layers
  95. logger.info("Detected layers:")
  96. for layer in layers:
  97. if layer['type'] == LayerType.ERROR_BSP_DISTRO:
  98. logger.error("%s: Can't be DISTRO and BSP type at the same time."\
  99. " Both conf/distro and conf/machine folders were found."\
  100. % layer['name'])
  101. layers.remove(layer)
  102. elif layer['type'] == LayerType.ERROR_NO_LAYER_CONF:
  103. logger.error("%s: Don't have conf/layer.conf file."\
  104. % layer['name'])
  105. layers.remove(layer)
  106. else:
  107. logger.info("%s: %s, %s" % (layer['name'], layer['type'],
  108. layer['path']))
  109. if not layers:
  110. return 1
  111. # Find all dependencies, and get them checked too
  112. if not args.no_auto_dependency:
  113. depends = []
  114. for layer in layers:
  115. layer_depends = get_layer_dependencies(layer, dep_layers, logger)
  116. if layer_depends:
  117. for d in layer_depends:
  118. if d not in depends:
  119. depends.append(d)
  120. for d in depends:
  121. if d not in layers:
  122. logger.info("Adding %s to the list of layers to test, as a dependency", d['name'])
  123. layers.append(d)
  124. shutil.copyfile(bblayersconf, bblayersconf + '.backup')
  125. def cleanup_bblayers(signum, frame):
  126. shutil.copyfile(bblayersconf + '.backup', bblayersconf)
  127. os.unlink(bblayersconf + '.backup')
  128. signal.signal(signal.SIGTERM, cleanup_bblayers)
  129. signal.signal(signal.SIGINT, cleanup_bblayers)
  130. td = {}
  131. results = collections.OrderedDict()
  132. results_status = collections.OrderedDict()
  133. layers_tested = 0
  134. for layer in layers:
  135. if layer['type'] == LayerType.ERROR_NO_LAYER_CONF or \
  136. layer['type'] == LayerType.ERROR_BSP_DISTRO:
  137. continue
  138. # Reset to a clean backup copy for each run
  139. shutil.copyfile(bblayersconf + '.backup', bblayersconf)
  140. if check_bblayers(bblayersconf, layer['path'], logger):
  141. logger.info("%s already in %s. To capture initial signatures, layer under test should not present "
  142. "in BBLAYERS. Please remove %s from BBLAYERS." % (layer['name'], bblayersconf, layer['name']))
  143. results[layer['name']] = None
  144. results_status[layer['name']] = 'SKIPPED (Layer under test should not present in BBLAYERS)'
  145. continue
  146. logger.info('')
  147. logger.info("Setting up for %s(%s), %s" % (layer['name'], layer['type'],
  148. layer['path']))
  149. missing_dependencies = not add_layer_dependencies(bblayersconf, layer, dep_layers, logger)
  150. if not missing_dependencies:
  151. for additional_layer in additional_layers:
  152. if not add_layer_dependencies(bblayersconf, additional_layer, dep_layers, logger):
  153. missing_dependencies = True
  154. break
  155. if missing_dependencies:
  156. logger.info('Skipping %s due to missing dependencies.' % layer['name'])
  157. results[layer['name']] = None
  158. results_status[layer['name']] = 'SKIPPED (Missing dependencies)'
  159. layers_tested = layers_tested + 1
  160. continue
  161. if any(map(lambda additional_layer: not add_layers(bblayersconf, [additional_layer], logger),
  162. additional_layers)):
  163. logger.info('Skipping %s due to missing additional layers.' % layer['name'])
  164. results[layer['name']] = None
  165. results_status[layer['name']] = 'SKIPPED (Missing additional layers)'
  166. layers_tested = layers_tested + 1
  167. continue
  168. logger.info('Getting initial bitbake variables ...')
  169. td['bbvars'] = get_bb_vars()
  170. logger.info('Getting initial signatures ...')
  171. td['builddir'] = builddir
  172. try:
  173. td['sigs'], td['tunetasks'] = get_signatures(td['builddir'])
  174. except RuntimeError as e:
  175. logger.info(str(e))
  176. results[layer['name']] = None
  177. results_status[layer['name']] = 'FAIL (Generating world signatures)'
  178. layers_tested = layers_tested + 1
  179. continue
  180. td['machines'] = args.machines
  181. if not add_layers(bblayersconf, [layer], logger):
  182. logger.info('Skipping %s ???.' % layer['name'])
  183. results[layer['name']] = None
  184. results_status[layer['name']] = 'SKIPPED (Unknown)'
  185. layers_tested = layers_tested + 1
  186. continue
  187. result = test_layer(td, layer, args.test_software_layer_signatures)
  188. results[layer['name']] = result
  189. results_status[layer['name']] = 'PASS' if results[layer['name']].wasSuccessful() else 'FAIL'
  190. layers_tested = layers_tested + 1
  191. ret = 0
  192. if layers_tested:
  193. logger.info('')
  194. logger.info('Summary of results:')
  195. logger.info('')
  196. for layer_name in results_status:
  197. logger.info('%s ... %s' % (layer_name, results_status[layer_name]))
  198. if not results[layer_name] or not results[layer_name].wasSuccessful():
  199. ret = 2 # ret = 1 used for initialization errors
  200. cleanup_bblayers(None, None)
  201. return ret
  202. if __name__ == '__main__':
  203. try:
  204. ret = main()
  205. except Exception:
  206. ret = 1
  207. import traceback
  208. traceback.print_exc()
  209. sys.exit(ret)