2020-03-17 01:25:38 +01:00
|
|
|
import datetime
|
2020-03-17 23:26:57 +01:00
|
|
|
|
2018-07-03 13:00:18 +02:00
|
|
|
import pytest
|
|
|
|
|
|
|
|
import stix2
|
2020-05-28 22:48:51 +02:00
|
|
|
import stix2.exceptions
|
2020-03-17 01:25:38 +01:00
|
|
|
import stix2.utils
|
2020-05-28 22:48:51 +02:00
|
|
|
import stix2.v21
|
|
|
|
import stix2.versioning
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
from .constants import CAMPAIGN_MORE_KWARGS
|
|
|
|
|
|
|
|
|
|
|
|
def test_making_new_version():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
campaign_v2 = campaign_v1.new_version(name="fred")
|
|
|
|
|
|
|
|
assert campaign_v1.id == campaign_v2.id
|
2018-07-03 15:40:51 +02:00
|
|
|
assert campaign_v1.spec_version == campaign_v2.spec_version
|
2018-07-03 13:00:18 +02:00
|
|
|
assert campaign_v1.created_by_ref == campaign_v2.created_by_ref
|
|
|
|
assert campaign_v1.created == campaign_v2.created
|
|
|
|
assert campaign_v1.name != campaign_v2.name
|
|
|
|
assert campaign_v2.name == "fred"
|
|
|
|
assert campaign_v1.description == campaign_v2.description
|
|
|
|
assert campaign_v1.modified < campaign_v2.modified
|
|
|
|
|
|
|
|
|
|
|
|
def test_making_new_version_with_unset():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
campaign_v2 = campaign_v1.new_version(description=None)
|
|
|
|
|
|
|
|
assert campaign_v1.id == campaign_v2.id
|
2018-07-03 15:40:51 +02:00
|
|
|
assert campaign_v1.spec_version == campaign_v2.spec_version
|
2018-07-03 13:00:18 +02:00
|
|
|
assert campaign_v1.created_by_ref == campaign_v2.created_by_ref
|
|
|
|
assert campaign_v1.created == campaign_v2.created
|
|
|
|
assert campaign_v1.name == campaign_v2.name
|
|
|
|
with pytest.raises(AttributeError):
|
|
|
|
assert campaign_v2.description
|
|
|
|
assert campaign_v1.modified < campaign_v2.modified
|
|
|
|
|
|
|
|
|
|
|
|
def test_making_new_version_with_embedded_object():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(
|
2018-07-03 13:00:18 +02:00
|
|
|
external_references=[{
|
|
|
|
"source_name": "capec",
|
2018-07-13 17:10:05 +02:00
|
|
|
"external_id": "CAPEC-163",
|
2018-07-03 13:00:18 +02:00
|
|
|
}],
|
|
|
|
**CAMPAIGN_MORE_KWARGS
|
|
|
|
)
|
|
|
|
|
2021-01-13 23:52:15 +01:00
|
|
|
campaign_v2 = campaign_v1.new_version(
|
|
|
|
external_references=[{
|
|
|
|
"source_name": "capec",
|
|
|
|
"external_id": "CAPEC-164",
|
|
|
|
}],
|
|
|
|
)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert campaign_v1.id == campaign_v2.id
|
2018-07-03 15:40:51 +02:00
|
|
|
assert campaign_v1.spec_version == campaign_v2.spec_version
|
2018-07-03 13:00:18 +02:00
|
|
|
assert campaign_v1.created_by_ref == campaign_v2.created_by_ref
|
|
|
|
assert campaign_v1.created == campaign_v2.created
|
|
|
|
assert campaign_v1.name == campaign_v2.name
|
|
|
|
assert campaign_v1.description == campaign_v2.description
|
|
|
|
assert campaign_v1.modified < campaign_v2.modified
|
|
|
|
assert campaign_v1.external_references[0].external_id != campaign_v2.external_references[0].external_id
|
|
|
|
|
|
|
|
|
|
|
|
def test_revoke():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
campaign_v2 = campaign_v1.revoke()
|
|
|
|
|
|
|
|
assert campaign_v1.id == campaign_v2.id
|
2018-07-03 15:40:51 +02:00
|
|
|
assert campaign_v1.spec_version == campaign_v2.spec_version
|
2018-07-03 13:00:18 +02:00
|
|
|
assert campaign_v1.created_by_ref == campaign_v2.created_by_ref
|
|
|
|
assert campaign_v1.created == campaign_v2.created
|
|
|
|
assert campaign_v1.name == campaign_v2.name
|
|
|
|
assert campaign_v1.description == campaign_v2.description
|
|
|
|
assert campaign_v1.modified < campaign_v2.modified
|
|
|
|
|
|
|
|
assert campaign_v2.revoked
|
|
|
|
|
|
|
|
|
|
|
|
def test_versioning_error_invalid_property():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
with pytest.raises(stix2.exceptions.UnmodifiablePropertyError) as excinfo:
|
|
|
|
campaign_v1.new_version(type="threat-actor")
|
|
|
|
|
|
|
|
assert str(excinfo.value) == "These properties cannot be changed when making a new version: type."
|
|
|
|
|
|
|
|
|
|
|
|
def test_versioning_error_bad_modified_value():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
with pytest.raises(stix2.exceptions.InvalidValueError) as excinfo:
|
|
|
|
campaign_v1.new_version(modified="2015-04-06T20:03:00.000Z")
|
|
|
|
|
2018-07-03 15:40:51 +02:00
|
|
|
assert excinfo.value.cls == stix2.v21.Campaign
|
2018-07-03 13:00:18 +02:00
|
|
|
assert excinfo.value.prop_name == "modified"
|
2018-07-13 17:10:05 +02:00
|
|
|
assert excinfo.value.reason == (
|
|
|
|
"The new modified datetime cannot be before than or equal to the current modified datetime."
|
|
|
|
"It cannot be equal, as according to STIX 2 specification, objects that are different "
|
|
|
|
"but have the same id and modified timestamp do not have defined consumer behavior."
|
|
|
|
)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
msg = "Invalid value for {0} '{1}': {2}"
|
2018-07-13 17:10:05 +02:00
|
|
|
msg = msg.format(
|
|
|
|
stix2.v21.Campaign.__name__, "modified",
|
|
|
|
"The new modified datetime cannot be before than or equal to the current modified datetime."
|
|
|
|
"It cannot be equal, as according to STIX 2 specification, objects that are different "
|
|
|
|
"but have the same id and modified timestamp do not have defined consumer behavior.",
|
|
|
|
)
|
2018-07-03 13:00:18 +02:00
|
|
|
assert str(excinfo.value) == msg
|
|
|
|
|
|
|
|
|
|
|
|
def test_versioning_error_usetting_required_property():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
with pytest.raises(stix2.exceptions.MissingPropertiesError) as excinfo:
|
|
|
|
campaign_v1.new_version(name=None)
|
|
|
|
|
2018-07-03 15:40:51 +02:00
|
|
|
assert excinfo.value.cls == stix2.v21.Campaign
|
2018-07-03 13:00:18 +02:00
|
|
|
assert excinfo.value.properties == ["name"]
|
|
|
|
|
|
|
|
msg = "No values for required properties for {0}: ({1})."
|
2018-07-03 15:40:51 +02:00
|
|
|
msg = msg.format(stix2.v21.Campaign.__name__, "name")
|
2018-07-03 13:00:18 +02:00
|
|
|
assert str(excinfo.value) == msg
|
|
|
|
|
|
|
|
|
|
|
|
def test_versioning_error_new_version_of_revoked():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2018-07-03 13:00:18 +02:00
|
|
|
campaign_v2 = campaign_v1.revoke()
|
|
|
|
|
|
|
|
with pytest.raises(stix2.exceptions.RevokeError) as excinfo:
|
|
|
|
campaign_v2.new_version(name="barney")
|
|
|
|
assert str(excinfo.value) == "Cannot create a new version of a revoked object."
|
|
|
|
|
|
|
|
assert excinfo.value.called_by == "new_version"
|
|
|
|
assert str(excinfo.value) == "Cannot create a new version of a revoked object."
|
|
|
|
|
|
|
|
|
|
|
|
def test_versioning_error_revoke_of_revoked():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2018-07-03 13:00:18 +02:00
|
|
|
campaign_v2 = campaign_v1.revoke()
|
|
|
|
|
|
|
|
with pytest.raises(stix2.exceptions.RevokeError) as excinfo:
|
|
|
|
campaign_v2.revoke()
|
|
|
|
assert str(excinfo.value) == "Cannot revoke an already revoked object."
|
|
|
|
|
|
|
|
assert excinfo.value.called_by == "revoke"
|
|
|
|
assert str(excinfo.value) == "Cannot revoke an already revoked object."
|
|
|
|
|
|
|
|
|
|
|
|
def test_making_new_version_dict():
|
|
|
|
campaign_v1 = CAMPAIGN_MORE_KWARGS
|
2020-05-28 22:48:51 +02:00
|
|
|
campaign_v2 = stix2.versioning.new_version(CAMPAIGN_MORE_KWARGS, name="fred")
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert campaign_v1['id'] == campaign_v2['id']
|
2018-07-03 15:40:51 +02:00
|
|
|
assert campaign_v1['spec_version'] == campaign_v2['spec_version']
|
2018-07-03 13:00:18 +02:00
|
|
|
assert campaign_v1['created_by_ref'] == campaign_v2['created_by_ref']
|
|
|
|
assert campaign_v1['created'] == campaign_v2['created']
|
|
|
|
assert campaign_v1['name'] != campaign_v2['name']
|
|
|
|
assert campaign_v2['name'] == "fred"
|
|
|
|
assert campaign_v1['description'] == campaign_v2['description']
|
|
|
|
assert stix2.utils.parse_into_datetime(campaign_v1['modified'], precision='millisecond') < campaign_v2['modified']
|
|
|
|
|
|
|
|
|
|
|
|
def test_versioning_error_dict_bad_modified_value():
|
|
|
|
with pytest.raises(stix2.exceptions.InvalidValueError) as excinfo:
|
2020-05-28 22:48:51 +02:00
|
|
|
stix2.versioning.new_version(CAMPAIGN_MORE_KWARGS, modified="2015-04-06T20:03:00.000Z")
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert excinfo.value.cls == dict
|
|
|
|
assert excinfo.value.prop_name == "modified"
|
|
|
|
assert excinfo.value.reason == "The new modified datetime cannot be before than or equal to the current modified datetime." \
|
|
|
|
"It cannot be equal, as according to STIX 2 specification, objects that are different " \
|
|
|
|
"but have the same id and modified timestamp do not have defined consumer behavior."
|
|
|
|
|
|
|
|
|
|
|
|
def test_versioning_error_dict_no_modified_value():
|
|
|
|
campaign_v1 = {
|
|
|
|
'type': 'campaign',
|
|
|
|
'id': "campaign--8e2e2d2b-17d4-4cbf-938f-98ee46b3cd3f",
|
|
|
|
'created': "2016-04-06T20:03:00.000Z",
|
|
|
|
'name': "Green Group Attacks Against Finance",
|
|
|
|
}
|
2020-05-28 22:48:51 +02:00
|
|
|
campaign_v2 = stix2.versioning.new_version(campaign_v1, modified="2017-04-06T20:03:00.000Z")
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert str(campaign_v2['modified']) == "2017-04-06T20:03:00.000Z"
|
|
|
|
|
|
|
|
|
|
|
|
def test_making_new_version_invalid_cls():
|
|
|
|
campaign_v1 = "This is a campaign."
|
|
|
|
with pytest.raises(ValueError) as excinfo:
|
2020-05-28 22:48:51 +02:00
|
|
|
stix2.versioning.new_version(campaign_v1, name="fred")
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert 'cannot create new version of object of this type' in str(excinfo.value)
|
|
|
|
|
|
|
|
|
|
|
|
def test_revoke_dict():
|
|
|
|
campaign_v1 = CAMPAIGN_MORE_KWARGS
|
2020-05-28 22:48:51 +02:00
|
|
|
campaign_v2 = stix2.versioning.revoke(campaign_v1)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert campaign_v1['id'] == campaign_v2['id']
|
2018-07-03 15:40:51 +02:00
|
|
|
assert campaign_v1['spec_version'] == campaign_v2['spec_version']
|
2018-07-03 13:00:18 +02:00
|
|
|
assert campaign_v1['created_by_ref'] == campaign_v2['created_by_ref']
|
|
|
|
assert campaign_v1['created'] == campaign_v2['created']
|
|
|
|
assert campaign_v1['name'] == campaign_v2['name']
|
|
|
|
assert campaign_v1['description'] == campaign_v2['description']
|
|
|
|
assert stix2.utils.parse_into_datetime(campaign_v1['modified'], precision='millisecond') < campaign_v2['modified']
|
|
|
|
|
|
|
|
assert campaign_v2['revoked']
|
|
|
|
|
|
|
|
|
2020-05-28 22:48:51 +02:00
|
|
|
def test_revoke_unversionable():
|
|
|
|
sco = stix2.v21.File(name="data.txt")
|
|
|
|
with pytest.raises(ValueError):
|
|
|
|
sco.revoke()
|
|
|
|
|
|
|
|
|
2018-07-03 13:00:18 +02:00
|
|
|
def test_versioning_error_revoke_of_revoked_dict():
|
|
|
|
campaign_v1 = CAMPAIGN_MORE_KWARGS
|
2020-05-28 22:48:51 +02:00
|
|
|
campaign_v2 = stix2.versioning.revoke(campaign_v1)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
with pytest.raises(stix2.exceptions.RevokeError) as excinfo:
|
2020-05-28 22:48:51 +02:00
|
|
|
stix2.versioning.revoke(campaign_v2)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert excinfo.value.called_by == "revoke"
|
|
|
|
|
|
|
|
|
|
|
|
def test_revoke_invalid_cls():
|
|
|
|
campaign_v1 = "This is a campaign."
|
|
|
|
with pytest.raises(ValueError) as excinfo:
|
2020-05-28 22:48:51 +02:00
|
|
|
stix2.versioning.revoke(campaign_v1)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert 'cannot revoke object of this type' in str(excinfo.value)
|
|
|
|
|
|
|
|
|
|
|
|
def test_remove_custom_stix_property():
|
2018-07-13 17:10:05 +02:00
|
|
|
mal = stix2.v21.Malware(
|
|
|
|
name="ColePowers",
|
|
|
|
malware_types=["rootkit"],
|
|
|
|
x_custom="armada",
|
|
|
|
allow_custom=True,
|
2019-07-01 21:26:30 +02:00
|
|
|
is_family=False,
|
2018-07-13 17:10:05 +02:00
|
|
|
)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
2020-05-28 22:48:51 +02:00
|
|
|
mal_nc = stix2.versioning.remove_custom_stix(mal)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert "x_custom" not in mal_nc
|
2020-03-17 01:25:38 +01:00
|
|
|
assert mal["modified"] < mal_nc["modified"]
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
|
|
|
|
def test_remove_custom_stix_object():
|
2018-07-13 17:10:05 +02:00
|
|
|
@stix2.v21.CustomObject(
|
|
|
|
"x-animal", [
|
|
|
|
("species", stix2.properties.StringProperty(required=True)),
|
|
|
|
("animal_class", stix2.properties.StringProperty()),
|
|
|
|
],
|
|
|
|
)
|
2018-07-03 13:00:18 +02:00
|
|
|
class Animal(object):
|
|
|
|
pass
|
|
|
|
|
|
|
|
animal = Animal(species="lion", animal_class="mammal")
|
|
|
|
|
2020-05-28 22:48:51 +02:00
|
|
|
nc = stix2.versioning.remove_custom_stix(animal)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert nc is None
|
|
|
|
|
|
|
|
|
|
|
|
def test_remove_custom_stix_no_custom():
|
2018-07-03 15:40:51 +02:00
|
|
|
campaign_v1 = stix2.v21.Campaign(**CAMPAIGN_MORE_KWARGS)
|
2020-05-28 22:48:51 +02:00
|
|
|
campaign_v2 = stix2.versioning.remove_custom_stix(campaign_v1)
|
2018-07-03 13:00:18 +02:00
|
|
|
|
|
|
|
assert len(campaign_v1.keys()) == len(campaign_v2.keys())
|
|
|
|
assert campaign_v1.id == campaign_v2.id
|
|
|
|
assert campaign_v1.description == campaign_v2.description
|
2020-03-17 01:25:38 +01:00
|
|
|
|
|
|
|
|
2020-03-17 23:26:57 +01:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"old, candidate_new, expected_new, use_stix21", [
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:07.001Z", "1999-08-15T00:19:07.001Z", False),
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:07.000Z", "1999-08-15T00:19:07.001Z", False),
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:06.000Z", "1999-08-15T00:19:07.001Z", False),
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:06.999Z", "1999-08-15T00:19:07.001Z", False),
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:07.0001Z", "1999-08-15T00:19:07.001Z", False),
|
|
|
|
("1999-08-15T00:19:07.999Z", "1999-08-15T00:19:07.9999Z", "1999-08-15T00:19:08.000Z", False),
|
|
|
|
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:07.001Z", "1999-08-15T00:19:07.001Z", True),
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:07.000Z", "1999-08-15T00:19:07.000001Z", True),
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:06.000Z", "1999-08-15T00:19:07.000001Z", True),
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:06.999999Z", "1999-08-15T00:19:07.000001Z", True),
|
|
|
|
("1999-08-15T00:19:07.000Z", "1999-08-15T00:19:07.000001Z", "1999-08-15T00:19:07.000001Z", True),
|
|
|
|
("1999-08-15T00:19:07.999Z", "1999-08-15T00:19:07.999999Z", "1999-08-15T00:19:07.999999Z", True),
|
|
|
|
],
|
|
|
|
)
|
2020-03-17 01:25:38 +01:00
|
|
|
def test_fudge_modified(old, candidate_new, expected_new, use_stix21):
|
|
|
|
old_dt = datetime.datetime.strptime(old, "%Y-%m-%dT%H:%M:%S.%fZ")
|
|
|
|
candidate_new_dt = datetime.datetime.strptime(
|
2020-03-17 23:26:57 +01:00
|
|
|
candidate_new, "%Y-%m-%dT%H:%M:%S.%fZ",
|
2020-03-17 01:25:38 +01:00
|
|
|
)
|
|
|
|
expected_new_dt = datetime.datetime.strptime(
|
2020-03-17 23:26:57 +01:00
|
|
|
expected_new, "%Y-%m-%dT%H:%M:%S.%fZ",
|
2020-03-17 01:25:38 +01:00
|
|
|
)
|
|
|
|
|
2020-05-28 22:48:51 +02:00
|
|
|
fudged = stix2.versioning._fudge_modified(
|
|
|
|
old_dt, candidate_new_dt, use_stix21,
|
|
|
|
)
|
2020-03-17 01:25:38 +01:00
|
|
|
assert fudged == expected_new_dt
|
2020-05-28 22:48:51 +02:00
|
|
|
|
|
|
|
|
|
|
|
def test_version_unversionable_dict():
|
|
|
|
f = {
|
|
|
|
"type": "file",
|
|
|
|
"id": "file--4efb5217-e987-4438-9a1b-c800099401df",
|
|
|
|
"name": "data.txt",
|
|
|
|
}
|
|
|
|
|
|
|
|
with pytest.raises(ValueError):
|
|
|
|
stix2.versioning.new_version(f)
|
|
|
|
|
|
|
|
|
|
|
|
def test_version_sco_with_custom():
|
|
|
|
"""
|
|
|
|
If we add custom properties named like versioning properties to an object
|
|
|
|
type which is otherwise unversionable, versioning should start working.
|
|
|
|
"""
|
|
|
|
|
|
|
|
file_sco_obj = stix2.v21.File(
|
|
|
|
name="data.txt",
|
|
|
|
created="1973-11-23T02:31:37Z",
|
|
|
|
modified="1991-05-13T19:24:57Z",
|
|
|
|
revoked=False,
|
|
|
|
allow_custom=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
new_file_sco_obj = stix2.versioning.new_version(
|
2020-06-05 01:50:52 +02:00
|
|
|
file_sco_obj, size=1234,
|
2020-05-28 22:48:51 +02:00
|
|
|
)
|
|
|
|
|
2020-06-05 01:50:52 +02:00
|
|
|
assert new_file_sco_obj.size == 1234
|
2020-05-28 22:48:51 +02:00
|
|
|
|
|
|
|
revoked_obj = stix2.versioning.revoke(new_file_sco_obj)
|
|
|
|
assert revoked_obj.revoked
|
|
|
|
|
|
|
|
|
2021-01-21 01:16:22 +01:00
|
|
|
def test_version_sco_id_contributing_properties():
|
|
|
|
file_sco_obj = stix2.v21.File(
|
|
|
|
name="data.txt",
|
|
|
|
created="1973-11-23T02:31:37Z",
|
|
|
|
modified="1991-05-13T19:24:57Z",
|
|
|
|
revoked=False,
|
|
|
|
allow_custom=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
with pytest.raises(stix2.exceptions.UnmodifiablePropertyError) as e:
|
|
|
|
stix2.versioning.new_version(file_sco_obj, name="foo.dat")
|
|
|
|
|
|
|
|
assert e.value.unchangable_properties == {"name"}
|
|
|
|
|
|
|
|
|
|
|
|
def test_version_sco_id_contributing_properties_dict():
|
|
|
|
file_sco_dict = {
|
|
|
|
"type": "file",
|
|
|
|
"id": "file--c27c572c-2e17-5ce1-817e-67bb97629a56",
|
|
|
|
"spec_version": "2.1",
|
|
|
|
"name": "data.txt",
|
|
|
|
"created": "1973-11-23T02:31:37Z",
|
|
|
|
"modified": "1991-05-13T19:24:57Z",
|
|
|
|
"revoked": False
|
|
|
|
}
|
|
|
|
|
|
|
|
with pytest.raises(stix2.exceptions.UnmodifiablePropertyError) as e:
|
|
|
|
stix2.versioning.new_version(file_sco_dict, name="foo.dat")
|
|
|
|
|
|
|
|
assert e.value.unchangable_properties == {"name"}
|
|
|
|
|
|
|
|
|
2020-05-28 22:48:51 +02:00
|
|
|
def test_version_disable_custom():
|
|
|
|
m = stix2.v21.Malware(
|
|
|
|
name="foo", description="Steals your identity!", is_family=False,
|
|
|
|
x_custom=123, allow_custom=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Remove the custom property, and disallow custom properties in the
|
|
|
|
# resulting object.
|
|
|
|
m2 = stix2.versioning.new_version(m, x_custom=None, allow_custom=False)
|
|
|
|
assert "x_custom" not in m2
|
|
|
|
|
|
|
|
# Remove a regular property and leave the custom one, disallow custom
|
|
|
|
# properties, and make sure we get an error.
|
|
|
|
with pytest.raises(stix2.exceptions.ExtraPropertiesError):
|
|
|
|
stix2.versioning.new_version(m, description=None, allow_custom=False)
|
|
|
|
|
|
|
|
|
|
|
|
def test_version_enable_custom():
|
|
|
|
m = stix2.v21.Malware(
|
|
|
|
name="foo", description="Steals your identity!", is_family=False,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Add a custom property to an object for which it was previously disallowed
|
|
|
|
m2 = stix2.versioning.new_version(m, x_custom=123, allow_custom=True)
|
|
|
|
assert "x_custom" in m2
|
|
|
|
|
|
|
|
# Add a custom property without enabling it, make sure we get an error
|
|
|
|
with pytest.raises(stix2.exceptions.ExtraPropertiesError):
|
|
|
|
stix2.versioning.new_version(m, x_custom=123, allow_custom=False)
|
|
|
|
|
|
|
|
|
|
|
|
def test_version_propagate_custom():
|
|
|
|
m = stix2.v21.Malware(
|
|
|
|
name="foo", is_family=False,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Remember custom-not-allowed setting from original; produce error
|
|
|
|
with pytest.raises(stix2.exceptions.ExtraPropertiesError):
|
|
|
|
stix2.versioning.new_version(m, x_custom=123)
|
|
|
|
|
|
|
|
m2 = stix2.versioning.new_version(m, description="Steals your identity!")
|
|
|
|
assert "description" in m2
|
|
|
|
assert m2.description == "Steals your identity!"
|
|
|
|
|
|
|
|
m_custom = stix2.v21.Malware(
|
|
|
|
name="foo", is_family=False, x_custom=123, allow_custom=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Remember custom-allowed setting from original; should work
|
|
|
|
m2_custom = stix2.versioning.new_version(m_custom, x_other_custom="abc")
|
|
|
|
assert "x_other_custom" in m2_custom
|
|
|
|
assert m2_custom.x_other_custom == "abc"
|