port-ocean 0.30.0__py3-none-any.whl → 0.30.1__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.
@@ -1,5 +1,6 @@
1
1
  import asyncio
2
2
  import json
3
+ from collections import Counter
3
4
  from typing import Any, Literal
4
5
  from urllib.parse import quote_plus
5
6
 
@@ -355,6 +356,19 @@ class EntityClientMixin:
355
356
  entities_results: list[tuple[bool, Entity]] = []
356
357
  blueprint = entities[0].blueprint
357
358
 
359
+ identifier_counts = Counter((e.blueprint, e.identifier) for e in entities)
360
+ duplicate_count = sum(
361
+ count - 1 for count in identifier_counts.values() if count > 1
362
+ )
363
+
364
+ if duplicate_count:
365
+ duplicate_examples = [
366
+ key for key, cnt in identifier_counts.items() if cnt > 1
367
+ ][:5]
368
+ logger.warning(
369
+ f"Detected {duplicate_count} duplicate entities (by blueprint and identifier) that may not be ingested because an identical identifier existed. Examples: {duplicate_examples}"
370
+ )
371
+
358
372
  bulk_size = self.calculate_entities_batch_size(entities)
359
373
  bulks = [
360
374
  entities[i : i + bulk_size] for i in range(0, len(entities), bulk_size)
@@ -1,19 +1,27 @@
1
1
  import asyncio
2
+ import json
3
+ import re
2
4
  from asyncio import Task
3
5
  from dataclasses import dataclass, field
4
6
  from functools import lru_cache
5
- import json
6
7
  from typing import Any, Optional
8
+
7
9
  import jq # type: ignore
8
10
  from loguru import logger
11
+
9
12
  from port_ocean.context.ocean import ocean
10
13
  from port_ocean.core.handlers.entity_processor.base import BaseEntityProcessor
14
+ from port_ocean.core.handlers.entity_processor.jq_input_evaluator import (
15
+ InputClassifyingResult,
16
+ can_expression_run_with_no_input,
17
+ classify_input,
18
+ )
11
19
  from port_ocean.core.handlers.port_app_config.models import ResourceConfig
12
20
  from port_ocean.core.models import Entity
13
21
  from port_ocean.core.ocean_types import (
14
22
  RAW_ITEM,
15
- EntitySelectorDiff,
16
23
  CalculationResult,
24
+ EntitySelectorDiff,
17
25
  )
18
26
  from port_ocean.core.utils.utils import (
19
27
  gather_and_split_errors_from_results,
@@ -21,11 +29,6 @@ from port_ocean.core.utils.utils import (
21
29
  )
22
30
  from port_ocean.exceptions.core import EntityProcessorException
23
31
  from port_ocean.utils.queue_utils import process_in_queue
24
- from port_ocean.core.handlers.entity_processor.jq_input_evaluator import (
25
- InputClassifyingResult,
26
- classify_input,
27
- can_expression_run_with_no_input,
28
- )
29
32
 
30
33
 
31
34
  class ExampleStates:
@@ -92,8 +95,29 @@ class JQEntityProcessor(BaseEntityProcessor):
92
95
  searching for data in dictionaries, and transforming data based on object mappings.
93
96
  """
94
97
 
98
+ @staticmethod
99
+ def _format_filter(filter: str) -> str:
100
+ """
101
+ Convert single quotes to double quotes in JQ expressions.
102
+ Only replaces single quotes that are opening or closing string delimiters,
103
+ not single quotes that are part of string content.
104
+ """
105
+ # Escape single quotes only if they are opening or closing a string
106
+ # Pattern matches:
107
+ # - Single quote at start of string or after whitespace (opening quote)
108
+ # - Single quote before whitespace or end of string (closing quote)
109
+ # Uses negative lookahead/lookbehind to avoid replacing quotes inside strings
110
+ # \1 and \2 will be empty for the alternative that didn't match, so \1"\2 works for both cases
111
+ # This matches the TypeScript pattern: /(^|\s)'(?!\s|")|(?<!\s|")'(\s|$)/g
112
+ formatted_filter = re.sub(
113
+ r'(^|\s)\'(?!\s|")|(?<!\s|")\'(\s|$)', r'\1"\2', filter
114
+ )
115
+ return formatted_filter
116
+
95
117
  @lru_cache
96
118
  def _compile(self, pattern: str) -> Any:
119
+ # Convert single quotes to double quotes for JQ compatibility
120
+ pattern = self._format_filter(pattern)
97
121
  if not ocean.config.allow_environment_variables_jq_access:
98
122
  pattern = "def env: {}; {} as $ENV | " + pattern
99
123
  return jq.compile(pattern)
@@ -119,7 +143,6 @@ class JQEntityProcessor(BaseEntityProcessor):
119
143
  missing_required_fields: bool,
120
144
  entity_mapping_fault_counter: int,
121
145
  ) -> None:
122
-
123
146
  if len(entity_misconfigurations) > 0:
124
147
  logger.info(
125
148
  f"Unable to find valid data for: {entity_misconfigurations} (null, missing, or misconfigured)"
@@ -444,7 +467,6 @@ class JQEntityProcessor(BaseEntityProcessor):
444
467
  key: str,
445
468
  value: dict[str, Any],
446
469
  ) -> None:
447
-
448
470
  if key in ["properties", "relations"]:
449
471
  mapping_dicts: dict[InputClassifyingResult, dict[str, Any]] = {
450
472
  InputClassifyingResult.SINGLE: {},
@@ -1,8 +1,9 @@
1
- from typing import cast, Any
2
- from unittest.mock import AsyncMock, Mock
3
- from loguru import logger
4
- import pytest
5
1
  from io import StringIO
2
+ from typing import Any, cast
3
+ from unittest.mock import AsyncMock, Mock, patch
4
+
5
+ import pytest
6
+ from loguru import logger
6
7
 
7
8
  from port_ocean.context.ocean import PortOceanContext
8
9
  from port_ocean.core.handlers.entity_processor.jq_entity_processor import (
@@ -10,12 +11,10 @@ from port_ocean.core.handlers.entity_processor.jq_entity_processor import (
10
11
  )
11
12
  from port_ocean.core.ocean_types import CalculationResult
12
13
  from port_ocean.exceptions.core import EntityProcessorException
13
- from unittest.mock import patch
14
14
 
15
15
 
16
16
  @pytest.mark.asyncio
17
17
  class TestJQEntityProcessor:
18
-
19
18
  @pytest.fixture
20
19
  def mocked_processor(self, monkeypatch: Any) -> JQEntityProcessor:
21
20
  mock_context = AsyncMock()
@@ -33,6 +32,30 @@ class TestJQEntityProcessor:
33
32
  result = await mocked_processor._search(data, pattern)
34
33
  assert result == "bar"
35
34
 
35
+ async def test_search_with_single_quotes(
36
+ self, mocked_processor: JQEntityProcessor
37
+ ) -> None:
38
+ data = {"repository": "ocean", "organization": "port"}
39
+ pattern = ".organization + '/' + .repository"
40
+ result = await mocked_processor._search(data, pattern)
41
+ assert result == "port/ocean"
42
+
43
+ async def test_search_with_single_quotes_in_the_end(
44
+ self, mocked_processor: JQEntityProcessor
45
+ ) -> None:
46
+ data = {"organization": "port"}
47
+ pattern = ".organization + '/'"
48
+ result = await mocked_processor._search(data, pattern)
49
+ assert result == "port/"
50
+
51
+ async def test_search_with_single_quotes_in_the_start(
52
+ self, mocked_processor: JQEntityProcessor
53
+ ) -> None:
54
+ data = {"organization": "port"}
55
+ pattern = "'/' + .organization"
56
+ result = await mocked_processor._search(data, pattern)
57
+ assert result == "/port"
58
+
36
59
  async def test_search_as_bool(self, mocked_processor: JQEntityProcessor) -> None:
37
60
  data = {"foo": True}
38
61
  pattern = ".foo"
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: port-ocean
3
- Version: 0.30.0
3
+ Version: 0.30.1
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
@@ -61,7 +61,7 @@ port_ocean/clients/port/client.py,sha256=LHR6zKgCCCyhe3aPWH0kRFYS02BN-lIDZMPQbaz
61
61
  port_ocean/clients/port/mixins/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
62
62
  port_ocean/clients/port/mixins/actions.py,sha256=XkmK1C1zH-u8hy04No-SzsVh5iH6csKkYsEtyBzb84E,3479
63
63
  port_ocean/clients/port/mixins/blueprints.py,sha256=iAKwguhDpUL-YLd7GRNjS-monVgOG8UyKJFOengO_zM,4291
64
- port_ocean/clients/port/mixins/entities.py,sha256=UiVssYYqJeHhrLahx1mW24B7oGVMZV2WVvUze_htuBk,25279
64
+ port_ocean/clients/port/mixins/entities.py,sha256=n2Cwc944TADpQaLboZUJi7P7ibAa7mDCxvOAVKWJIRI,25900
65
65
  port_ocean/clients/port/mixins/integrations.py,sha256=rzmfv3BfsBXX21VZrhZLsH5B5spvVBo6xIiXKxOwNvg,12236
66
66
  port_ocean/clients/port/mixins/migrations.py,sha256=vdL_A_NNUogvzujyaRLIoZEu5vmKDY2BxTjoGP94YzI,1467
67
67
  port_ocean/clients/port/mixins/organization.py,sha256=A2cP5V49KnjoAXxjmnm_XGth4ftPSU0qURNfnyUyS_Y,1041
@@ -106,7 +106,7 @@ port_ocean/core/handlers/entities_state_applier/port/get_related_entities.py,sha
106
106
  port_ocean/core/handlers/entities_state_applier/port/order_by_entities_dependencies.py,sha256=lyv6xKzhYfd6TioUgR3AVRSJqj7JpAaj1LxxU2xAqeo,1720
107
107
  port_ocean/core/handlers/entity_processor/__init__.py,sha256=FvFCunFg44wNQoqlybem9MthOs7p1Wawac87uSXz9U8,156
108
108
  port_ocean/core/handlers/entity_processor/base.py,sha256=PsnpNRqjHth9xwOvDRe7gKu8cjnVV0XGmTIHGvOelX0,1867
109
- port_ocean/core/handlers/entity_processor/jq_entity_processor.py,sha256=Z0njO1z2FUh8hX4GTdH7CmO0Afv-WeYRtxPs34mxKWE,32181
109
+ port_ocean/core/handlers/entity_processor/jq_entity_processor.py,sha256=_ldJ71UIfZuNcLOgV5FYKllFohrtK13rDT9dI78ekPc,33279
110
110
  port_ocean/core/handlers/entity_processor/jq_input_evaluator.py,sha256=R88wf69RVtBl8t5m2IKGTmgt4JEQSbct_AmHI_tUOjg,5350
111
111
  port_ocean/core/handlers/port_app_config/__init__.py,sha256=8AAT5OthiVM7KCcM34iEgEeXtn2pRMrT4Dze5r1Ixbk,134
112
112
  port_ocean/core/handlers/port_app_config/api.py,sha256=r_Th66NEw38IpRdnXZcRvI8ACfvxW_A6V62WLwjWXlQ,1044
@@ -178,7 +178,7 @@ port_ocean/tests/core/defaults/test_common.py,sha256=sR7RqB3ZYV6Xn6NIg-c8k5K6JcG
178
178
  port_ocean/tests/core/event_listener/test_kafka.py,sha256=RN_JOCy4aRDUNvyQocO6WFvUMH2XeAZy-PIWHOYnD9M,2888
179
179
  port_ocean/tests/core/handlers/actions/test_execution_manager.py,sha256=5bOlcQ9qcXF_leoSFtw3FzTa7C1awUPo6TSss5e_76w,30753
180
180
  port_ocean/tests/core/handlers/entities_state_applier/test_applier.py,sha256=7XWgwUB9uVYRov4VbIz1A-7n2YLbHTTYT-4rKJxjB0A,10711
181
- port_ocean/tests/core/handlers/entity_processor/test_jq_entity_processor.py,sha256=-kLAOGuoA5yN1TT_T3SIPgGnsOqg1iK0qjVOQ9KiU3c,58498
181
+ port_ocean/tests/core/handlers/entity_processor/test_jq_entity_processor.py,sha256=gyyWbkQop50NsPaHGq1K1wlUD_SYYdnquQfOTqTbnGc,59403
182
182
  port_ocean/tests/core/handlers/entity_processor/test_jq_input_evaluator.py,sha256=xNlDK8d8YQOplgjZGSBq4rZYZx_atg2R5YyQnH0qfI4,42151
183
183
  port_ocean/tests/core/handlers/mixins/test_live_events.py,sha256=Sbv9IZAGQoZDhf27xDjMMVYxUSie9mHltDtxLSqckmM,12548
184
184
  port_ocean/tests/core/handlers/mixins/test_sync_raw.py,sha256=-Jd2rUG63fZM8LuyKtCp1tt4WEqO2m5woESjs1c91sU,44428
@@ -219,8 +219,8 @@ port_ocean/utils/repeat.py,sha256=U2OeCkHPWXmRTVoPV-VcJRlQhcYqPWI5NfmPlb1JIbc,32
219
219
  port_ocean/utils/signal.py,sha256=J1sI-e_32VHP_VUa5bskLMFoJjJOAk5isrnewKDikUI,2125
220
220
  port_ocean/utils/time.py,sha256=pufAOH5ZQI7gXvOvJoQXZXZJV-Dqktoj9Qp9eiRwmJ4,1939
221
221
  port_ocean/version.py,sha256=UsuJdvdQlazzKGD3Hd5-U7N69STh8Dq9ggJzQFnu9fU,177
222
- port_ocean-0.30.0.dist-info/LICENSE.md,sha256=WNHhf_5RCaeuKWyq_K39vmp9F28LxKsB4SpomwSZ2L0,11357
223
- port_ocean-0.30.0.dist-info/METADATA,sha256=uglxgjupYpTwJ-w_rUz6Tf17jeaTcdvT4xz1QNCsFpc,7054
224
- port_ocean-0.30.0.dist-info/WHEEL,sha256=Nq82e9rUAnEjt98J6MlVmMCZb-t9cYE2Ir1kpBmnWfs,88
225
- port_ocean-0.30.0.dist-info/entry_points.txt,sha256=F_DNUmGZU2Kme-8NsWM5LLE8piGMafYZygRYhOVtcjA,54
226
- port_ocean-0.30.0.dist-info/RECORD,,
222
+ port_ocean-0.30.1.dist-info/LICENSE.md,sha256=WNHhf_5RCaeuKWyq_K39vmp9F28LxKsB4SpomwSZ2L0,11357
223
+ port_ocean-0.30.1.dist-info/METADATA,sha256=yi01OJ2_QzuHHCv43m_K8j1JdM9cCwBbCMkBEG4ncfw,7054
224
+ port_ocean-0.30.1.dist-info/WHEEL,sha256=Nq82e9rUAnEjt98J6MlVmMCZb-t9cYE2Ir1kpBmnWfs,88
225
+ port_ocean-0.30.1.dist-info/entry_points.txt,sha256=F_DNUmGZU2Kme-8NsWM5LLE8piGMafYZygRYhOVtcjA,54
226
+ port_ocean-0.30.1.dist-info/RECORD,,