Source code for saltext.cassandra.returners.cassandra_cql_return

"""
Return data to a cassandra server

.. versionadded:: 2015.5.0

:maintainer:    Corin Kochenower<ckochenower@saltstack.com>
:maturity:      new as of 2015.2
:depends:       salt.modules.cassandra_cql
:depends:       DataStax Python Driver for Apache Cassandra
                https://github.com/datastax/python-driver
                pip install cassandra-driver
:platform:      all

:configuration:
    To enable this returner, the minion will need the DataStax Python Driver
    for Apache Cassandra ( https://github.com/datastax/python-driver )
    installed and the following values configured in the minion or master
    config. The list of cluster IPs must include at least one cassandra node
    IP address. No assumption or default will be used for the cluster IPs.
    The cluster IPs will be tried in the order listed. The port, username,
    and password values shown below will be the assumed defaults if you do
    not provide values.:

    .. code-block:: yaml

        cassandra:
          cluster:
            - 192.168.50.11
            - 192.168.50.12
            - 192.168.50.13
          port: 9042
          username: salt
          password: salt

    Use the following cassandra database schema:

    .. code-block:: text

        CREATE KEYSPACE IF NOT EXISTS salt
            WITH replication = {'class': 'SimpleStrategy', 'replication_factor' : 1};

        CREATE USER IF NOT EXISTS salt WITH PASSWORD 'salt' NOSUPERUSER;

        GRANT ALL ON KEYSPACE salt TO salt;

        USE salt;

        CREATE TABLE IF NOT EXISTS salt.salt_returns (
            jid text,
            minion_id text,
            fun text,
            alter_time timestamp,
            full_ret text,
            return text,
            success boolean,
            PRIMARY KEY (jid, minion_id, fun)
        ) WITH CLUSTERING ORDER BY (minion_id ASC, fun ASC);
        CREATE INDEX IF NOT EXISTS salt_returns_minion_id ON salt.salt_returns (minion_id);
        CREATE INDEX IF NOT EXISTS salt_returns_fun ON salt.salt_returns (fun);

        CREATE TABLE IF NOT EXISTS salt.jids (
            jid text PRIMARY KEY,
            load text
        );

        CREATE TABLE IF NOT EXISTS salt.minions (
            minion_id text PRIMARY KEY,
            last_fun text
        );
        CREATE INDEX IF NOT EXISTS minions_last_fun ON salt.minions (last_fun);

        CREATE TABLE IF NOT EXISTS salt.salt_events (
            id timeuuid,
            tag text,
            alter_time timestamp,
            data text,
            master_id text,
            PRIMARY KEY (id, tag)
        ) WITH CLUSTERING ORDER BY (tag ASC);
        CREATE INDEX tag ON salt.salt_events (tag);


Required python modules: cassandra-driver

To use the cassandra returner, append '--return cassandra_cql' to the salt command. ex:

.. code-block:: bash

    salt '*' test.ping --return_cql cassandra

Note: if your Cassandra instance has not been tuned much you may benefit from
altering some timeouts in `cassandra.yaml` like so:

.. code-block:: yaml

    # How long the coordinator should wait for read operations to complete
    read_request_timeout_in_ms: 5000
    # How long the coordinator should wait for seq or index scans to complete
    range_request_timeout_in_ms: 20000
    # How long the coordinator should wait for writes to complete
    write_request_timeout_in_ms: 20000
    # How long the coordinator should wait for counter writes to complete
    counter_write_request_timeout_in_ms: 10000
    # How long a coordinator should continue to retry a CAS operation
    # that contends with other proposals for the same row
    cas_contention_timeout_in_ms: 5000
    # How long the coordinator should wait for truncates to complete
    # (This can be much longer, because unless auto_snapshot is disabled
    # we need to flush first so we can snapshot before removing the data.)
    truncate_request_timeout_in_ms: 60000
    # The default timeout for other, miscellaneous operations
    request_timeout_in_ms: 20000

As always, your mileage may vary and your Cassandra cluster may have different
needs.  SaltStack has seen situations where these timeouts can resolve
some stacktraces that appear to come from the Datastax Python driver.

"""

import logging
import time
import uuid

import salt.exceptions
import salt.returners
import salt.utils.jid
import salt.utils.json
from salt.exceptions import CommandExecutionError

