Viewing file: accumulate.py (2.77 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
import asyncio import collections import contextlib import os from logging import getLogger
from defence360agent.api import inactivity from defence360agent.contracts.messages import MessageType, Splittable from defence360agent.contracts.plugins import ( MessageSink, MessageSource, expect, ) from defence360agent.utils import recurring_check
logger = getLogger(__name__)
class Accumulate(MessageSink, MessageSource): PROCESSING_ORDER = MessageSink.ProcessingOrder.POST_PROCESS_MESSAGE DEFAULT_AGGREGATE_TIMEOUT = int( os.environ.get("IMUNIFY360_AGGREGATE_MESSAGES_TIMEOUT", 60) )
def __init__(self, period=DEFAULT_AGGREGATE_TIMEOUT, **kwargs): super().__init__(**kwargs) self._period = period self._data = collections.defaultdict(list)
async def create_source(self, loop, sink): self._loop = loop self._sink = sink self._task = loop.create_task( recurring_check(self._period)(self._flush)() )
async def create_sink(self, loop): self._loop = loop
async def shutdown(self): for list_type, messages in self._data.items(): logger.error( "Drop %s(<items=%s>) on shutdown", list_type.__name__, len(messages), ) if self._task is not None: self._task.cancel() with contextlib.suppress(asyncio.CancelledError): await self._task
@expect(MessageType.Accumulatable) async def collect(self, message): list_types = ( message.LIST_CLASS if isinstance(message.LIST_CLASS, tuple) else (message.LIST_CLASS,) ) if message.do_accumulate(): with inactivity.track.task("accumulate"): for list_type in list_types: self._data[list_type].append(message)
async def _flush(self): copy_data = self._data self._data = collections.defaultdict(list)
for list_type, messages in copy_data.items(): batched = ( list_type.batched(messages) if issubclass(list_type, Splittable) else (messages,) )
for batch in batched: logger.info( f"Prepare {list_type.__name__}(<items={len(batch)}>) " "for further processing" ) try: # FIXME: remove this try..except block after # we have forbidden to create Accumulatable class # without LIST_CLASS. await self._sink.process_message(list_type(items=batch)) except TypeError: logger.error("%s, %s", list_type, batch) raise
|