port-ocean 0.24.19__py3-none-any.whl → 0.24.21__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -228,9 +228,11 @@ class IntegrationClientMixin:
228
228
  async def put_integration_sync_metrics(self, kind_metrics: dict[str, Any]) -> None:
229
229
  logger.debug("starting PUT metrics request", kind_metrics=kind_metrics)
230
230
  metrics_attributes = await self.get_metrics_attributes()
231
+ event_id = quote_plus(kind_metrics["eventId"])
232
+ kind_identifier = quote_plus(kind_metrics["kindIdentifier"])
231
233
  url = (
232
234
  metrics_attributes["ingestUrl"]
233
- + f"/syncMetrics/resync/{kind_metrics['eventId']}/kind/{kind_metrics['kindIdentifier']}"
235
+ + f"/syncMetrics/resync/{event_id}/kind/{kind_identifier}"
234
236
  )
235
237
  headers = await self.auth.headers()
236
238
  response = await self.client.put(
@@ -617,22 +617,23 @@ class SyncRawMixin(HandlerMixin, EventsMixin):
617
617
  logger.info(f"Process finished for {resource.kind} with index {index}")
618
618
 
619
619
  async def process_resource(self, resource: ResourceConfig, index: int, user_agent_type: UserAgentType) -> tuple[list[Entity], list[Exception]]:
620
- if ocean.app.process_execution_mode == ProcessExecutionMode.multi_process:
621
- id = uuid.uuid4()
622
- logger.info(f"Starting subprocess with id {id}")
623
- file_ipc_map = {
624
- "process_resource": FileIPC(id, "process_resource",([],[IntegrationSubProcessFailedException(f"Subprocess failed for {resource.kind} with index {index}")])),
625
- "topological_entities": FileIPC(id, "topological_entities",[]),
626
- }
627
- process = ProcessWrapper(target=self.process_resource_in_subprocess, args=(file_ipc_map,resource,index,user_agent_type))
628
- process.start()
629
- await process.join_async()
630
-
631
- event.entity_topological_sorter.entities.extend(file_ipc_map["topological_entities"].load())
632
- return file_ipc_map["process_resource"].load()
620
+ with logger.contextualize(resource_kind=resource.kind, index=index):
621
+ if ocean.app.process_execution_mode == ProcessExecutionMode.multi_process:
622
+ id = uuid.uuid4()
623
+ logger.info(f"Starting subprocess with id {id}")
624
+ file_ipc_map = {
625
+ "process_resource": FileIPC(id, "process_resource",([],[IntegrationSubProcessFailedException(f"Subprocess failed for {resource.kind} with index {index}")])),
626
+ "topological_entities": FileIPC(id, "topological_entities",[]),
627
+ }
628
+ process = ProcessWrapper(target=self.process_resource_in_subprocess, args=(file_ipc_map,resource,index,user_agent_type))
629
+ process.start()
630
+ await process.join_async()
631
+
632
+ event.entity_topological_sorter.entities.extend(file_ipc_map["topological_entities"].load())
633
+ return file_ipc_map["process_resource"].load()
633
634
 
634
- else:
635
- return await self._process_resource(resource,index,user_agent_type)
635
+ else:
636
+ return await self._process_resource(resource,index,user_agent_type)
636
637
 
637
638
  async def _process_resource(self,resource: ResourceConfig, index: int, user_agent_type: UserAgentType)-> tuple[list[Entity], list[Exception]]:
638
639
  # create resource context per resource kind, so resync method could have access to the resource
@@ -0,0 +1,205 @@
1
+ from typing import Any
2
+ from unittest.mock import MagicMock, patch, AsyncMock
3
+
4
+ import pytest
5
+ import httpx
6
+
7
+ from port_ocean.clients.port.mixins.integrations import IntegrationClientMixin
8
+
9
+
10
+ TEST_INTEGRATION_IDENTIFIER = "test-integration"
11
+ TEST_INTEGRATION_VERSION = "1.0.0"
12
+ TEST_INGEST_URL = "https://api.example.com"
13
+
14
+ BASIC_KIND_METRICS = {
15
+ "eventId": "event-123",
16
+ "kindIdentifier": "service",
17
+ "metrics": {"count": 5},
18
+ }
19
+
20
+ KIND_METRICS_WITH_SLASH = {
21
+ "eventId": "event-456",
22
+ "kindIdentifier": "kind/kind1",
23
+ "metrics": {"count": 10},
24
+ }
25
+
26
+ EVENT_METRICS_WITH_SLASH = {
27
+ "eventId": "event/123",
28
+ "kindIdentifier": "service",
29
+ "metrics": {"count": 15},
30
+ }
31
+
32
+ BOTH_METRICS_WITH_SLASH = {
33
+ "eventId": "namespace/event/123",
34
+ "kindIdentifier": "app/service/v1",
35
+ "metrics": {"count": 20},
36
+ }
37
+
38
+ COMPLEX_KIND_METRICS = {
39
+ "eventId": "complete/test/123",
40
+ "kindIdentifier": "complex/kind/identifier",
41
+ "syncStart": "2024-01-01T00:00:00Z",
42
+ "syncEnd": "2024-01-01T01:00:00Z",
43
+ "metrics": {"totalEntities": 100, "successfulEntities": 95, "failedEntities": 5},
44
+ }
45
+
46
+
47
+ @pytest.fixture
48
+ def integration_client(monkeypatch: Any) -> IntegrationClientMixin:
49
+ """Create an IntegrationClientMixin instance with mocked dependencies."""
50
+ auth = MagicMock()
51
+ auth.headers = AsyncMock()
52
+ auth.headers.return_value = {"Authorization": "Bearer test-token"}
53
+
54
+ client = MagicMock()
55
+ client.put = AsyncMock()
56
+ client.put.return_value = MagicMock()
57
+ client.put.return_value.status_code = 200
58
+ client.put.return_value.is_error = False
59
+
60
+ integration_client = IntegrationClientMixin(
61
+ integration_identifier=TEST_INTEGRATION_IDENTIFIER,
62
+ integration_version=TEST_INTEGRATION_VERSION,
63
+ auth=auth,
64
+ client=client,
65
+ )
66
+
67
+ mock_get_metrics_attributes = AsyncMock()
68
+ mock_get_metrics_attributes.return_value = {"ingestUrl": TEST_INGEST_URL}
69
+ monkeypatch.setattr(
70
+ integration_client, "get_metrics_attributes", mock_get_metrics_attributes
71
+ )
72
+
73
+ return integration_client
74
+
75
+
76
+ async def test_put_integration_sync_metrics_basic(
77
+ integration_client: IntegrationClientMixin,
78
+ ) -> None:
79
+ """Test basic functionality of put_integration_sync_metrics."""
80
+ with patch(
81
+ "port_ocean.clients.port.mixins.integrations.handle_port_status_code"
82
+ ) as mock_handle:
83
+ await integration_client.put_integration_sync_metrics(BASIC_KIND_METRICS)
84
+
85
+ integration_client.get_metrics_attributes.assert_called_once()
86
+
87
+ integration_client.auth.headers.assert_called_once()
88
+
89
+ integration_client.client.put.assert_called_once()
90
+ call_args = integration_client.client.put.call_args
91
+
92
+ expected_url = f"{TEST_INGEST_URL}/syncMetrics/resync/event-123/kind/service"
93
+ assert call_args[0][0] == expected_url
94
+
95
+ expected_headers = {"Authorization": "Bearer test-token"}
96
+ assert call_args[1]["headers"] == expected_headers
97
+
98
+ expected_json = {"syncKindMetrics": BASIC_KIND_METRICS}
99
+ assert call_args[1]["json"] == expected_json
100
+
101
+ mock_handle.assert_called_once_with(
102
+ integration_client.client.put.return_value, should_log=False
103
+ )
104
+
105
+
106
+ async def test_put_integration_sync_metrics_with_slash_in_kind_identifier(
107
+ integration_client: IntegrationClientMixin,
108
+ ) -> None:
109
+ """Test put_integration_sync_metrics with forward slash in kindIdentifier."""
110
+ with patch("port_ocean.clients.port.mixins.integrations.handle_port_status_code"):
111
+ await integration_client.put_integration_sync_metrics(KIND_METRICS_WITH_SLASH)
112
+
113
+ integration_client.client.put.assert_called_once()
114
+ call_args = integration_client.client.put.call_args
115
+
116
+ expected_url = (
117
+ f"{TEST_INGEST_URL}/syncMetrics/resync/event-456/kind/kind%2Fkind1"
118
+ )
119
+ assert call_args[0][0] == expected_url
120
+
121
+ expected_json = {"syncKindMetrics": KIND_METRICS_WITH_SLASH}
122
+ assert call_args[1]["json"] == expected_json
123
+
124
+
125
+ async def test_put_integration_sync_metrics_with_slash_in_event_id(
126
+ integration_client: IntegrationClientMixin,
127
+ ) -> None:
128
+ """Test put_integration_sync_metrics with forward slash in eventId."""
129
+ with patch("port_ocean.clients.port.mixins.integrations.handle_port_status_code"):
130
+ await integration_client.put_integration_sync_metrics(EVENT_METRICS_WITH_SLASH)
131
+
132
+ integration_client.client.put.assert_called_once()
133
+ call_args = integration_client.client.put.call_args
134
+
135
+ expected_url = f"{TEST_INGEST_URL}/syncMetrics/resync/event%2F123/kind/service"
136
+ assert call_args[0][0] == expected_url
137
+
138
+
139
+ async def test_put_integration_sync_metrics_with_slashes_in_both_fields(
140
+ integration_client: IntegrationClientMixin,
141
+ ) -> None:
142
+ """Test put_integration_sync_metrics with forward slashes in both eventId and kindIdentifier."""
143
+ with patch("port_ocean.clients.port.mixins.integrations.handle_port_status_code"):
144
+ await integration_client.put_integration_sync_metrics(BOTH_METRICS_WITH_SLASH)
145
+
146
+ integration_client.client.put.assert_called_once()
147
+ call_args = integration_client.client.put.call_args
148
+
149
+ expected_url = f"{TEST_INGEST_URL}/syncMetrics/resync/namespace%2Fevent%2F123/kind/app%2Fservice%2Fv1"
150
+ assert call_args[0][0] == expected_url
151
+
152
+
153
+ async def test_put_integration_sync_metrics_with_special_characters(
154
+ integration_client: IntegrationClientMixin,
155
+ ) -> None:
156
+ """Test put_integration_sync_metrics with various special characters that need URL encoding."""
157
+ special_metrics = {
158
+ "eventId": "event@123#test",
159
+ "kindIdentifier": "kind with spaces+symbols",
160
+ "metrics": {"count": 25},
161
+ }
162
+
163
+ with patch("port_ocean.clients.port.mixins.integrations.handle_port_status_code"):
164
+ await integration_client.put_integration_sync_metrics(special_metrics)
165
+
166
+ integration_client.client.put.assert_called_once()
167
+ call_args = integration_client.client.put.call_args
168
+
169
+ expected_url = f"{TEST_INGEST_URL}/syncMetrics/resync/event%40123%23test/kind/kind+with+spaces%2Bsymbols"
170
+ assert call_args[0][0] == expected_url
171
+
172
+
173
+ async def test_put_integration_sync_metrics_complete_flow(
174
+ integration_client: IntegrationClientMixin,
175
+ ) -> None:
176
+ """Test the complete flow of put_integration_sync_metrics method."""
177
+ with patch("port_ocean.clients.port.mixins.integrations.handle_port_status_code"):
178
+ await integration_client.put_integration_sync_metrics(COMPLEX_KIND_METRICS)
179
+
180
+ integration_client.get_metrics_attributes.assert_called_once()
181
+ integration_client.auth.headers.assert_called_once()
182
+ integration_client.client.put.assert_called_once()
183
+
184
+ call_args = integration_client.client.put.call_args
185
+ assert "complete%2Ftest%2F123" in call_args[0][0]
186
+ assert "complex%2Fkind%2Fidentifier" in call_args[0][0]
187
+
188
+ assert call_args[1]["json"]["syncKindMetrics"] == COMPLEX_KIND_METRICS
189
+
190
+
191
+ async def test_put_integration_sync_metrics_error_handling(
192
+ integration_client: IntegrationClientMixin,
193
+ monkeypatch: Any,
194
+ ) -> None:
195
+ """Test that put_integration_sync_metrics properly handles errors."""
196
+ mock_get_metrics_attributes = AsyncMock()
197
+ mock_get_metrics_attributes.side_effect = httpx.HTTPStatusError(
198
+ message="Test error", request=MagicMock(), response=MagicMock()
199
+ )
200
+ monkeypatch.setattr(
201
+ integration_client, "get_metrics_attributes", mock_get_metrics_attributes
202
+ )
203
+
204
+ with pytest.raises(httpx.HTTPStatusError):
205
+ await integration_client.put_integration_sync_metrics(BASIC_KIND_METRICS)
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: port-ocean
3
- Version: 0.24.19
3
+ Version: 0.24.21
4
4
  Summary: Port Ocean is a CLI tool for managing your Port projects.
5
5
  Home-page: https://app.getport.io
6
6
  Keywords: ocean,port-ocean,port
@@ -60,7 +60,7 @@ port_ocean/clients/port/client.py,sha256=dv0mxIOde6J-wFi1FXXZkoNPVHrZzY7RSMhNkDD
60
60
  port_ocean/clients/port/mixins/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
61
61
  port_ocean/clients/port/mixins/blueprints.py,sha256=aMCG4zePsMSMjMLiGrU37h5z5_ElfMzTcTvqvOI5wXY,4683
62
62
  port_ocean/clients/port/mixins/entities.py,sha256=_FKoSEUxc_fGcdoWm4ZAeUKUnnzkGPMDQEsxmpbj8Vo,23352
63
- port_ocean/clients/port/mixins/integrations.py,sha256=s6paomK9bYWW-Tu3y2OIaEGSxsXCHyhapVi4JIhhO64,11162
63
+ port_ocean/clients/port/mixins/integrations.py,sha256=DdMLxSHL2zmlLecnkJk1lWJ2fxkY89pSSL-m7H0zBuI,11256
64
64
  port_ocean/clients/port/mixins/migrations.py,sha256=vdL_A_NNUogvzujyaRLIoZEu5vmKDY2BxTjoGP94YzI,1467
65
65
  port_ocean/clients/port/mixins/organization.py,sha256=A2cP5V49KnjoAXxjmnm_XGth4ftPSU0qURNfnyUyS_Y,1041
66
66
  port_ocean/clients/port/retry_transport.py,sha256=PtIZOAZ6V-ncpVysRUsPOgt8Sf01QLnTKB5YeKBxkJk,1861
@@ -121,7 +121,7 @@ port_ocean/core/integrations/mixins/events.py,sha256=2L7P3Jhp8XBqddh2_o9Cn4N261n
121
121
  port_ocean/core/integrations/mixins/handler.py,sha256=mZ7-0UlG3LcrwJttFbMe-R4xcOU2H_g33tZar7PwTv8,3771
122
122
  port_ocean/core/integrations/mixins/live_events.py,sha256=zM24dhNc7uHx9XYZ6toVhDADPA90EnpOmZxgDegFZbA,4196
123
123
  port_ocean/core/integrations/mixins/sync.py,sha256=Vm_898pLKBwfVewtwouDWsXoxcOLicnAy6pzyqqk6U8,4053
124
- port_ocean/core/integrations/mixins/sync_raw.py,sha256=NTclUHBYH_RK895zM3VGfd3MyPRYvfqnWudXYN5RFqQ,33885
124
+ port_ocean/core/integrations/mixins/sync_raw.py,sha256=mbFRZzG91McaoeLP7rR9TP3Bb1N5zayHwF3t71u-Ncc,34022
125
125
  port_ocean/core/integrations/mixins/utils.py,sha256=N76dNu1Y6UEg0_pcSdF4RO6dQIZ8EBfX3xMelgWdMxM,3779
126
126
  port_ocean/core/models.py,sha256=DNbKpStMINI2lIekKprTqBevqkw_wFuFayN19w1aDfQ,2893
127
127
  port_ocean/core/ocean_types.py,sha256=4VipWFOHEh_d9LmWewQccwx1p2dtrRYW0YURVgNsAjo,1398
@@ -159,6 +159,7 @@ port_ocean/tests/clients/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZ
159
159
  port_ocean/tests/clients/oauth/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
160
160
  port_ocean/tests/clients/oauth/test_oauth_client.py,sha256=2XVMQUalDpiD539Z7_dk5BK_ngXQzsTmb2lNBsfEm9c,3266
161
161
  port_ocean/tests/clients/port/mixins/test_entities.py,sha256=DeWbAQcaxT3LQQf_j9HA5nG7YgsQDvXmgK2aghlG9ug,6619
162
+ port_ocean/tests/clients/port/mixins/test_integrations.py,sha256=vRt_EMsLozQC1LJNXxlvnHj3-FlOBGgAYxg5T0IAqtA,7621
162
163
  port_ocean/tests/clients/port/mixins/test_organization_mixin.py,sha256=zzKYz3h8dl4Z5A2QG_924m0y9U6XTth1XYOfwNrd_24,914
163
164
  port_ocean/tests/conftest.py,sha256=JXASSS0IY0nnR6bxBflhzxS25kf4iNaABmThyZ0mZt8,101
164
165
  port_ocean/tests/core/conftest.py,sha256=7K_M1--wQ08VmiQRB0vo1nst2X00cwsuBS5UfERsnG8,7589
@@ -200,8 +201,8 @@ port_ocean/utils/repeat.py,sha256=U2OeCkHPWXmRTVoPV-VcJRlQhcYqPWI5NfmPlb1JIbc,32
200
201
  port_ocean/utils/signal.py,sha256=mMVq-1Ab5YpNiqN4PkiyTGlV_G0wkUDMMjTZp5z3pb0,1514
201
202
  port_ocean/utils/time.py,sha256=pufAOH5ZQI7gXvOvJoQXZXZJV-Dqktoj9Qp9eiRwmJ4,1939
202
203
  port_ocean/version.py,sha256=UsuJdvdQlazzKGD3Hd5-U7N69STh8Dq9ggJzQFnu9fU,177
203
- port_ocean-0.24.19.dist-info/LICENSE.md,sha256=WNHhf_5RCaeuKWyq_K39vmp9F28LxKsB4SpomwSZ2L0,11357
204
- port_ocean-0.24.19.dist-info/METADATA,sha256=HE9-dswSXnOaFu4p3hJkDWy6syqUIADmnYhVSN2F4Vg,6856
205
- port_ocean-0.24.19.dist-info/WHEEL,sha256=Nq82e9rUAnEjt98J6MlVmMCZb-t9cYE2Ir1kpBmnWfs,88
206
- port_ocean-0.24.19.dist-info/entry_points.txt,sha256=F_DNUmGZU2Kme-8NsWM5LLE8piGMafYZygRYhOVtcjA,54
207
- port_ocean-0.24.19.dist-info/RECORD,,
204
+ port_ocean-0.24.21.dist-info/LICENSE.md,sha256=WNHhf_5RCaeuKWyq_K39vmp9F28LxKsB4SpomwSZ2L0,11357
205
+ port_ocean-0.24.21.dist-info/METADATA,sha256=CMim2FITINDDdUII_ENejo3t677lzYlELYVfULtdfPs,6856
206
+ port_ocean-0.24.21.dist-info/WHEEL,sha256=Nq82e9rUAnEjt98J6MlVmMCZb-t9cYE2Ir1kpBmnWfs,88
207
+ port_ocean-0.24.21.dist-info/entry_points.txt,sha256=F_DNUmGZU2Kme-8NsWM5LLE8piGMafYZygRYhOVtcjA,54
208
+ port_ocean-0.24.21.dist-info/RECORD,,