try:
    # The following imports are not directly required by this module. Rather,
    # they are required by the modules/cassandra_cql execution module, on which
    # this module depends.
    #
    # This returner cross-calls the cassandra_cql execution module using the __salt__ dunder.
    #
    # The modules/cassandra_cql execution module will not load if the DataStax Python Driver
    # for Apache Cassandra is not installed.
    #
    # This module will try to load all of the 3rd party dependencies on which the
    # modules/cassandra_cql execution module depends.
    #
    # Effectively, if the DataStax Python Driver for Apache Cassandra is not
    # installed, both the modules/cassandra_cql execution module and this returner module
    # will not be loaded by Salt's loader system.
    # pylint: disable=unused-import,no-name-in-module
    from cassandra.auth import PlainTextAuthProvider
    from cassandra.cluster import Cluster
    from cassandra.cluster import NoHostAvailable
    from cassandra.connection import ConnectionException
    from cassandra.connection import ConnectionShutdown
    from cassandra.query import dict_factory

    # pylint: enable=unused-import,no-name-in-module
    HAS_CASSANDRA_DRIVER = True
except ImportError as e:
    HAS_CASSANDRA_DRIVER = False

log = logging.getLogger(__name__)

# Define the module's virtual name
#
# The 'cassandra' __virtualname__ is already taken by the
# returners/cassandra_return module, which utilizes nodetool. This module
# cross-calls the modules/cassandra_cql execution module, which uses the
# DataStax Python Driver for Apache Cassandra. Namespacing allows both the
# modules/cassandra_cql and returners/cassandra_cql modules to use the
# virtualname 'cassandra_cql'.
__virtualname__ = "cassandra_cql"


def __virtual__():
    if not HAS_CASSANDRA_DRIVER:
        return (
            False,
            "Could not import cassandra_cql returner; cassandra-driver is not installed.",
        )

    return True


