spdx.py 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. # Copyright (c) 2021 The Linux Foundation
  2. #
  3. # SPDX-License-Identifier: Apache-2.0
  4. import os
  5. import uuid
  6. from west.commands import WestCommand
  7. from west import log
  8. from zspdx.sbom import SBOMConfig, makeSPDX, setupCmakeQuery
  9. SPDX_DESCRIPTION = """\
  10. This command creates an SPDX 2.2 tag-value bill of materials
  11. following the completion of a Zephyr build.
  12. Prior to the build, an empty file must be created at
  13. BUILDDIR/.cmake/api/v1/query/codemodel-v2 in order to enable
  14. the CMake file-based API, which the SPDX command relies upon.
  15. This can be done by calling `west spdx --init` prior to
  16. calling `west build`."""
  17. class ZephyrSpdx(WestCommand):
  18. def __init__(self):
  19. super().__init__(
  20. 'spdx',
  21. 'create SPDX bill of materials',
  22. SPDX_DESCRIPTION)
  23. def do_add_parser(self, parser_adder):
  24. parser = parser_adder.add_parser(self.name,
  25. help=self.help,
  26. description = self.description)
  27. # If you update these options, make sure to keep the docs in
  28. # doc/guides/west/zephyr-cmds.rst up to date.
  29. parser.add_argument('-i', '--init', action="store_true",
  30. help="initialize CMake file-based API")
  31. parser.add_argument('-d', '--build-dir',
  32. help="build directory")
  33. parser.add_argument('-n', '--namespace-prefix',
  34. help="namespace prefix")
  35. parser.add_argument('-s', '--spdx-dir',
  36. help="SPDX output directory")
  37. parser.add_argument('--analyze-includes', action="store_true",
  38. help="also analyze included header files")
  39. parser.add_argument('--include-sdk', action="store_true",
  40. help="also generate SPDX document for SDK")
  41. return parser
  42. def do_run(self, args, unknown_args):
  43. log.dbg(f"running zephyr SPDX generator")
  44. log.dbg(f" --init is", args.init)
  45. log.dbg(f" --build-dir is", args.build_dir)
  46. log.dbg(f" --namespace-prefix is", args.namespace_prefix)
  47. log.dbg(f" --spdx-dir is", args.spdx_dir)
  48. log.dbg(f" --analyze-includes is", args.analyze_includes)
  49. log.dbg(f" --include-sdk is", args.include_sdk)
  50. if args.init:
  51. do_run_init(args)
  52. else:
  53. do_run_spdx(args)
  54. def do_run_init(args):
  55. log.inf("initializing Cmake file-based API prior to build")
  56. if not args.build_dir:
  57. log.die("Build directory not specified; call `west spdx --init --build-dir=BUILD_DIR`")
  58. # initialize CMake file-based API - empty query file
  59. query_ready = setupCmakeQuery(args.build_dir)
  60. if query_ready:
  61. log.inf("initialized; run `west build` then run `west spdx`")
  62. else:
  63. log.err("Couldn't create Cmake file-based API query directory")
  64. log.err("You can manually create an empty file at $BUILDDIR/.cmake/api/v1/query/codemodel-v2")
  65. def do_run_spdx(args):
  66. if not args.build_dir:
  67. log.die("Build directory not specified; call `west spdx --build-dir=BUILD_DIR`")
  68. # create the SPDX files
  69. cfg = SBOMConfig()
  70. cfg.buildDir = args.build_dir
  71. if args.namespace_prefix:
  72. cfg.namespacePrefix = args.namespace_prefix
  73. else:
  74. # create default namespace according to SPDX spec
  75. # note that this is intentionally _not_ an actual URL where
  76. # this document will be stored
  77. cfg.namespacePrefix = f"http://spdx.org/spdxdocs/zephyr-{str(uuid.uuid4())}"
  78. if args.spdx_dir:
  79. cfg.spdxDir = args.spdx_dir
  80. else:
  81. cfg.spdxDir = os.path.join(args.build_dir, "spdx")
  82. if args.analyze_includes:
  83. cfg.analyzeIncludes = True
  84. if args.include_sdk:
  85. cfg.includeSDK = True
  86. # make sure SPDX directory exists, or create it if it doesn't
  87. if os.path.exists(cfg.spdxDir):
  88. if not os.path.isdir(cfg.spdxDir):
  89. log.err(f'SPDX output directory {cfg.spdxDir} exists but is not a directory')
  90. return
  91. # directory exists, we're good
  92. else:
  93. # create the directory
  94. os.makedirs(cfg.spdxDir, exist_ok=False)
  95. makeSPDX(cfg)