data.py•3.08 kB
"""
Tool for retrieving data from Splunk KV Store collections.
"""
from typing import Any
from fastmcp import Context
from src.core.base import BaseTool, ToolMetadata
from src.core.utils import log_tool_execution
class GetKvstoreData(BaseTool):
    """
    Retrieve data from a specific KV Store collection.
    """
    METADATA = ToolMetadata(
        name="get_kvstore_data",
        description=(
            "Get documents from a KV Store collection with optional MongoDB-style query filtering. Use this "
            "to fetch lookup/configuration data or narrow results by field values.\n\n"
            "Args:\n"
            "    collection (str): Collection name\n"
            "    app (str, optional): App where the collection resides (defaults to current/app context)\n"
            "    query (object, optional): MongoDB-style filter object (e.g., {\"status\": \"active\"})\n\n"
            "Outputs: 'documents' array and 'count'.\n"
            "Security: access and results are constrained by the authenticated user's permissions."
        ),
        category="kvstore",
        tags=["kvstore", "data", "query", "storage"],
        requires_connection=True,
    )
    async def execute(
        self, ctx: Context, collection: str, app: str | None = None, query: dict | None = None
    ) -> dict[str, Any]:
        """
        Retrieve data from a KV Store collection.
        Args:
            collection: Name of the collection to retrieve data from
            app: Optional app name where the collection resides
            query: Optional MongoDB-style query filter
        Returns:
            Dict containing retrieved documents
        """
        log_tool_execution("get_kvstore_data", collection=collection, app=app, query=query)
        is_available, service, error_msg = self.check_splunk_available(ctx)
        if not is_available:
            return self.format_error_response(error_msg)
        self.logger.info(f"Retrieving data from KV Store collection: {collection}")
        await ctx.info(f"Retrieving data from KV Store collection: {collection}")
        try:
            # Get the collection from the appropriate app context
            if app:
                kvstore = service.kvstore[app]
            else:
                kvstore = service.kvstore
            collection_obj = kvstore[collection]
            # Retrieve data with optional query filter
            if query:
                documents = collection_obj.data.query(**query)
            else:
                documents = collection_obj.data.query()
            # Convert to list for response
            doc_list = list(documents)
            await ctx.info(f"Retrieved {len(doc_list)} documents from collection {collection}")
            return self.format_success_response({"count": len(doc_list), "documents": doc_list})
        except Exception as e:
            self.logger.error(f"Failed to retrieve KV Store data: {str(e)}")
            await ctx.error(f"Failed to retrieve KV Store data: {str(e)}")
            return self.format_error_response(str(e))