|
7 | 7 | import sys |
8 | 8 |
|
9 | 9 | from ..bomwriter import BomWriter |
10 | | -from .input import SbomInput, RepackInput |
| 10 | +from .input import SbomInput, RepackInput, SourceBinaryInput |
11 | 11 | from ..generate.generate import Debsbom |
12 | 12 | from ..repack.packer import BomTransformer, Packer |
13 | 13 | from ..resolver.resolver import PackageStreamResolver |
14 | 14 | from ..util.compression import Compression |
15 | | - |
| 15 | +from .download import DownloadCmd |
| 16 | +from ..sbom import SBOMType, SPDX_REF_DOCUMENT |
16 | 17 |
|
17 | 18 | logger = logging.getLogger(__name__) |
18 | 19 |
|
19 | 20 |
|
20 | | -class RepackCmd(SbomInput, RepackInput): |
| 21 | +class RepackCmd(SbomInput, RepackInput, SourceBinaryInput): |
21 | 22 | """ |
22 | 23 | Repacks the downloaded files into a uniform source archive, merging the |
23 | 24 | referenced source packages into a single archive and optionally applying |
@@ -49,11 +50,118 @@ def run(cls, args): |
49 | 50 | ) |
50 | 51 | resolvers = cls.get_sbom_resolvers(args) |
51 | 52 | for resolver in resolvers: |
| 53 | + filtered_pkgs = list( |
| 54 | + filter(lambda p: DownloadCmd._filter_pkg(p, args.sources, args.binaries), resolver) |
| 55 | + ) |
| 56 | + if not (args.sources and args.binaries): |
| 57 | + if resolver.sbom_type() == SBOMType.CycloneDX: |
| 58 | + if args.sources: |
| 59 | + from cyclonedx.model.dependency import Dependency |
| 60 | + |
| 61 | + resolver.document.components = [ |
| 62 | + comp |
| 63 | + for comp in resolver.document.components |
| 64 | + if "arch=source" in str(comp.bom_ref.value) |
| 65 | + ] |
| 66 | + |
| 67 | + root_ref = resolver.document.metadata.component.bom_ref |
| 68 | + source_refs = [ |
| 69 | + Dependency(ref=comp.bom_ref) for comp in resolver.document.components |
| 70 | + ] |
| 71 | + if root_ref: |
| 72 | + resolver.document.dependencies = [ |
| 73 | + Dependency(ref=root_ref, dependencies=source_refs) |
| 74 | + ] |
| 75 | + else: |
| 76 | + resolver.document.dependencies = [] |
| 77 | + |
| 78 | + elif args.binaries: |
| 79 | + resolver.document.components = [ |
| 80 | + comp |
| 81 | + for comp in resolver.document.components |
| 82 | + if "arch=source" not in str(comp.bom_ref.value) |
| 83 | + ] |
| 84 | + |
| 85 | + resolver.document.dependencies = [ |
| 86 | + dep |
| 87 | + for dep in resolver.document.dependencies |
| 88 | + if "arch=source" not in str(dep.ref.value) |
| 89 | + ] |
| 90 | + for dep in resolver.document.dependencies: |
| 91 | + dep.dependencies = [ |
| 92 | + deps |
| 93 | + for deps in dep.dependencies |
| 94 | + if "arch=source" not in str(deps.ref.value) |
| 95 | + ] |
| 96 | + |
| 97 | + elif resolver.sbom_type() == SBOMType.SPDX: |
| 98 | + if args.sources: |
| 99 | + from spdx_tools.spdx.model.relationship import ( |
| 100 | + Relationship, |
| 101 | + RelationshipType, |
| 102 | + ) |
| 103 | + |
| 104 | + resolver.document.packages = [ |
| 105 | + pkg |
| 106 | + for pkg in resolver.document.packages |
| 107 | + if any( |
| 108 | + "arch=source" in ref.locator |
| 109 | + for ref in pkg.external_references |
| 110 | + if ref.reference_type == "purl" |
| 111 | + ) |
| 112 | + ] |
| 113 | + |
| 114 | + src_pkg_ids = [pkg.spdx_id for pkg in resolver.document.packages] |
| 115 | + root_ref = None |
| 116 | + for rel in resolver.document.relationships: |
| 117 | + if rel.spdx_element_id == SPDX_REF_DOCUMENT: |
| 118 | + root_ref = rel.related_spdx_element_id |
| 119 | + break |
| 120 | + |
| 121 | + new_relationships = [] |
| 122 | + |
| 123 | + if root_ref: |
| 124 | + new_relationships.append( |
| 125 | + Relationship( |
| 126 | + spdx_element_id=SPDX_REF_DOCUMENT, |
| 127 | + related_spdx_element_id=root_ref, |
| 128 | + relationship_type=RelationshipType.DESCRIBES, |
| 129 | + ) |
| 130 | + ) |
| 131 | + |
| 132 | + for pkg_id in src_pkg_ids: |
| 133 | + new_relationships.append( |
| 134 | + Relationship( |
| 135 | + spdx_element_id=pkg_id, |
| 136 | + related_spdx_element_id=root_ref, |
| 137 | + relationship_type=RelationshipType.DEPENDS_ON, |
| 138 | + ) |
| 139 | + ) |
| 140 | + resolver.document.relationships = new_relationships |
| 141 | + |
| 142 | + elif args.binaries: |
| 143 | + resolver.document.packages = [ |
| 144 | + pkg |
| 145 | + for pkg in resolver.document.packages |
| 146 | + if any( |
| 147 | + "arch=source" not in ref.locator |
| 148 | + for ref in pkg.external_references |
| 149 | + if ref.reference_type == "purl" |
| 150 | + ) |
| 151 | + ] |
| 152 | + |
| 153 | + binary_ids = {pkg.spdx_id: pkg for pkg in resolver.document.packages} |
| 154 | + resolver.document.relationships = [ |
| 155 | + rel |
| 156 | + for rel in resolver.document.relationships |
| 157 | + if rel.spdx_element_id in binary_ids |
| 158 | + and rel.related_spdx_element_id in binary_ids |
| 159 | + ] |
52 | 160 | bt = BomTransformer.create(args.format, resolver.sbom_type(), resolver.document) |
53 | 161 | if pkg_subset: |
54 | | - pkgs = filter(lambda p: p in pkg_subset, resolver) |
| 162 | + pkgs = filter(lambda p: p in pkg_subset, filtered_pkgs) |
55 | 163 | else: |
56 | | - pkgs = resolver |
| 164 | + pkgs = filtered_pkgs |
57 | 165 | repacked = filter( |
58 | 166 | lambda p: p, |
59 | 167 | map( |
@@ -93,3 +201,4 @@ def setup_parser(cls, parser): |
93 | 201 | help="validate generated SBOM (only for SPDX)", |
94 | 202 | action="store_true", |
95 | 203 | ) |
| 204 | + cls.parser_add_source_binary_args(parser) |
0 commit comments