[docs] def returner(ret): """ Return data to one of potentially many clustered cassandra nodes """ query = """INSERT INTO salt.salt_returns ( jid, minion_id, fun, alter_time, full_ret, return, success ) VALUES (?, ?, ?, ?, ?, ?, ?)""" ret_fun = ret["fun"] ret_id = ret["id"] ret_jid = ret["jid"] statement_arguments = [ f"{ret_jid}", f"{ret_id}", f"{ret_fun}", int(time.time() * 1000), salt.utils.json.dumps(ret).replace("'", "''"), salt.utils.json.dumps(ret["return"]).replace("'", "''"), ret.get("success", False), ] # cassandra_cql.cql_query may raise a CommandExecutionError try: __salt__["cassandra_cql.cql_query_with_prepare"]( query, "returner_return", tuple(statement_arguments), asynchronous=True ) except CommandExecutionError: log.critical("Could not insert into salt_returns with Cassandra returner.") raise except Exception as e: # pylint: disable=broad-except log.critical("Unexpected error while inserting into salt_returns: %s", e) raise # Store the last function called by the minion # The data in salt.minions will be used by get_fun and get_minions query = """INSERT INTO salt.minions ( minion_id, last_fun ) VALUES (?, ?)""" statement_arguments = [f"{ret_id}", f"{ret_fun}"] # cassandra_cql.cql_query may raise a CommandExecutionError try: __salt__["cassandra_cql.cql_query_with_prepare"]( query, "returner_minion", tuple(statement_arguments), asynchronous=True ) except CommandExecutionError: log.critical("Could not store minion ID with Cassandra returner.") raise except Exception as e: # pylint: disable=broad-except log.critical( "Unexpected error while inserting minion ID into the minions table: %s", e, ) raise
[docs] def event_return(events): """ Return event to one of potentially many clustered cassandra nodes Requires that configuration be enabled via 'event_return' option in master config. Cassandra does not support an auto-increment feature due to the highly inefficient nature of creating a monotonically increasing number across all nodes in a distributed database. Each event will be assigned a uuid by the connecting client. """ for event in events: tag = event.get("tag", "") data = event.get("data", "") query = """INSERT INTO salt.salt_events ( id, alter_time, data, master_id, tag ) VALUES ( ?, ?, ?, ?, ?) """ statement_arguments = [ str(uuid.uuid1()), int(time.time() * 1000), salt.utils.json.dumps(data).replace("'", "''"), __opts__["id"], tag, ] # cassandra_cql.cql_query may raise a CommandExecutionError try: __salt__["cassandra_cql.cql_query_with_prepare"]( query, "salt_events", statement_arguments, asynchronous=True ) except CommandExecutionError: log.critical("Could not store events with Cassandra returner.") raise except Exception as e: # pylint: disable=broad-except log.critical("Unexpected error while inserting into salt_events: %s", e) raise
[docs] def save_load(jid, load): """ Save the load to the specified jid id """ # Load is being stored as a text datatype. Single quotes are used in the # VALUES list. Therefore, all single quotes contained in the results from # salt.utils.json.dumps(load) must be escaped Cassandra style. query = """INSERT INTO salt.jids ( jid, load ) VALUES (?, ?)""" statement_arguments = [jid, salt.utils.json.dumps(load).replace("'", "''")] # cassandra_cql.cql_query may raise a CommandExecutionError try: __salt__["cassandra_cql.cql_query_with_prepare"]( query, "save_load", statement_arguments, asynchronous=True ) except CommandExecutionError: log.critical("Could not save load in jids table.") raise except Exception as e: # pylint: disable=broad-except log.critical("Unexpected error while inserting into jids: %s", e) raise
[docs] def save_minions(jid, minions, syndic_id=None): # pylint: disable=unused-argument """ Included for API consistency """
# salt-run jobs.list_jobs FAILED
[docs] def get_load(jid): """ Return the load data that marks a specified jid """ query = """SELECT load FROM salt.jids WHERE jid = ?;""" ret = {} # cassandra_cql.cql_query may raise a CommandExecutionError try: data = __salt__["cassandra_cql.cql_query_with_prepare"](query, "get_load", [jid]) if data: load = data[0].get("load") if load: ret = salt.utils.json.loads(load) except CommandExecutionError: log.critical("Could not get load from jids table.") raise except Exception as e: # pylint: disable=broad-except log.critical("Unexpected error while getting load from jids: %s", e) raise return ret
# salt-call ret.get_jid cassandra_cql 20150327234537907315 PASSED
[docs] def get_jid(jid): """ Return the information returned when the specified job id was executed """ query = """SELECT minion_id, full_ret FROM salt.salt_returns WHERE jid = ?;""" ret = {} # cassandra_cql.cql_query may raise a CommandExecutionError try: data = __salt__["cassandra_cql.cql_query_with_prepare"](query, "get_jid", [jid]) if data: for row in data: minion = row.get("minion_id") full_ret = row.get("full_ret") if minion and full_ret: ret[minion] = salt.utils.json.loads(full_ret) except CommandExecutionError: log.critical("Could not select job specific information.") raise except Exception as e: # pylint: disable=broad-except log.critical("Unexpected error while getting job specific information: %s", e) raise return ret
# salt-call ret.get_fun cassandra_cql test.ping PASSED
[docs] def get_fun(fun): """ Return a dict of the last function called for all minions """ query = """SELECT minion_id, last_fun FROM salt.minions where last_fun = ?;""" ret = {} # cassandra_cql.cql_query may raise a CommandExecutionError try: data = __salt__["cassandra_cql.cql_query"](query, "get_fun", [fun]) if data: for row in data: minion = row.get("minion_id") last_fun = row.get("last_fun") if minion and last_fun: ret[minion] = last_fun except CommandExecutionError: log.critical("Could not get the list of minions.") raise except Exception as e: # pylint: disable=broad-except log.critical("Unexpected error while getting list of minions: %s", e) raise return ret
# salt-call ret.get_jids cassandra_cql PASSED
[docs] def get_jids(): """ Return a list of all job ids """ query = """SELECT jid, load FROM salt.jids;""" ret = {} # cassandra_cql.cql_query may raise a CommandExecutionError try: data = __salt__["cassandra_cql.cql_query"](query) if data: for row in data: jid = row.get("jid") load = row.get("load") if jid and load: ret[jid] = salt.utils.jid.format_jid_instance(jid, salt.utils.json.loads(load)) except CommandExecutionError: log.critical("Could not get a list of all job ids.") raise except Exception as e: # pylint: disable=broad-except log.critical("Unexpected error while getting list of all job ids: %s", e) raise return ret
# salt-call ret.get_minions cassandra_cql PASSED
[docs] def get_minions(): """ Return a list of minions """ query = """SELECT DISTINCT minion_id FROM salt.minions;""" ret = [] # cassandra_cql.cql_query may raise a CommandExecutionError try: data = __salt__["cassandra_cql.cql_query"](query) if data: for row in data: minion = row.get("minion_id") if minion: ret.append(minion) except CommandExecutionError: log.critical("Could not get the list of minions.") raise except Exception as e: # pylint: disable=broad-except log.critical("Unexpected error while getting list of minions: %s", e) raise return ret
[docs] def prep_jid(nocache, passed_jid=None): # pylint: disable=unused-argument """ Do any work necessary to prepare a JID, including sending a custom id """ return passed_jid if passed_jid is not None else salt.utils.jid.gen_jid(__opts__)