diff --git a/Dockerfile-builder b/Dockerfile-builder index 0bddbe60..6d842de1 100644 --- a/Dockerfile-builder +++ b/Dockerfile-builder @@ -7,7 +7,7 @@ FROM python:${VERSION}-bookworm # .github/workflows/python-package.yml # .github/workflows/python-publish.yml # -ENV DEBIAN_FRONTEND noninteractive +ENV DEBIAN_FRONTEND=noninteractive RUN curl -fsSOL https://taskfile.dev/install.sh \ && sh install.sh \ diff --git a/archivist/archivist.py b/archivist/archivist.py index a4f104f1..6929928e 100644 --- a/archivist/archivist.py +++ b/archivist/archivist.py @@ -108,11 +108,15 @@ def __init__( fixtures: "dict[str,dict[Any,Any]]|None" = None, verify: bool = True, max_time: float = MAX_TIME, + partner_id: str = "", + user_agent: str = "", ): super().__init__( fixtures=fixtures, verify=verify, max_time=max_time, + partner_id=partner_id, + user_agent=user_agent, ) if isinstance(auth, tuple): @@ -214,10 +218,12 @@ def __copy__(self) -> "Archivist": fixtures=deepcopy(self._fixtures), verify=self._verify, max_time=self._max_time, + partner_id=self._partner_id, + user_agent=self._user_agent, ) def _add_headers(self, headers: "dict[str,str]|None") -> "dict[str,Any]": - newheaders = {**headers} if isinstance(headers, dict) else {} + newheaders = super()._add_headers(headers) auth = self.auth # this may trigger a refetch so only do it once here # for appidp endpoint there may not be an authtoken diff --git a/archivist/archivistpublic.py b/archivist/archivistpublic.py index 9480b207..2c390395 100644 --- a/archivist/archivistpublic.py +++ b/archivist/archivistpublic.py @@ -33,12 +33,16 @@ from requests.models import Response +from .about import __version__ as VERSION from .assetattachments import _AssetAttachmentsClient from .assets import _AssetsPublic from .confirmer import MAX_TIME from .constants import ( HEADERS_REQUEST_TOTAL_COUNT, HEADERS_TOTAL_COUNT, + PARTNER_ID, + USER_AGENT, + USER_AGENT_PREFIX, ) from .dictmerge import _deepmerge, _dotdict from .errors import ( @@ -82,12 +86,16 @@ def __init__( fixtures: "dict[str, Any]|None" = None, verify: bool = True, max_time: float = MAX_TIME, + partner_id: str = "", + user_agent: str = "", ): self._verify = verify self._response_ring_buffer = deque(maxlen=self.RING_BUFFER_MAX_LEN) self._session = None self._max_time = max_time self._fixtures = fixtures or {} + self._partner_id = partner_id + self._user_agent = user_agent # Type hints for IDE autocomplete, keep in sync with CLIENTS map above self.assets: _AssetsPublic @@ -151,6 +159,21 @@ def max_time(self) -> float: """bool: Returns maximum time in seconds to wait for confirmation""" return self._max_time + @property + def version(self) -> str: + """str: Returns version of the archivist package""" + return VERSION + + @property + def partner_id(self) -> str: + """str: Returns partner id if set when initialising an instance of this class""" + return self._partner_id + + @property + def user_agent(self) -> str: + """str: Returns partner id if set when initialising an instance of this class""" + return self._user_agent + @property def fixtures(self) -> "dict[str, Any]": """dict: Contains predefined attributes for each endpoint""" @@ -170,6 +193,17 @@ def __copy__(self): def _add_headers(self, headers: "dict[str, str]|None") -> "dict[str, str]": newheaders = {**headers} if headers is not None else {} + u = self.user_agent + if u: + newheaders[USER_AGENT] = ( + f"{self.user_agent} {USER_AGENT_PREFIX}{self.version}" + ) + else: + newheaders[USER_AGENT] = f"{USER_AGENT_PREFIX}{self.version}" + + p = self.partner_id + if p: + newheaders[PARTNER_ID] = p return newheaders diff --git a/archivist/constants.py b/archivist/constants.py index b48b6b67..065dc16c 100644 --- a/archivist/constants.py +++ b/archivist/constants.py @@ -13,6 +13,10 @@ "RecordEvidence", ] +USER_AGENT = "DataTrails-User-Agent" +USER_AGENT_PREFIX = "pysdk/" +PARTNER_ID = "DataTrails-Partner-ID" + # define in MIME canonical form HEADERS_REQUEST_TOTAL_COUNT = "X-Request-Total-Count" HEADERS_TOTAL_COUNT = "X-Total-Count" diff --git a/archivist/parser.py b/archivist/parser.py index 278ec890..9ed401d1 100644 --- a/archivist/parser.py +++ b/archivist/parser.py @@ -129,6 +129,15 @@ def common_parser(description: str): default=None, help="namespace of item population", ) + parser.add_argument( + "-p", + "--partner_id", + type=str, + dest="partner_id", + action="store", + default=None, + help="partner id", + ) return parser @@ -171,7 +180,12 @@ def endpoint(args): LOGGER.error("Critical error. Aborting.") sys_exit(1) - arch = Archivist(args.url, auth, fixtures=fixtures) + arch = Archivist( + args.url, + auth, + fixtures=fixtures, + partner_id=args.partner_id, + ) if arch is None: LOGGER.error("Critical error. Aborting.") sys_exit(1) diff --git a/examples/create_asset.py b/examples/create_asset.py index 7df12af1..cd2ee158 100644 --- a/examples/create_asset.py +++ b/examples/create_asset.py @@ -75,7 +75,7 @@ def main(): create an example archivist connection and create an asset. """ - # optional call to set the logger level for all subsystems. The argumant can + # optional call to set the logger level for all subsystems. The argument can # be either "INFO" or "DEBUG". For more sophisticated logging control see the # documentation. set_logger("INFO") @@ -97,10 +97,14 @@ def main(): # Initialize connection to Archivist. max_time is the time to wait for confirmation # of an asset or event creation - the default is 300 seconds but one can optionally # specify a different value. + # The optional partner id field is allocated by Datatrails to partners - partners are then + # expected to specify this value when submitting any request to the archivist product. + # Leave blank if if you do not have a partner ID. with Archivist( "https://app.datatrails.ai", auth, max_time=300, + partner_id="acme/f7a6beef-f01c-4b39-a494-3fa6b45d6bf4", ) as arch: # Create a new asset asset = create_asset(arch) diff --git a/functests/execaccess_policies.py b/functests/execaccess_policies.py index 7fe6559c..ee397273 100644 --- a/functests/execaccess_policies.py +++ b/functests/execaccess_policies.py @@ -95,6 +95,7 @@ }, ], } +PARTNER_IDENTIFIER = "datatrails/1234567890" if getenv("DATATRAILS_LOGLEVEL") is not None: logger.set_logger(getenv("DATATRAILS_LOGLEVEL")) @@ -117,7 +118,11 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + partner_id=PARTNER_IDENTIFIER, + ) # these are for access_policies self.ac_props = deepcopy(PROPS) @@ -298,8 +303,11 @@ def setUp(self): super().setUp() with open(getenv("DATATRAILS_AUTHTOKEN_FILENAME_2"), encoding="utf-8") as fd: auth_2 = fd.read().strip() - self.arch_2 = Archivist(getenv("DATATRAILS_URL"), auth_2) - + self.arch_2 = Archivist( + getenv("DATATRAILS_URL"), + auth_2, + partner_id=PARTNER_IDENTIFIER, + ) # creates reciprocal subjects for arch 1 and arch 2. # subject 1 contains details of subject 2 to be shared self.subject_1, self.subject_2 = self.arch.subjects.share( diff --git a/functests/execapplications.py b/functests/execapplications.py index 489ba32e..b8a435cb 100644 --- a/functests/execapplications.py +++ b/functests/execapplications.py @@ -32,6 +32,7 @@ "arc_description": "Traffic flow control light at A603 North East", "some_custom_attribute": "value", } +PARTNER_IDENTIFIER = "datatrails/1234567890" if getenv("DATATRAILS_LOGLEVEL") is not None: logger.set_logger(getenv("DATATRAILS_LOGLEVEL")) @@ -54,7 +55,11 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + partner_id=PARTNER_IDENTIFIER, + ) self.display_name = f"{DISPLAY_NAME} {uuid4()}" def tearDown(self): diff --git a/functests/execassets.py b/functests/execassets.py index 10fdceb0..9c357a9f 100644 --- a/functests/execassets.py +++ b/functests/execassets.py @@ -70,6 +70,7 @@ }, ], } +PARTNER_IDENTIFIER = "datatrails/1234567890" class TestAssetCreate(TestCase): @@ -87,7 +88,12 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth, max_time=30) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + max_time=30, + partner_id=PARTNER_IDENTIFIER, + ) self.attrs = deepcopy(ATTRS) self.traffic_light = deepcopy(ATTRS) self.traffic_light["arc_display_type"] = "Traffic light with violation camera" @@ -233,7 +239,12 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth, max_time=600) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + max_time=30, + partner_id=PARTNER_IDENTIFIER, + ) def tearDown(self): self.arch.close() diff --git a/functests/execattachments.py b/functests/execattachments.py index 63ead67d..8a7bd366 100644 --- a/functests/execattachments.py +++ b/functests/execattachments.py @@ -16,6 +16,8 @@ from .constants import TestCase +PARTNER_IDENTIFIER = "datatrails/1234567890" + if getenv("DATATRAILS_LOGLEVEL") is not None: logger.set_logger(getenv("DATATRAILS_LOGLEVEL")) @@ -46,7 +48,11 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + partner_id=PARTNER_IDENTIFIER, + ) self.file_uuid: str = "" with suppress(FileNotFoundError): @@ -208,7 +214,11 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + partner_id=PARTNER_IDENTIFIER, + ) def tearDown(self): self.arch.close() diff --git a/functests/execcompliance_policies.py b/functests/execcompliance_policies.py index 0d881d1f..916b707e 100644 --- a/functests/execcompliance_policies.py +++ b/functests/execcompliance_policies.py @@ -82,6 +82,7 @@ ["rad<7"], ], ) +PARTNER_IDENTIFIER = "datatrails/1234567890" class TestCompliancePoliciesBase(TestCase): @@ -99,7 +100,11 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + partner_id=PARTNER_IDENTIFIER, + ) self.identities = [] def tearDown(self): diff --git a/functests/execnotebooks.py b/functests/execnotebooks.py index a0e05212..d75095b5 100644 --- a/functests/execnotebooks.py +++ b/functests/execnotebooks.py @@ -16,6 +16,8 @@ from .constants import TestCase +PARTNER_IDENTIFIER = "datatrails/1234567890" + if getenv("DATATRAILS_LOGLEVEL") is not None: logger.set_logger(getenv("DATATRAILS_LOGLEVEL")) @@ -37,7 +39,11 @@ def setUp(self): self.client_id = getenv("DATATRAILS_APPREG_CLIENT") self.client_secret = getenv("DATATRAILS_APPREG_SECRET") self.url = getenv("DATATRAILS_URL") - self.arch = Archivist(self.url, (self.client_id, self.client_secret)) + self.arch = Archivist( + self.url, + (self.client_id, self.client_secret), + partner_id=PARTNER_IDENTIFIER, + ) def tearDown(self): self.arch.close() diff --git a/functests/execpublicassets.py b/functests/execpublicassets.py index abfa8490..6ee87101 100644 --- a/functests/execpublicassets.py +++ b/functests/execpublicassets.py @@ -66,6 +66,7 @@ ], "public": True, } +PARTNER_IDENTIFIER = "datatrails/1234567890" class TestPublicAssetCreate(TestCase): @@ -84,7 +85,12 @@ def setUp(self): client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) self.url = getenv("DATATRAILS_URL") - self.arch = Archivist(self.url, auth, max_time=600) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + max_time=30, + partner_id=PARTNER_IDENTIFIER, + ) self.attrs = deepcopy(ATTRS) self.traffic_light = deepcopy(ATTRS) self.traffic_light["arc_display_type"] = "Traffic light with violation camera" diff --git a/functests/execrunner.py b/functests/execrunner.py index 9521e549..c1fbeb99 100644 --- a/functests/execrunner.py +++ b/functests/execrunner.py @@ -18,6 +18,7 @@ # pylint: disable=missing-docstring # pylint: disable=unused-variable +PARTNER_IDENTIFIER = "datatrails/1234567890" if getenv("DATATRAILS_LOGLEVEL") is not None: logger.set_logger(getenv("DATATRAILS_LOGLEVEL")) @@ -40,7 +41,12 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth, max_time=300) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + max_time=30, + partner_id=PARTNER_IDENTIFIER, + ) def tearDown(self): self.arch.close() diff --git a/functests/execsubjects.py b/functests/execsubjects.py index a56efc81..36dd5294 100644 --- a/functests/execsubjects.py +++ b/functests/execsubjects.py @@ -44,6 +44,7 @@ "0aW9uX3N0YXR1cyI6ICJDT05GSVJNQVRJT05fU1RBVFVTX1VOU1BFQ0lGSU" "VEIn0=" ) +PARTNER_IDENTIFIER = "datatrails/1234567890" class TestSubjects(TestCase): @@ -61,7 +62,11 @@ def setUp(self): client_secret=getenv("DATATRAILS_APPREG_SECRET"), client_secret_filename=getenv("DATATRAILS_APPREG_SECRET_FILENAME"), ) - self.arch = Archivist(getenv("DATATRAILS_URL"), auth) + self.arch = Archivist( + getenv("DATATRAILS_URL"), + auth, + partner_id=PARTNER_IDENTIFIER, + ) self.display_name = f"{DISPLAY_NAME} {uuid4()}" def tearDown(self): diff --git a/pyproject.toml b/pyproject.toml index 9c331fb1..fb1eee1c 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -259,7 +259,7 @@ valid-metaclass-classmethod-first-arg = ["cls"] # ignored-parents = # Maximum number of arguments for function / method. -max-args = 6 +max-args = 8 # Maximum number of attributes for a class (see R0902). max-attributes = 7 diff --git a/unittests/testaccess_policies.py b/unittests/testaccess_policies.py index 301f2b46..640621b6 100644 --- a/unittests/testaccess_policies.py +++ b/unittests/testaccess_policies.py @@ -4,6 +4,7 @@ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.constants import ( ACCESS_POLICIES_LABEL, @@ -12,7 +13,10 @@ ASSETS_LABEL, HEADERS_REQUEST_TOTAL_COUNT, HEADERS_TOTAL_COUNT, + PARTNER_ID, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistBadRequestError @@ -67,6 +71,8 @@ IDENTITY = f"{ACCESS_POLICIES_LABEL}/xxxxxxxx" SUBPATH = f"{ACCESS_POLICIES_SUBPATH}/{ACCESS_POLICIES_LABEL}" ASSET_ID = f"{ASSETS_LABEL}/yyyyyyyy" +PARTNER_ID_VALUE = "acme/1234567890" +USER_AGENT_VALUE = "archivist-samples/v0.1.0" RESPONSE = { **PROPS, @@ -81,6 +87,64 @@ } +class TestAccessPoliciesPartnerID(TestCase): + """ + Test Archivist AccessPolicies Create method + """ + + maxDiff = None + + def setUp(self): + self.arch = Archivist( + "url", + "authauthauth", + partner_id=PARTNER_ID_VALUE, + user_agent=USER_AGENT_VALUE, + ) + + def tearDown(self): + self.arch.close() + + def test_access_policies_create(self): + """ + Test access_policy creation + """ + with mock.patch.object(self.arch.session, "post") as mock_post: + mock_post.return_value = MockResponse(200, **RESPONSE) + + access_policy = self.arch.access_policies.create( + PROPS, FILTERS, ACCESS_PERMISSIONS + ) + args, kwargs = mock_post.call_args + self.assertEqual( + args, + (f"url/{ROOT}/{SUBPATH}",), + msg="CREATE method args called incorrectly", + ) + self.assertEqual( + kwargs, + { + "json": REQUEST, + "headers": { + "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_VALUE} {USER_AGENT_PREFIX}{VERSION}", + PARTNER_ID: PARTNER_ID_VALUE, + }, + }, + msg="CREATE method called incorrectly", + ) + self.assertEqual( + access_policy, + RESPONSE, + msg="CREATE method called incorrectly", + ) + self.assertEqual( + access_policy.name, + ACCESS_POLICY_NAME, + msg="Incorrect name property", + ) + + class TestAccessPolicies(TestCase): """ Test Archivist AccessPolicies Create method @@ -126,6 +190,7 @@ def test_access_policies_create(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method called incorrectly", @@ -156,6 +221,7 @@ def test_access_policies_read(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -178,6 +244,7 @@ def test_access_policies_delete(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, ), @@ -207,6 +274,7 @@ def test_access_policies_update(self): "json": PROPS, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="PATCH method kwargs called incorrectly", @@ -243,6 +311,7 @@ def test_access_policies_count(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 1}, }, @@ -279,6 +348,7 @@ def test_access_policies_count_by_name(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 1, @@ -322,6 +392,7 @@ def test_access_policies_list(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -366,6 +437,7 @@ def test_access_policies_list_by_name(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"display_name": "Policy display name"}, }, @@ -411,6 +483,7 @@ def test_access_policies_list_matching_access_policies(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -453,6 +526,7 @@ def test_access_policies_list_matching_assets(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, diff --git a/unittests/testapplications.py b/unittests/testapplications.py index c79f57e1..5f0a59ff 100644 --- a/unittests/testapplications.py +++ b/unittests/testapplications.py @@ -4,12 +4,15 @@ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.constants import ( APPLICATIONS_LABEL, APPLICATIONS_REGENERATE, APPLICATIONS_SUBPATH, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistBadRequestError @@ -94,6 +97,7 @@ def test_applications_create(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -119,6 +123,7 @@ def test_applications_read(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -141,6 +146,7 @@ def test_applications_delete(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, ), @@ -166,6 +172,7 @@ def test_applications_update(self): "json": UPDATE_DATA, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, ), @@ -214,6 +221,7 @@ def test_applications_list(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {}, }, @@ -258,6 +266,7 @@ def test_applications_list_by_name(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "display_name": "Application display name", @@ -287,6 +296,7 @@ def test_applications_regenerate(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "json": None, }, diff --git a/unittests/testarchivist.py b/unittests/testarchivist.py index 881bb609..a4dc691b 100644 --- a/unittests/testarchivist.py +++ b/unittests/testarchivist.py @@ -6,8 +6,14 @@ from os import environ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist -from archivist.constants import HEADERS_RETRY_AFTER, HEADERS_TOTAL_COUNT +from archivist.constants import ( + HEADERS_RETRY_AFTER, + HEADERS_TOTAL_COUNT, + USER_AGENT, + USER_AGENT_PREFIX, +) from archivist.errors import ( ArchivistBadRequestError, ArchivistError, @@ -268,6 +274,7 @@ def test_patch(self): "json": request, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="POST method kwargs called incorrectly", @@ -295,7 +302,10 @@ def test_patch_with_headers(self): self.arch.patch( "path/path/entity/xxxx", request, - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) args, kwargs = mock_patch.call_args self.assertEqual( @@ -310,6 +320,7 @@ def test_patch_with_headers(self): "headers": { "authorization": "Bearer authauthauth", "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="PATCH method kwargs called incorrectly", @@ -386,6 +397,7 @@ def test_patch_with_429_retry_and_success(self): "json": request, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="PATCH method kwargs called incorrectly", diff --git a/unittests/testarchivistdelete.py b/unittests/testarchivistdelete.py index f60f25c8..67db14f6 100644 --- a/unittests/testarchivistdelete.py +++ b/unittests/testarchivistdelete.py @@ -4,8 +4,13 @@ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist -from archivist.constants import HEADERS_RETRY_AFTER +from archivist.constants import ( + HEADERS_RETRY_AFTER, + USER_AGENT, + USER_AGENT_PREFIX, +) from archivist.errors import ( ArchivistNotFoundError, ArchivistTooManyRequestsError, @@ -49,6 +54,7 @@ def test_delete(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, ), @@ -72,7 +78,10 @@ def test_delete_with_headers(self): mock_delete.return_value = MockResponse(200) self.arch.delete( "path/path/id/xxxxxxxx", - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) self.assertEqual( tuple(mock_delete.call_args), @@ -82,6 +91,7 @@ def test_delete_with_headers(self): "headers": { "authorization": "Bearer authauthauth", "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, ), @@ -97,7 +107,10 @@ def test_delete_with_429(self): with self.assertRaises(ArchivistTooManyRequestsError): self.arch.delete( "path/path/id/xxxxxxxx", - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) def test_delete_with_429_retry_and_fail(self): @@ -144,6 +157,7 @@ def test_delete_with_429_retry_and_success(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, ), diff --git a/unittests/testarchivistget.py b/unittests/testarchivistget.py index 151da3b7..c0993ec7 100644 --- a/unittests/testarchivistget.py +++ b/unittests/testarchivistget.py @@ -5,8 +5,13 @@ from io import BytesIO from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist -from archivist.constants import HEADERS_RETRY_AFTER +from archivist.constants import ( + HEADERS_RETRY_AFTER, + USER_AGENT, + USER_AGENT_PREFIX, +) from archivist.errors import ( ArchivistNotFoundError, ArchivistTooManyRequestsError, @@ -50,6 +55,7 @@ def test_get(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -84,7 +90,10 @@ def test_get_with_headers(self): mock_get.return_value = MockResponse(200) self.arch.get( "path/path/id/xxxxxxxx", - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) self.assertEqual( tuple(mock_get.call_args), @@ -94,6 +103,7 @@ def test_get_with_headers(self): "headers": { "authorization": "Bearer authauthauth", "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -110,7 +120,10 @@ def test_get_with_429(self): with self.assertRaises(ArchivistTooManyRequestsError): self.arch.get( "path/path/id/xxxxxxxx", - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) def test_get_with_429_retry_and_fail(self): @@ -157,6 +170,7 @@ def test_get_with_429_retry_and_success(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -206,6 +220,7 @@ def filedata(chunk_size=4096): # pylint: disable=unused-argument { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "stream": True, "params": None, @@ -298,6 +313,7 @@ def filedata(chunk_size=4096): # pylint: disable=unused-argument { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "stream": True, "params": None, diff --git a/unittests/testarchivistlist.py b/unittests/testarchivistlist.py index 5c848663..ad065d16 100644 --- a/unittests/testarchivistlist.py +++ b/unittests/testarchivistlist.py @@ -5,7 +5,13 @@ from os import environ from unittest import mock -from archivist.constants import HEADERS_RETRY_AFTER, HEADERS_TOTAL_COUNT +from archivist.about import __version__ as VERSION +from archivist.constants import ( + HEADERS_RETRY_AFTER, + HEADERS_TOTAL_COUNT, + USER_AGENT, + USER_AGENT_PREFIX, +) from archivist.errors import ( ArchivistBadFieldError, ArchivistBadRequestError, @@ -56,6 +62,7 @@ def test_list(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -112,7 +119,10 @@ def test_list_with_headers(self): self.arch.list( "path/path", "things", - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) ) self.assertEqual( @@ -129,6 +139,7 @@ def test_list_with_headers(self): "headers": { "authorization": "Bearer authauthauth", "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -169,6 +180,7 @@ def test_list_with_params(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"paramsfield1": "paramsvalue1"}, }, @@ -213,6 +225,7 @@ def test_list_with_page_size(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 2}, }, @@ -279,6 +292,7 @@ def test_list_with_multiple_pages(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": paging[i], }, @@ -350,6 +364,7 @@ def test_list_with_multiple_pages_and_params(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": paging[i], }, @@ -431,6 +446,7 @@ def test_list_with_429_retry_and_success(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, diff --git a/unittests/testarchivistpost.py b/unittests/testarchivistpost.py index 579f101a..a78a14b4 100644 --- a/unittests/testarchivistpost.py +++ b/unittests/testarchivistpost.py @@ -5,8 +5,13 @@ from io import BytesIO from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist -from archivist.constants import HEADERS_RETRY_AFTER +from archivist.constants import ( + HEADERS_RETRY_AFTER, + USER_AGENT, + USER_AGENT_PREFIX, +) from archivist.errors import ( ArchivistBadRequestError, ArchivistTooManyRequestsError, @@ -56,6 +61,7 @@ def test_post(self): "json": request, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="POST method kwargs called incorrectly", @@ -133,6 +139,7 @@ def test_post_with_429_retry_and_success(self): "json": request, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="POST method kwargs called incorrectly", @@ -148,7 +155,10 @@ def test_post_with_headers(self): self.arch.post( "path/path", request, - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) args, kwargs = mock_post.call_args self.assertEqual( @@ -163,6 +173,7 @@ def test_post_with_headers(self): "headers": { "authorization": "Bearer authauthauth", "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="POST method kwargs called incorrectly", @@ -453,7 +464,9 @@ def test_post_without_auth(self): kwargs, { "json": request, - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, }, msg="POST method kwargs called incorrectly", ) diff --git a/unittests/testarchivistsignature.py b/unittests/testarchivistsignature.py index dddccf69..3d1ad1de 100644 --- a/unittests/testarchivistsignature.py +++ b/unittests/testarchivistsignature.py @@ -5,6 +5,11 @@ from os import environ from unittest import mock +from archivist.about import __version__ as VERSION +from archivist.constants import ( + USER_AGENT, + USER_AGENT_PREFIX, +) from archivist.errors import ( ArchivistBadFieldError, ArchivistDuplicateError, @@ -50,6 +55,7 @@ def test_get_by_signature(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"field1": "value1", "page_size": 2}, }, diff --git a/unittests/testassetattachments.py b/unittests/testassetattachments.py index 0d6d02a6..784c5484 100644 --- a/unittests/testassetattachments.py +++ b/unittests/testassetattachments.py @@ -5,6 +5,7 @@ from io import BytesIO from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.constants import ( ASSETATTACHMENTS_LABEL, @@ -12,6 +13,8 @@ ASSETS_LABEL, ATTACHMENTS_LABEL, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from .mock_response import MockResponse @@ -134,6 +137,7 @@ def filedata(chunk_size=4096): # pylint: disable=unused-argument { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": expected_params, "stream": True, @@ -203,6 +207,7 @@ def test_assetattachments_info(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, diff --git a/unittests/testassets.py b/unittests/testassets.py index 29308c9c..1db523f8 100644 --- a/unittests/testassets.py +++ b/unittests/testassets.py @@ -2,15 +2,18 @@ Test assets """ -from copy import copy +from copy import copy, deepcopy from logging import getLogger from os import environ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ASSETS_LABEL, ASSETS_SUBPATH, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistNotFoundError, ArchivistUnconfirmedError from archivist.logger import set_logger @@ -93,9 +96,11 @@ def test_assets_create(self): (f"url/{ROOT}/{ASSETS_SUBPATH}/{ASSETS_LABEL}",), msg="CREATE method args called incorrectly", ) + req_kwargs = deepcopy(REQUEST_KWARGS) + req_kwargs["headers"][USER_AGENT] = f"{USER_AGENT_PREFIX}{VERSION}" self.assertEqual( kwargs, - REQUEST_KWARGS, + req_kwargs, msg="CREATE method kwargs called incorrectly", ) self.assertEqual( @@ -130,9 +135,11 @@ def test_assets_create_merkle_log(self): (f"url/{ROOT}/{ASSETS_SUBPATH}/{ASSETS_LABEL}",), msg="CREATE method args called incorrectly", ) + req_kwargs = deepcopy(REQUEST_KWARGS_MERKLE_LOG) + req_kwargs["headers"][USER_AGENT] = f"{USER_AGENT_PREFIX}{VERSION}" self.assertEqual( kwargs, - REQUEST_KWARGS_MERKLE_LOG, + req_kwargs, msg="CREATE method kwargs called incorrectly", ) self.assertEqual( @@ -166,9 +173,11 @@ def test_assets_create_with_fixtures(self): (f"url/{ROOT}/{ASSETS_SUBPATH}/{ASSETS_LABEL}",), msg="CREATE method args called incorrectly", ) + req_kwargs = deepcopy(REQUEST_FIXTURES_KWARGS) + req_kwargs["headers"][USER_AGENT] = f"{USER_AGENT_PREFIX}{VERSION}" self.assertEqual( kwargs, - REQUEST_FIXTURES_KWARGS, + req_kwargs, msg="CREATE method kwargs called incorrectly", ) self.assertEqual( @@ -342,6 +351,7 @@ def test_assets_create_if_not_exists_existing_asset(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "attributes.arc_namespace": "namespace", @@ -399,6 +409,7 @@ def common_assets_create_if_not_exists_nonexistent_asset( (f"url/{ROOT}/{ASSETS_SUBPATH}/{ASSETS_LABEL}",), msg="CREATE method args called incorrectly", ) + req_kwargs["headers"][USER_AGENT] = f"{USER_AGENT_PREFIX}{VERSION}" self.assertEqual( kwargs, req_kwargs, diff --git a/unittests/testassetslist.py b/unittests/testassetslist.py index f7d0ec7a..be286e83 100644 --- a/unittests/testassetslist.py +++ b/unittests/testassetslist.py @@ -6,8 +6,11 @@ from os import environ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.logger import set_logger @@ -66,6 +69,7 @@ def test_assets_list(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {}, }, @@ -113,6 +117,7 @@ def test_assets_list_with_params(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "confirmation_status": "CONFIRMED", @@ -149,6 +154,7 @@ def test_assets_read_by_signature(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 2}, }, diff --git a/unittests/testassetsread.py b/unittests/testassetsread.py index 5860c0c6..09f2ea03 100644 --- a/unittests/testassetsread.py +++ b/unittests/testassetsread.py @@ -6,10 +6,13 @@ from os import environ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( HEADERS_REQUEST_TOTAL_COUNT, HEADERS_TOTAL_COUNT, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistBadFieldError from archivist.logger import set_logger @@ -125,6 +128,7 @@ def test_assets_count(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 1}, }, @@ -158,6 +162,7 @@ def test_assets_count_with_props_params(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 1, @@ -192,6 +197,7 @@ def test_assets_count_with_attrs_params(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 1, diff --git a/unittests/testassetswait.py b/unittests/testassetswait.py index 84d6992f..543df57c 100644 --- a/unittests/testassetswait.py +++ b/unittests/testassetswait.py @@ -6,10 +6,13 @@ from os import environ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( HEADERS_REQUEST_TOTAL_COUNT, HEADERS_TOTAL_COUNT, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistNotFoundError, ArchivistUnconfirmedError from archivist.logger import set_logger @@ -91,6 +94,7 @@ def test_assets_wait_for_confirmed(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": status[i], }, diff --git a/unittests/testattachments.py b/unittests/testattachments.py index a74082a9..52ee9f81 100644 --- a/unittests/testattachments.py +++ b/unittests/testattachments.py @@ -5,11 +5,14 @@ from io import BytesIO from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.constants import ( ATTACHMENTS_LABEL, ATTACHMENTS_SUBPATH, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from .mock_response import MockResponse @@ -301,6 +304,7 @@ def filedata(chunk_size=4096): # pylint: disable=unused-argument { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": expected_params, "stream": True, @@ -370,6 +374,7 @@ def test_attachments_info(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, diff --git a/unittests/testcompliance.py b/unittests/testcompliance.py index a3d9becc..ffcd9d52 100644 --- a/unittests/testcompliance.py +++ b/unittests/testcompliance.py @@ -6,12 +6,15 @@ from os import environ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.constants import ( COMPLIANCE_LABEL, COMPLIANCE_POLICIES_LABEL, COMPLIANCE_SUBPATH, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.logger import set_logger @@ -131,6 +134,7 @@ def test_compliance(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, diff --git a/unittests/testcompliance_policies.py b/unittests/testcompliance_policies.py index 65ced53c..7a08c9b5 100644 --- a/unittests/testcompliance_policies.py +++ b/unittests/testcompliance_policies.py @@ -4,6 +4,7 @@ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.compliance_policies import ( CompliancePolicy, @@ -17,6 +18,8 @@ HEADERS_REQUEST_TOTAL_COUNT, HEADERS_TOTAL_COUNT, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistBadRequestError @@ -123,6 +126,7 @@ def test_compliance_policies_create(self): "json": SINCE_REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -148,6 +152,7 @@ def test_compliance_policies_read(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -170,6 +175,7 @@ def test_compliance_policies_delete(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, ), @@ -207,6 +213,7 @@ def test_compliance_policies_count(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 1}, }, @@ -243,6 +250,7 @@ def test_compliance_policies_count_by_name(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 1, "compliance_type": "SINCE"}, }, @@ -283,6 +291,7 @@ def test_compliance_policies_list(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {}, }, @@ -327,6 +336,7 @@ def test_compliance_policies_list_by_name(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"compliance_type": "SINCE"}, }, @@ -360,6 +370,7 @@ def test_compliance_policies_read_by_signature(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 2}, }, diff --git a/unittests/testevents.py b/unittests/testevents.py index bbfd42ca..66bfd373 100644 --- a/unittests/testevents.py +++ b/unittests/testevents.py @@ -4,11 +4,14 @@ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ASSETS_LABEL, ASSETS_SUBPATH, EVENTS_LABEL, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.events import Event @@ -111,6 +114,7 @@ def test_events_read(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, diff --git a/unittests/testeventscount.py b/unittests/testeventscount.py index 632f1c78..019873f1 100644 --- a/unittests/testeventscount.py +++ b/unittests/testeventscount.py @@ -4,6 +4,7 @@ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ASSETS_LABEL, ASSETS_SUBPATH, @@ -12,6 +13,8 @@ HEADERS_REQUEST_TOTAL_COUNT, HEADERS_TOTAL_COUNT, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from .mock_response import MockResponse @@ -65,6 +68,7 @@ def test_events_count(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 1}, }, @@ -105,6 +109,7 @@ def test_events_count_with_props_params(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 1, "confirmation_status": "CONFIRMED"}, }, @@ -143,6 +148,7 @@ def test_events_count_with_attrs_params(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 1, @@ -183,6 +189,7 @@ def test_events_count_with_wildcard_asset(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 1, diff --git a/unittests/testeventscreate.py b/unittests/testeventscreate.py index 839958b7..b59719e8 100644 --- a/unittests/testeventscreate.py +++ b/unittests/testeventscreate.py @@ -4,11 +4,14 @@ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ASSETS_LABEL, ASSETS_SUBPATH, EVENTS_LABEL, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ( ArchivistUnconfirmedError, @@ -90,6 +93,7 @@ def test_events_create(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -133,6 +137,7 @@ def test_events_create_with_upload_attachments(self): "json": REQUEST_WITH_ATTACHMENTS, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -179,6 +184,7 @@ def test_events_create_with_upload_sbom_as_attachment(self): "json": REQUEST_WITH_SBOMATTACHMENT, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -222,6 +228,7 @@ def test_events_create_with_location(self): "json": REQUEST_WITH_LOCATION, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -262,6 +269,7 @@ def test_events_create_with_location_identity(self): "json": REQUEST_WITH_LOCATION, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -304,6 +312,7 @@ def test_events_create_with_asset_attrs(self): "json": REQUEST_WITH_ASSET_ATTRS, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", diff --git a/unittests/testeventslist.py b/unittests/testeventslist.py index 2ed91667..20e0ac2c 100644 --- a/unittests/testeventslist.py +++ b/unittests/testeventslist.py @@ -4,12 +4,15 @@ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ASSETS_LABEL, ASSETS_SUBPATH, ASSETS_WILDCARD, EVENTS_LABEL, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from .mock_response import MockResponse @@ -69,6 +72,7 @@ def test_events_list(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {}, }, @@ -123,6 +127,7 @@ def test_events_list_with_params(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "confirmation_status": "CONFIRMED", @@ -179,6 +184,7 @@ def test_events_list_with_wildcard_asset(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "confirmation_status": "CONFIRMED", @@ -221,6 +227,7 @@ def test_events_read_by_signature(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 2}, }, diff --git a/unittests/testeventsread.py b/unittests/testeventsread.py index ad524b06..b788b1b8 100644 --- a/unittests/testeventsread.py +++ b/unittests/testeventsread.py @@ -4,11 +4,14 @@ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ASSETS_LABEL, ASSETS_SUBPATH, EVENTS_LABEL, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistBadFieldError @@ -54,6 +57,7 @@ def test_events_read(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, diff --git a/unittests/testeventswait.py b/unittests/testeventswait.py index 5fc613dc..793d2f82 100644 --- a/unittests/testeventswait.py +++ b/unittests/testeventswait.py @@ -4,6 +4,7 @@ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ASSETS_SUBPATH, ASSETS_WILDCARD, @@ -11,6 +12,8 @@ HEADERS_REQUEST_TOTAL_COUNT, HEADERS_TOTAL_COUNT, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ( ArchivistNotFoundError, @@ -86,6 +89,7 @@ def test_events_wait_for_confirmed(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": status[i], }, diff --git a/unittests/testlocations.py b/unittests/testlocations.py index ec302195..86fa0ae0 100644 --- a/unittests/testlocations.py +++ b/unittests/testlocations.py @@ -4,6 +4,7 @@ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.constants import ( HEADERS_REQUEST_TOTAL_COUNT, @@ -11,6 +12,8 @@ LOCATIONS_LABEL, LOCATIONS_SUBPATH, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistBadRequestError, ArchivistNotFoundError from archivist.locations import Location @@ -188,6 +191,7 @@ def test_locations_create(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -234,6 +238,7 @@ def test_locations_create_if_not_exists_existing_location(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 2, @@ -281,6 +286,7 @@ def test_locations_create_if_not_exists_nonexistent_location(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -324,6 +330,7 @@ def test_locations_create_if_not_exists_nonexistent_location_selector_noattribut "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -344,6 +351,7 @@ def test_locations_read(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -382,6 +390,7 @@ def test_locations_count(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 1}, }, @@ -418,6 +427,7 @@ def test_locations_count_with_props_params(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 1, @@ -452,6 +462,7 @@ def test_locations_count_with_attrs_params(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 1, @@ -495,6 +506,7 @@ def test_locations_list(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {}, }, @@ -540,6 +552,7 @@ def test_locations_list_with_params(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "attributes.director": "John Smith", @@ -576,6 +589,7 @@ def test_locations_read_by_signature(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 2}, }, diff --git a/unittests/testpublicassetattachments.py b/unittests/testpublicassetattachments.py index 646dd8b3..dd97725c 100644 --- a/unittests/testpublicassetattachments.py +++ b/unittests/testpublicassetattachments.py @@ -7,6 +7,7 @@ from os import environ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivistpublic import ArchivistPublic from archivist.constants import ( ASSETATTACHMENTS_LABEL, @@ -14,6 +15,8 @@ ASSETS_LABEL, ATTACHMENTS_LABEL, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.logger import set_logger @@ -142,7 +145,9 @@ def filedata(chunk_size=4096): # pylint: disable=unused-argument self.assertEqual( kwargs, { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": expected_params, "stream": True, }, @@ -209,7 +214,9 @@ def test_assetattachments_info(self): self.assertEqual( kwargs, { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": None, }, msg="INFO method called incorrectly", diff --git a/unittests/testpublicassetsread.py b/unittests/testpublicassetsread.py index 4b3c37a1..6cd9fa9e 100644 --- a/unittests/testpublicassetsread.py +++ b/unittests/testpublicassetsread.py @@ -6,8 +6,11 @@ from os import environ from unittest import mock +from archivist.about import __version__ as VERSION from archivist.constants import ( ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.logger import set_logger @@ -64,7 +67,9 @@ def test_publicassets_read(self): self.assertEqual( kwargs, { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": None, }, msg="GET method kwargs called incorrectly", diff --git a/unittests/testpublicevents.py b/unittests/testpublicevents.py index 1f1dfcf4..25536d75 100644 --- a/unittests/testpublicevents.py +++ b/unittests/testpublicevents.py @@ -6,6 +6,7 @@ from os import environ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivistpublic import ArchivistPublic from archivist.constants import ( EVENTS_LABEL, @@ -13,6 +14,8 @@ HEADERS_TOTAL_COUNT, PUBLICASSETS_LABEL, ROOT, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.logger import set_logger @@ -82,7 +85,9 @@ def test_publicevents_read(self): ), ), { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": None, }, ), @@ -113,7 +118,9 @@ def test_publicevents_read_no_subpath(self): ), ), { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": None, }, ), @@ -161,6 +168,7 @@ def test_publicevents_count(self): { "headers": { HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "asset_attributes.external_container": "assets/xxxx", @@ -208,6 +216,7 @@ def test_publicevents_count_no_subpath(self): { "headers": { HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "asset_attributes.external_container": "assets/xxxx", @@ -256,7 +265,9 @@ def test_publicevents_list(self): ), ), { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": {}, }, ), @@ -300,7 +311,9 @@ def test_publicevents_list_no_subpath(self): ), ), { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": {}, }, ), @@ -337,7 +350,9 @@ def test_publicevents_read_by_signature(self): ), ), { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": {"page_size": 2}, }, ), diff --git a/unittests/testpublicget.py b/unittests/testpublicget.py index ad01052d..e462a254 100644 --- a/unittests/testpublicget.py +++ b/unittests/testpublicget.py @@ -5,8 +5,13 @@ from io import BytesIO from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivistpublic import ArchivistPublic -from archivist.constants import HEADERS_RETRY_AFTER +from archivist.constants import ( + HEADERS_RETRY_AFTER, + USER_AGENT, + USER_AGENT_PREFIX, +) from archivist.errors import ( ArchivistNotFoundError, ArchivistTooManyRequestsError, @@ -48,7 +53,7 @@ def test_get(self): ( ("https://path/path/entity/xxxxxxxx",), { - "headers": {}, + "headers": {USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}"}, "params": None, }, ), @@ -82,7 +87,10 @@ def test_get_with_headers(self): mock_get.return_value = MockResponse(200) self.public.get( "https://path/path/id/xxxxxxxx", - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) self.assertEqual( tuple(mock_get.call_args), @@ -91,6 +99,7 @@ def test_get_with_headers(self): { "headers": { "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -107,7 +116,10 @@ def test_get_with_429(self): with self.assertRaises(ArchivistTooManyRequestsError): self.public.get( "https://path/path/id/xxxxxxxx", - headers={"headerfield1": "headervalue1"}, + headers={ + "headerfield1": "headervalue1", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, ) def test_get_with_429_retry_and_fail(self): @@ -152,7 +164,9 @@ def test_get_with_429_retry_and_success(self): ( ("https://path/path/entity/xxxxxxxx",), { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "params": None, }, ), @@ -199,7 +213,9 @@ def filedata(chunk_size=4096): # pylint: disable=unused-argument ( ("https://path/path/entity/xxxxxxxx",), { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "stream": True, "params": None, }, @@ -289,7 +305,9 @@ def filedata(chunk_size=4096): # pylint: disable=unused-argument ( ("path/path/entity/xxxxxxxx",), { - "headers": {}, + "headers": { + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", + }, "stream": True, "params": None, }, diff --git a/unittests/testsubjects.py b/unittests/testsubjects.py index b546013e..38e66a3b 100644 --- a/unittests/testsubjects.py +++ b/unittests/testsubjects.py @@ -5,6 +5,7 @@ from os import environ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.constants import ( HEADERS_REQUEST_TOTAL_COUNT, @@ -12,6 +13,8 @@ ROOT, SUBJECTS_LABEL, SUBJECTS_SUBPATH, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.errors import ArchivistBadRequestError, ArchivistUnconfirmedError from archivist.logger import set_logger @@ -138,6 +141,7 @@ def test_subjects_create(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -181,6 +185,7 @@ def test_subjects_share(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -202,6 +207,7 @@ def test_subjects_share(self): "json": REQUEST2, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -232,6 +238,7 @@ def test_subjects_import_subject(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -267,6 +274,7 @@ def test_subjects_create_from_b64(self): "json": REQUEST, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="CREATE method kwargs called incorrectly", @@ -318,6 +326,7 @@ def test_subjects_read(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": None, }, @@ -340,6 +349,7 @@ def test_subjects_delete(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, ), @@ -369,6 +379,7 @@ def test_subjects_update(self): "json": UPDATE, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="PATCH method kwargs called incorrectly", @@ -405,6 +416,7 @@ def test_subjects_count(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"page_size": 1}, }, @@ -441,6 +453,7 @@ def test_subjects_count_by_name(self): "headers": { "authorization": "Bearer authauthauth", HEADERS_REQUEST_TOTAL_COUNT: "true", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": { "page_size": 1, @@ -484,6 +497,7 @@ def test_subjects_list(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {}, }, @@ -528,6 +542,7 @@ def test_subjects_list_by_name(self): { "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, "params": {"display_name": "Subject display name"}, }, diff --git a/unittests/testtenancies.py b/unittests/testtenancies.py index a04510dc..e7e5cfad 100644 --- a/unittests/testtenancies.py +++ b/unittests/testtenancies.py @@ -6,12 +6,15 @@ from os import environ from unittest import TestCase, mock +from archivist.about import __version__ as VERSION from archivist.archivist import Archivist from archivist.constants import ( ROOT, TENANCIES_LABEL, TENANCIES_PREFIX, TENANCIES_SUBPATH, + USER_AGENT, + USER_AGENT_PREFIX, ) from archivist.logger import set_logger @@ -84,6 +87,7 @@ def test_tenancies_publicinfo(self): "params": None, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="GET method kwargs called incorrectly", @@ -114,6 +118,7 @@ def test_tenancies_publicinfo_response_identity(self): "params": None, "headers": { "authorization": "Bearer authauthauth", + USER_AGENT: f"{USER_AGENT_PREFIX}{VERSION}", }, }, msg="GET method kwargs called incorrectly",