-
-
Notifications
You must be signed in to change notification settings - Fork 47
/
Copy pathtest_output_json.py
117 lines (103 loc) · 4.49 KB
/
test_output_json.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
# This file is part of CycloneDX Python Library
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# SPDX-License-Identifier: Apache-2.0
# Copyright (c) OWASP Foundation. All Rights Reserved.
import re
from typing import Any, Callable
from unittest import TestCase
from unittest.mock import Mock, patch
from warnings import warn
from ddt import data, ddt, idata, named_data, unpack
from cyclonedx.exception import CycloneDxException, MissingOptionalDependencyException
from cyclonedx.exception.model import (
InvalidOmniBorIdException,
InvalidSwhidException,
LicenseExpressionAlongWithOthersException,
UnknownComponentDependencyException,
)
from cyclonedx.exception.output import FormatNotSupportedException
from cyclonedx.model.bom import Bom
from cyclonedx.output.json import BY_SCHEMA_VERSION, Json
from cyclonedx.schema import OutputFormat, SchemaVersion
from cyclonedx.validation.json import JsonStrictValidator
from tests import SnapshotMixin, is_valid_for_schema_version, mksname
from tests._data.models import all_get_bom_funct_invalid, all_get_bom_funct_valid, bom_all_same_bomref
UNSUPPORTED_SV = frozenset((SchemaVersion.V1_1, SchemaVersion.V1_0,))
@ddt
class TestOutputJson(TestCase, SnapshotMixin):
@data(*UNSUPPORTED_SV)
def test_unsupported_schema_raises(self, sv: SchemaVersion) -> None:
outputter_class = BY_SCHEMA_VERSION[sv]
self.assertTrue(issubclass(outputter_class, Json))
outputter = outputter_class(Mock(spec=Bom))
with self.assertRaises(FormatNotSupportedException):
outputter.output_as_string()
@named_data(*(
(f'{n}-{sv.to_version()}', gb, sv)
for n, gb in all_get_bom_funct_valid
for sv in SchemaVersion
if sv not in UNSUPPORTED_SV
and is_valid_for_schema_version(gb, sv)
))
@unpack
@patch('cyclonedx.builder.this.__ThisVersion', 'TESTING')
def test_valid(self, get_bom: Callable[[], Bom], sv: SchemaVersion, *_: Any, **__: Any) -> None:
snapshot_name = mksname(get_bom, sv, OutputFormat.JSON)
bom = get_bom()
json = BY_SCHEMA_VERSION[sv](bom).output_as_string(indent=2)
try:
errors = JsonStrictValidator(sv).validate_str(json)
except MissingOptionalDependencyException:
warn('!!! skipped schema validation',
category=UserWarning, stacklevel=0)
else:
self.assertIsNone(errors, json)
self.assertEqualSnapshot(json, snapshot_name)
@named_data(*(
(f'{n}-{sv.to_version()}', gb, sv)
for n, gb in all_get_bom_funct_invalid
for sv in SchemaVersion
if sv not in UNSUPPORTED_SV
and is_valid_for_schema_version(gb, sv)
))
@unpack
def test_invalid(self, get_bom: Callable[[], Bom], sv: SchemaVersion) -> None:
with self.assertRaises(CycloneDxException) as error:
bom = get_bom()
outputter = BY_SCHEMA_VERSION[sv](bom)
outputter.output_as_string()
if isinstance(error.exception, (
LicenseExpressionAlongWithOthersException,
InvalidOmniBorIdException,
InvalidSwhidException,
UnknownComponentDependencyException,
)):
return None # expected
raise error.exception
def test_bomref_not_duplicate(self) -> None:
bom, nr_bomrefs = bom_all_same_bomref()
output = BY_SCHEMA_VERSION[SchemaVersion.V1_4](bom).output_as_string()
found = re.findall(r'"bom-ref":\s*"(.*?)"', output)
self.assertEqual(nr_bomrefs, len(found))
self.assertCountEqual(set(found), found, 'expected unique items')
@ddt
class TestFunctionalBySchemaVersion(TestCase):
@idata(SchemaVersion)
def test_get_outputter_expected(self, sv: SchemaVersion) -> None:
outputter_class = BY_SCHEMA_VERSION[sv]
self.assertTrue(issubclass(outputter_class, Json))
outputter = outputter_class(Mock(spec=Bom))
self.assertIs(outputter.schema_version, sv)
self.assertIs(outputter.output_format, OutputFormat.JSON)