opensistemas-hub/osbrain

View on GitHub
osbrain/helper.py

Summary

Maintainability
B
4 hrs
Test Coverage
"""
Helper functions for osbrain (usually for testing purposes).
"""
import re
import time
from uuid import uuid4


def regex_count_in_list(regex, strings):
    """
    Returns
    -------
    int
        The number of times a regular expression was found in a list of
        strings.
    """
    matches = 0
    for candidate in strings:
        if re.findall(regex, candidate, re.DOTALL):
            matches += 1
    return matches


def logger_received(
    logger, message, log_name='log_history_info', position=None, timeout=1.0
):
    """
    Check if a logger receives a message.

    Parameters
    ----------
    logger : Proxy
        Proxy to the logger.
    log_name : str, default is `'log_history_info'`
        The name of the attribute to look for in the logger.
    message : anything
        Message to look for in the log. Can be a partial match. Regular
        expressions are allowed.
    position : int, default is None
        Where to look for the message in the log. If not set, the message
        will be searched for through all the log.
    timeout : float
        After this number of seconds the function will return `False`.

    Returns
    -------
    int
        The number of times the logger received a message that matched the
        expression. Can be higher than 1 if `position` is not set.
    """
    t0 = time.time()
    while True:
        time.sleep(0.01)
        log_history = logger.get_attr(log_name)
        if position is not None:
            log_history = [log_history[position]]
        matches = regex_count_in_list(message, log_history)
        if matches:
            return matches
        if timeout and time.time() - t0 > timeout:
            return 0


def sync_agent_logger(agent, logger):
    """
    Make sure and agent and a logger are synchronized.

    An agent is synchronized with its logger when we make sure the logger has
    started receiving messages from the agent.

    Parameters
    ----------
    agent : Proxy
        Proxy to the agent.
    logger : Proxy
        Proxy to the logger.
    """
    message = str(uuid4())
    delay = 0.01
    while (
        not len(logger.get_attr('log_history_info'))
        or message not in logger.get_attr('log_history_info')[-1]
    ):
        message = str(uuid4())
        agent.log_info(message)
        time.sleep(delay)
        delay *= 2


def agent_dies(agent, nsproxy, timeout=1.0):
    """
    Check if an agent dies within a given period.

    Parameters
    ----------
    agent : str
        Name of the agent, as registered in the name server.
    nsproxy : NSProxy
        Proxy to the name server.
    timeout : float
        After this number of seconds the function will return `False`.

    Returns
    -------
    bool
        Whether the agent died (was unregistered from the name server) within
        the given period.
    """
    assert isinstance(agent, str), 'Wrong type for `agent`!'
    t0 = time.time()
    while True:
        time.sleep(0.01)
        if agent not in nsproxy.agents():
            break
        if timeout and time.time() - t0 > timeout:
            return False
    return True


def attribute_match_all(
    attribute, length=None, data=None, value=None, endswith=None
):
    """
    Check if an attribute matches all of the following specified conditions:

    - Minimum length.
    - Contains an item.
    - Is exactly a value.

    Note that only those conditions explicitly passed will be checked.

    Parameters
    ----------
    attribute : anything
        The attribute to match against.
    length : int, default is None
        If specified, the attribute must reach this length (or higher).
    data : anything, default is None
        If specified, the attribute must contain this value.
    value : anything, default is None
        If specified, the attribute must be this value.
    endswith : sequence, default is None
        If specified, the attribute must end with this sequence.

    Returns
    -------
    bool
        Whether the attribute matches any condition.
    """
    assert (
        length is not None
        or data is not None
        or value is not None
        or endswith is not None
    ), 'No condition passed! Will return True always...'
    if length is not None and len(attribute) < length:
        return False
    if data is not None and data not in attribute:
        return False
    if value is not None and attribute != value:
        return False
    if endswith is not None and attribute[-len(endswith) :] != endswith:
        return False
    return True


def wait_agent_attr(agent, name='received', timeout=3, **kwargs):
    """
    Wait for an agent's attribute to match all of the following specified
    conditions:

    - Reach a minimum length.
    - Contain a particular item.
    - Become a given value.

    Note that only those conditions explicitly passed will be checked.

    Parameters
    ----------
    agent : Proxy
        A proxy to the agent.
    name : str, default is `'received'`
        Name of the agent's attribute to look for.
    timeout : float, default is 3
        After this number of seconds the function will return `False`.
    kwargs : dict
        Keyword arguments passed to `attribute_match_all` function.

    Returns
    -------
    bool
        Whether the specified conditions where met within the given time.
    """
    t0 = time.time()
    while True:
        attribute = agent.get_attr(name)
        if attribute_match_all(attribute, **kwargs):
            return True
        if time.time() - t0 > timeout:
            break
        time.sleep(0.01)
    return False


def wait_condition(condition, *args, negate=False, timeout=3, **kwargs):
    """
    Wait for a condition to be true.

    The condition is passed as a callable and must evaluate to a boolean
    result.

    Parameters
    ----------
    condition : Callable
        A function that evaluates the desired condition.
    timeout : float, default is 3
        After this number of seconds the function will return `False`.
    negate : bool, default is False
        Negate the condition (wait for the condition to be false instead).

    Returns
    -------
    bool
        Whether the specified condition was True.
    """
    t0 = time.time()
    while True:
        if condition(*args, **kwargs) != negate:
            return True
        if time.time() - t0 > timeout:
            return False
        time.sleep(0.01)


def wait_agent_condition(agent, condition, *args, timeout=3, **kwargs):
    """
    Wait for an agent's condition to be true.

    The condition is passed as a function which must take, at least, the
    actual agent as a parameter, and must evaluate to a boolean result.

    Parameters
    ----------
    agent : Proxy
        A proxy to the agent.
    condition : Callable
        A function that evaluates the desired condition.
    timeout : float, default is 3
        After this number of seconds the function will return `False`.

    Returns
    -------
    bool
        Whether the specified condition was True.
    """
    return wait_condition(
        agent.execute_as_method, condition, *args, timeout=timeout, **kwargs
    )


def last_received_endswith(agent, tail):
    """
    Check if the agent's last received message ends with `tail`.

    It is supposed to be executed as an agent method (in example: use it as
    the condition in `wait_agent_condition` or execute it with the method
    `execute_as_method`).

    Parameters
    ----------
    agent : Proxy
        A proxy to the agent.
    tail : str or bytes
        The expected termination of the last received message.

    Returns
    -------
    bool
        Whether the last message received ends with `tail`.
    """
    if not hasattr(agent, 'received'):
        return False
    if not agent.received:
        return False
    if not agent.received[-1][-len(tail) :] == tail:
        return False
    return True