port-ocean 0.12.2__tar.gz → 0.12.2.dev1__tar.gz
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.
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/PKG-INFO +1 -1
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/event_listener/base.py +11 -2
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/log/handlers.py +8 -7
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/ocean.py +12 -11
- port_ocean-0.12.2.dev1/port_ocean/utils/async_iterators.py +49 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/pyproject.toml +1 -1
- port_ocean-0.12.2/port_ocean/tests/utils/test_async_iterators.py +0 -45
- port_ocean-0.12.2/port_ocean/utils/async_iterators.py +0 -109
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/LICENSE.md +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/README.md +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/bootstrap.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cli.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/defaults/__init___.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/defaults/clean.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/defaults/dock.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/defaults/group.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/list_integrations.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/main.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/new.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/pull.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/sail.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/commands/version.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/cookiecutter.json +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/extensions.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/hooks/post_gen_project.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/.dockerignore +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/.env.example +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/.gitignore +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/.port/resources/.gitignore +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/.port/resources/blueprints.json +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/.port/resources/port-app-config.yml +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/.port/spec.yaml +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/CHANGELOG.md +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/CONTRIBUTING.md +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/Dockerfile +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/README.md +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/changelog/.gitignore +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/debug.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/main.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/poetry.toml +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/pyproject.toml +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/sonar-project.properties +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/tests/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/{{cookiecutter.integration_slug}}/tests/test_sample.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/utils.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/authentication.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/client.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/mixins/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/mixins/blueprints.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/mixins/entities.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/mixins/integrations.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/mixins/migrations.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/retry_transport.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/types.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/clients/port/utils.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/config/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/config/base.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/config/dynamic.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/config/settings.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/consumers/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/consumers/kafka_consumer.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/context/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/context/event.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/context/ocean.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/context/resource.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/defaults/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/defaults/clean.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/defaults/common.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/defaults/initialize.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/event_listener/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/event_listener/factory.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/event_listener/http.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/event_listener/kafka.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/event_listener/once.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/event_listener/polling.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/base.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entities_state_applier/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entities_state_applier/base.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entities_state_applier/port/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entities_state_applier/port/applier.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entities_state_applier/port/get_related_entities.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entities_state_applier/port/order_by_entities_dependencies.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entity_processor/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entity_processor/base.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entity_processor/jq_entity_processor.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/port_app_config/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/port_app_config/api.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/port_app_config/base.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/port_app_config/models.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/resync_state_updater/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/resync_state_updater/updater.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/base.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/mixins/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/mixins/events.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/mixins/handler.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/mixins/sync.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/mixins/sync_raw.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/mixins/utils.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/models.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/ocean_types.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/utils.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/exceptions/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/exceptions/api.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/exceptions/base.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/exceptions/clients.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/exceptions/context.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/exceptions/core.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/exceptions/port_defaults.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/exceptions/utils.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/helpers/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/helpers/async_client.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/helpers/retry.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/log/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/log/logger_setup.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/log/sensetive.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/middlewares.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/py.typed +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/run.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/sonar-project.properties +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/clients/port/mixins/test_entities.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/conftest.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/core/handlers/entity_processor/test_jq_entity_processor.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/helpers/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/helpers/fixtures.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/helpers/integration.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/helpers/ocean_app.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/helpers/port_client.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/helpers/smoke_test.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/test_smoke.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/utils/__init__.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/utils/async_http.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/utils/cache.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/utils/misc.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/utils/queue_utils.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/utils/repeat.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/utils/signal.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/utils/time.py +0 -0
- {port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/version.py +0 -0
|
@@ -1,12 +1,14 @@
|
|
|
1
|
+
import asyncio
|
|
1
2
|
from abc import abstractmethod
|
|
3
|
+
from concurrent.futures import ThreadPoolExecutor
|
|
2
4
|
from typing import TypedDict, Callable, Any, Awaitable
|
|
3
5
|
|
|
4
6
|
from pydantic import Extra
|
|
5
7
|
|
|
6
8
|
from port_ocean.config.base import BaseOceanModel
|
|
7
|
-
from port_ocean.utils.signal import signal_handler
|
|
8
9
|
from port_ocean.context.ocean import ocean
|
|
9
10
|
from port_ocean.utils.misc import IntegrationStateStatus
|
|
11
|
+
from port_ocean.utils.signal import signal_handler
|
|
10
12
|
|
|
11
13
|
|
|
12
14
|
class EventListenerEvents(TypedDict):
|
|
@@ -67,7 +69,14 @@ class BaseEventListener:
|
|
|
67
69
|
"""
|
|
68
70
|
await self._before_resync()
|
|
69
71
|
try:
|
|
70
|
-
|
|
72
|
+
loop = asyncio.get_event_loop()
|
|
73
|
+
with ThreadPoolExecutor() as executor:
|
|
74
|
+
e = executor.submit(
|
|
75
|
+
lambda: asyncio.run_coroutine_threadsafe(
|
|
76
|
+
self.events["on_resync"](resync_args), loop
|
|
77
|
+
)
|
|
78
|
+
)
|
|
79
|
+
signal_handler.register(e.cancel)
|
|
71
80
|
await self._after_resync()
|
|
72
81
|
except Exception as e:
|
|
73
82
|
await self._on_resync_failure(e)
|
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import asyncio
|
|
2
2
|
import logging
|
|
3
3
|
import sys
|
|
4
|
-
import threading
|
|
5
4
|
import time
|
|
5
|
+
from concurrent.futures import ThreadPoolExecutor
|
|
6
6
|
from datetime import datetime
|
|
7
7
|
from logging.handlers import MemoryHandler
|
|
8
8
|
from typing import Any
|
|
@@ -65,18 +65,19 @@ class HTTPMemoryHandler(MemoryHandler):
|
|
|
65
65
|
if self.ocean is None or not self.buffer:
|
|
66
66
|
return
|
|
67
67
|
|
|
68
|
-
def _wrap_event_loop(_ocean: Ocean, logs_to_send: list[dict[str, Any]]) -> None:
|
|
69
|
-
loop = asyncio.new_event_loop()
|
|
70
|
-
loop.run_until_complete(self.send_logs(_ocean, logs_to_send))
|
|
71
|
-
loop.close()
|
|
72
|
-
|
|
73
68
|
self.acquire()
|
|
74
69
|
logs = list(self._serialized_buffer)
|
|
75
70
|
if logs:
|
|
76
71
|
self.buffer.clear()
|
|
77
72
|
self._serialized_buffer.clear()
|
|
78
73
|
self.last_flush_time = time.time()
|
|
79
|
-
|
|
74
|
+
loop = asyncio.new_event_loop()
|
|
75
|
+
with ThreadPoolExecutor() as executor:
|
|
76
|
+
executor.submit(
|
|
77
|
+
lambda: asyncio.run_coroutine_threadsafe(
|
|
78
|
+
self.send_logs(self.ocean, logs), loop
|
|
79
|
+
)
|
|
80
|
+
)
|
|
80
81
|
self.release()
|
|
81
82
|
|
|
82
83
|
async def send_logs(
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import asyncio
|
|
2
2
|
import sys
|
|
3
|
-
import
|
|
3
|
+
from concurrent.futures import ThreadPoolExecutor
|
|
4
4
|
from contextlib import asynccontextmanager
|
|
5
5
|
from typing import Callable, Any, Dict, AsyncIterator, Type
|
|
6
6
|
|
|
@@ -9,8 +9,6 @@ from loguru import logger
|
|
|
9
9
|
from pydantic import BaseModel
|
|
10
10
|
from starlette.types import Scope, Receive, Send
|
|
11
11
|
|
|
12
|
-
from port_ocean.core.handlers.resync_state_updater import ResyncStateUpdater
|
|
13
|
-
from port_ocean.core.models import Runtime
|
|
14
12
|
from port_ocean.clients.port.client import PortClient
|
|
15
13
|
from port_ocean.config.settings import (
|
|
16
14
|
IntegrationConfiguration,
|
|
@@ -20,13 +18,15 @@ from port_ocean.context.ocean import (
|
|
|
20
18
|
ocean,
|
|
21
19
|
initialize_port_ocean_context,
|
|
22
20
|
)
|
|
21
|
+
from port_ocean.core.handlers.resync_state_updater import ResyncStateUpdater
|
|
23
22
|
from port_ocean.core.integrations.base import BaseIntegration
|
|
23
|
+
from port_ocean.core.models import Runtime
|
|
24
24
|
from port_ocean.log.sensetive import sensitive_log_filter
|
|
25
25
|
from port_ocean.middlewares import request_handler
|
|
26
|
+
from port_ocean.utils.misc import IntegrationStateStatus
|
|
26
27
|
from port_ocean.utils.repeat import repeat_every
|
|
27
28
|
from port_ocean.utils.signal import signal_handler
|
|
28
29
|
from port_ocean.version import __integration_version__
|
|
29
|
-
from port_ocean.utils.misc import IntegrationStateStatus
|
|
30
30
|
|
|
31
31
|
|
|
32
32
|
class Ocean:
|
|
@@ -92,6 +92,13 @@ class Ocean:
|
|
|
92
92
|
)
|
|
93
93
|
raise e
|
|
94
94
|
|
|
95
|
+
def pool_executor_wrapper(event_loop) -> None:
|
|
96
|
+
with ThreadPoolExecutor() as executor:
|
|
97
|
+
e = executor.submit(lambda: asyncio.run_coroutine_threadsafe(
|
|
98
|
+
execute_resync_all(), event_loop
|
|
99
|
+
))
|
|
100
|
+
signal_handler.register(e.cancel)
|
|
101
|
+
|
|
95
102
|
interval = self.config.scheduled_resync_interval
|
|
96
103
|
loop = asyncio.get_event_loop()
|
|
97
104
|
if interval is not None:
|
|
@@ -103,13 +110,7 @@ class Ocean:
|
|
|
103
110
|
seconds=interval * 60,
|
|
104
111
|
# Not running the resync immediately because the event listener should run resync on startup
|
|
105
112
|
wait_first=True,
|
|
106
|
-
)(
|
|
107
|
-
lambda: threading.Thread(
|
|
108
|
-
target=lambda: asyncio.run_coroutine_threadsafe(
|
|
109
|
-
execute_resync_all(), loop
|
|
110
|
-
)
|
|
111
|
-
).start()
|
|
112
|
-
)
|
|
113
|
+
)(lambda _: pool_executor_wrapper(loop))
|
|
113
114
|
await repeated_function()
|
|
114
115
|
|
|
115
116
|
async def __call__(self, scope: Scope, receive: Receive, send: Send) -> None:
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import typing
|
|
2
|
+
|
|
3
|
+
import aiostream
|
|
4
|
+
|
|
5
|
+
|
|
6
|
+
async def stream_async_iterators_tasks(
|
|
7
|
+
*tasks: typing.AsyncIterable[typing.Any],
|
|
8
|
+
) -> typing.AsyncIterable[typing.Any]:
|
|
9
|
+
"""
|
|
10
|
+
This function takes a list of async iterators and streams the results of each iterator as they are available.
|
|
11
|
+
By using this function you can combine multiple async iterators into a single stream of results, instead of waiting
|
|
12
|
+
for each iterator to finish before starting the next one.
|
|
13
|
+
|
|
14
|
+
Usage:
|
|
15
|
+
```python
|
|
16
|
+
async def async_iterator1():
|
|
17
|
+
for i in range(10):
|
|
18
|
+
yield i
|
|
19
|
+
await asyncio.sleep(1)
|
|
20
|
+
|
|
21
|
+
async def async_iterator2():
|
|
22
|
+
for i in range(10, 20):
|
|
23
|
+
yield i
|
|
24
|
+
await asyncio.sleep(1)
|
|
25
|
+
|
|
26
|
+
async def main():
|
|
27
|
+
async for result in stream_async_iterators_tasks([async_iterator1(), async_iterator2()]):
|
|
28
|
+
print(result)
|
|
29
|
+
```
|
|
30
|
+
|
|
31
|
+
Caution - Before using this function, make sure that the third-party API you are calling allows the number of
|
|
32
|
+
concurrent requests you are making. If the API has a rate limit, you may need to adjust the number of concurrent
|
|
33
|
+
requests to avoid hitting the rate limit.
|
|
34
|
+
|
|
35
|
+
:param tasks: A list of async iterators
|
|
36
|
+
:return: A stream of results
|
|
37
|
+
"""
|
|
38
|
+
if not tasks:
|
|
39
|
+
return
|
|
40
|
+
|
|
41
|
+
if len(tasks) == 1:
|
|
42
|
+
async for batch_items in tasks[0]:
|
|
43
|
+
yield batch_items
|
|
44
|
+
return
|
|
45
|
+
|
|
46
|
+
combine = aiostream.stream.merge(tasks[0], *tasks[1:])
|
|
47
|
+
async with combine.stream() as streamer:
|
|
48
|
+
async for batch_items in streamer:
|
|
49
|
+
yield batch_items
|
|
@@ -1,45 +0,0 @@
|
|
|
1
|
-
from typing import Any, AsyncGenerator
|
|
2
|
-
import asyncio
|
|
3
|
-
from port_ocean.utils.async_iterators import semaphore_async_iterator
|
|
4
|
-
import pytest
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
@pytest.mark.asyncio
|
|
8
|
-
async def test_semaphore_async_iterator() -> None:
|
|
9
|
-
max_concurrency = 5
|
|
10
|
-
semaphore = asyncio.BoundedSemaphore(max_concurrency)
|
|
11
|
-
|
|
12
|
-
concurrent_tasks = 0
|
|
13
|
-
max_concurrent_tasks = 0
|
|
14
|
-
lock = asyncio.Lock() # Protect shared variables
|
|
15
|
-
|
|
16
|
-
num_tasks = 20
|
|
17
|
-
|
|
18
|
-
async def mock_function() -> AsyncGenerator[str, None]:
|
|
19
|
-
nonlocal concurrent_tasks, max_concurrent_tasks
|
|
20
|
-
|
|
21
|
-
async with lock:
|
|
22
|
-
concurrent_tasks += 1
|
|
23
|
-
if concurrent_tasks > max_concurrent_tasks:
|
|
24
|
-
max_concurrent_tasks = concurrent_tasks
|
|
25
|
-
|
|
26
|
-
await asyncio.sleep(0.1)
|
|
27
|
-
yield "result"
|
|
28
|
-
|
|
29
|
-
async with lock:
|
|
30
|
-
concurrent_tasks -= 1
|
|
31
|
-
|
|
32
|
-
async def consume_iterator(async_iterator: Any) -> None:
|
|
33
|
-
async for _ in async_iterator:
|
|
34
|
-
pass
|
|
35
|
-
|
|
36
|
-
tasks = [
|
|
37
|
-
consume_iterator(semaphore_async_iterator(semaphore, mock_function))
|
|
38
|
-
for _ in range(num_tasks)
|
|
39
|
-
]
|
|
40
|
-
await asyncio.gather(*tasks)
|
|
41
|
-
|
|
42
|
-
assert (
|
|
43
|
-
max_concurrent_tasks <= max_concurrency
|
|
44
|
-
), f"Max concurrent tasks {max_concurrent_tasks} exceeded semaphore limit {max_concurrency}"
|
|
45
|
-
assert concurrent_tasks == 0, "Not all tasks have completed"
|
|
@@ -1,109 +0,0 @@
|
|
|
1
|
-
import typing
|
|
2
|
-
|
|
3
|
-
import aiostream
|
|
4
|
-
|
|
5
|
-
if typing.TYPE_CHECKING:
|
|
6
|
-
from asyncio import Semaphore
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
async def stream_async_iterators_tasks(
|
|
10
|
-
*tasks: typing.AsyncIterable[typing.Any],
|
|
11
|
-
) -> typing.AsyncIterable[typing.Any]:
|
|
12
|
-
"""
|
|
13
|
-
This function takes a list of async iterators and streams the results of each iterator as they are available.
|
|
14
|
-
By using this function you can combine multiple async iterators into a single stream of results, instead of waiting
|
|
15
|
-
for each iterator to finish before starting the next one.
|
|
16
|
-
|
|
17
|
-
Usage:
|
|
18
|
-
```python
|
|
19
|
-
async def async_iterator1():
|
|
20
|
-
for i in range(10):
|
|
21
|
-
yield i
|
|
22
|
-
await asyncio.sleep(1)
|
|
23
|
-
|
|
24
|
-
async def async_iterator2():
|
|
25
|
-
for i in range(10, 20):
|
|
26
|
-
yield i
|
|
27
|
-
await asyncio.sleep(1)
|
|
28
|
-
|
|
29
|
-
async def main():
|
|
30
|
-
async for result in stream_async_iterators_tasks([async_iterator1(), async_iterator2()]):
|
|
31
|
-
print(result)
|
|
32
|
-
```
|
|
33
|
-
|
|
34
|
-
Caution - Before using this function, make sure that the third-party API you are calling allows the number of
|
|
35
|
-
concurrent requests you are making. If the API has a rate limit, you may need to adjust the number of concurrent
|
|
36
|
-
requests to avoid hitting the rate limit.
|
|
37
|
-
|
|
38
|
-
:param tasks: A list of async iterators
|
|
39
|
-
:return: A stream of results
|
|
40
|
-
"""
|
|
41
|
-
if not tasks:
|
|
42
|
-
return
|
|
43
|
-
|
|
44
|
-
if len(tasks) == 1:
|
|
45
|
-
async for batch_items in tasks[0]:
|
|
46
|
-
yield batch_items
|
|
47
|
-
return
|
|
48
|
-
|
|
49
|
-
combine = aiostream.stream.merge(tasks[0], *tasks[1:])
|
|
50
|
-
async with combine.stream() as streamer:
|
|
51
|
-
async for batch_items in streamer:
|
|
52
|
-
yield batch_items
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
async def semaphore_async_iterator(
|
|
56
|
-
semaphore: "Semaphore",
|
|
57
|
-
function: typing.Callable[[], typing.AsyncIterator[typing.Any]],
|
|
58
|
-
) -> typing.AsyncIterator[typing.Any]:
|
|
59
|
-
"""
|
|
60
|
-
Executes an asynchronous iterator function under a semaphore to limit concurrency.
|
|
61
|
-
|
|
62
|
-
This function ensures that the provided asynchronous iterator function is executed
|
|
63
|
-
while respecting the concurrency limit imposed by the semaphore. It acquires the
|
|
64
|
-
semaphore before executing the function and releases it after the function completes,
|
|
65
|
-
thus controlling the number of concurrent executions.
|
|
66
|
-
|
|
67
|
-
Parameters:
|
|
68
|
-
semaphore (asyncio.Semaphore | asyncio.BoundedSemaphore): The semaphore used to limit concurrency.
|
|
69
|
-
function (Callable[[], AsyncIterator[Any]]): A nullary asynchronous function, - apply arguments with `functools.partial` or an anonymous function (lambda)
|
|
70
|
-
that returns an asynchronous iterator. This function is executed under the semaphore.
|
|
71
|
-
|
|
72
|
-
Yields:
|
|
73
|
-
Any: The items yielded by the asynchronous iterator function.
|
|
74
|
-
|
|
75
|
-
Usage:
|
|
76
|
-
```python
|
|
77
|
-
import asyncio
|
|
78
|
-
|
|
79
|
-
async def async_iterator_function(param1, param2):
|
|
80
|
-
# Your async code here
|
|
81
|
-
yield ...
|
|
82
|
-
|
|
83
|
-
async def async_generator_function():
|
|
84
|
-
# Your async code to retrieve items
|
|
85
|
-
param1 = "your_param1"
|
|
86
|
-
yield param1
|
|
87
|
-
|
|
88
|
-
async def main():
|
|
89
|
-
semaphore = asyncio.BoundedSemaphore(50)
|
|
90
|
-
param2 = "your_param2"
|
|
91
|
-
|
|
92
|
-
tasks = [
|
|
93
|
-
semaphore_async_iterator(
|
|
94
|
-
semaphore,
|
|
95
|
-
lambda: async_iterator_function(param1, param2) # functools.partial(async_iterator_function, param1, param2)
|
|
96
|
-
)
|
|
97
|
-
async for param1 in async_generator_function()
|
|
98
|
-
]
|
|
99
|
-
|
|
100
|
-
async for batch in stream_async_iterators_tasks(*tasks):
|
|
101
|
-
# Process each batch
|
|
102
|
-
pass
|
|
103
|
-
|
|
104
|
-
asyncio.run(main())
|
|
105
|
-
```
|
|
106
|
-
"""
|
|
107
|
-
async with semaphore:
|
|
108
|
-
async for result in function():
|
|
109
|
-
yield result
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/cli/cookiecutter/hooks/post_gen_project.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entities_state_applier/base.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entity_processor/__init__.py
RENAMED
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/entity_processor/base.py
RENAMED
|
File without changes
|
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/port_app_config/__init__.py
RENAMED
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/port_app_config/api.py
RENAMED
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/port_app_config/base.py
RENAMED
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/handlers/port_app_config/models.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/mixins/__init__.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/core/integrations/mixins/sync_raw.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{port_ocean-0.12.2 → port_ocean-0.12.2.dev1}/port_ocean/tests/clients/port/mixins/test_entities.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|