All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.smartcat.cassandra.diagnostics.connector.QueryProcessorWrapper Maven / Gradle / Ivy

package io.smartcat.cassandra.diagnostics.connector;

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

import org.apache.cassandra.cql3.CQLStatement;
import org.apache.cassandra.cql3.QueryOptions;
import org.apache.cassandra.cql3.QueryProcessor;
import org.apache.cassandra.cql3.statements.ModificationStatement;
import org.apache.cassandra.cql3.statements.ParsedStatement;
import org.apache.cassandra.cql3.statements.SelectStatement;
import org.apache.cassandra.service.QueryState;
import org.apache.cassandra.transport.messages.ResultMessage;
import org.apache.cassandra.utils.MD5Digest;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.googlecode.concurrentlinkedhashmap.ConcurrentLinkedHashMap;

import io.smartcat.cassandra.diagnostics.Query;
import io.smartcat.cassandra.diagnostics.Query.ConsistencyLevel;

/**
 * This class is a Diagnostics wrapper for {@link org.apache.cassandra.cql3.QueryProcessor}.
 */
public class QueryProcessorWrapper extends AbstractEventProcessor {

    private static final Logger logger = LoggerFactory.getLogger(QueryProcessorWrapper.class);

    private static final ConcurrentMap PREPARED_STATEMENT_QUERIES = new ConcurrentHashMap<>();

    private boolean slowQueryTracingEnabled = false;

    /**
     * Constructor.
     *
     * @param queryReporter QueryReporter used to report queries
     * @param configuration Connector configuration
     */
    public QueryProcessorWrapper(QueryReporter queryReporter, ConnectorConfiguration configuration) {
        super(queryReporter, configuration);
        slowQueryTracingEnabled = configuration.enableTracing;
    }

    /**
     * Wrapper for
     * {@link org.apache.cassandra.cql3.QueryProcessor#processPrepared(CQLStatement, QueryState, QueryOptions)} method.
     * This method is invoked after the original method, measures the execution time and reports query.
     *
     * @param statement  QueryProcessor#processPrepared(CQLStatement, QueryState, QueryOptions)
     * @param queryState QueryProcessor#processPrepared(CQLStatement, QueryState, QueryOptions)
     * @param options    QueryProcessor#processPrepared(CQLStatement, QueryState, QueryOptions)
     * @param startTime  query execution start time
     * @param result statement execution result
     * @param preparedStatements QueryProcessor's internal list of prepared statements
     */
    public void processPrepared(CQLStatement statement, QueryState queryState, QueryOptions options, long startTime,
            ResultMessage result, ConcurrentLinkedHashMap preparedStatements) {
        final long execTime = System.currentTimeMillis() - startTime;
        report(startTime, execTime, null, statement, queryState, options, preparedStatements);
    }

    /**
     * Wrapper for
     * {@link org.apache.cassandra.cql3.QueryProcessor#process(String, QueryState, QueryOptions)} method.
     * This method is invoked after the original method, measures the execution time and reports query.
     *
     * @param queryString  QueryProcessor#process(String, QueryState, QueryOptions)
     * @param queryState QueryProcessor#process(String, QueryState, QueryOptions)
     * @param options    QueryProcessor#process(String, QueryState, QueryOptions)
     * @param startTime  query execution start time
     * @param result statement execution result
     */
    public void process(String queryString, QueryState queryState, QueryOptions options, long startTime,
            ResultMessage result) {
        final long execTime = System.currentTimeMillis() - startTime;
        report(startTime, execTime, queryString, null, queryState, options, null);
    }

    /**
     * Wrapper for
     * {@link org.apache.cassandra.cql3.QueryProcessor#storePreparedStatement(String, String,
     * ParsedStatement.Prepared, boolean)} method.
     * This method is invoked after the original method, measures the execution time and reports query.
     *
     * @param queryString QueryProcessor#storePreparedStatement(String, String, ParsedStatement.Prepared, boolean)
     * @param keyspace    QueryProcessor#storePreparedStatement(String, String, ParsedStatement.Prepared, boolean)
     * @param forThrift   QueryProcessor#storePreparedStatement(String, String, ParsedStatement.Prepared, boolean)
     * @param prepared    QueryProcessor#storePreparedStatement(String, String, ParsedStatement.Prepared, boolean)
     * @param preparedStatements  QueryProcessor#preparedStatements
     */
    public void storePrepared(String queryString, String keyspace, boolean forThrift,
            ParsedStatement.Prepared prepared,
            ConcurrentLinkedHashMap preparedStatements) {
        for (MD5Digest digest : preparedStatements.keySet()) {
            ParsedStatement.Prepared existingPrepared = preparedStatements.get(digest);
            if (prepared == existingPrepared) {
                PREPARED_STATEMENT_QUERIES.put(digest, queryString);
            }
        }
    }

    /**
     * Submits a query reports asynchronously.
     *
     * @param startTime  execution start time, in milliseconds
     * @param execTime   execution time, in milliseconds
     * @param preparedStatementKey  prepared statement digest/key
     * @param queryState CQL query state
     * @param options    CQL query options
     */
    private void report(final long startTime, final long execTime, final String queryString,
            final CQLStatement statement, final QueryState queryState, final QueryOptions options,
            final ConcurrentLinkedHashMap preparedStatements) {
        if (queryState.getClientState().isInternal) {
            return;
        }
        report(new Runnable() {
            @Override
            public void run() {
                try {
                    CQLStatement cqlStatement;
                    String cqlQuery = "";

                    if (statement == null && queryString == null) {
                        throw new IllegalStateException("Both prepared statement and query string are missing.");
                    } else if (statement == null) {
                        cqlStatement = QueryProcessor.parseStatement(queryString).prepare().statement;

                        if (slowQueryTracingEnabled) {
                            cqlQuery = queryString;
                        }
                    } else {
                        cqlStatement = statement;

                        if (slowQueryTracingEnabled) {
                            MD5Digest preparedStatementKey = null;
                            for (MD5Digest digest : preparedStatements.keySet()) {
                                ParsedStatement.Prepared prepared = preparedStatements.get(digest);
                                if (prepared.statement == statement) {
                                    preparedStatementKey = digest;
                                }
                            }
                            cqlQuery = PREPARED_STATEMENT_QUERIES.get(preparedStatementKey);
                        }
                    }

                    Query query = createQuery(startTime, execTime, cqlQuery, cqlStatement, queryState, options);
                    logger.trace("Reporting query: {}.", query);
                    queryReporter.report(query);
                } catch (Exception e) {
                    logger.warn("An error occured while reporting query", e);
                }
            }
        });
    }

    private Query createQuery(final long startTime, final long execTime, final String queryString,
            final CQLStatement statement, final QueryState queryState, final QueryOptions options) {
        Query query;
        if (statement instanceof SelectStatement) {
            query = createQuery(startTime, execTime, queryString, (SelectStatement) statement,
                    queryState, options);
        } else if (statement instanceof ModificationStatement) {
            query = createQuery(startTime, execTime, queryString, (ModificationStatement) statement,
                    queryState, options);
        } else {
            query = createGenericQuery(startTime, execTime, queryString, statement, queryState, options);
        }
        return query;
    }

    private Query createQuery(final long startTime, final long execTime, final String queryString,
            final SelectStatement statement, final QueryState queryState, final QueryOptions options) {
        return Query.create(startTime, execTime, queryState.getClientState().getRemoteAddress().toString(),
                Query.StatementType.SELECT, statement.keyspace(), statement.columnFamily(), queryString,
                extractConsistencyLevel(options));
    }

    private Query createQuery(final long startTime, final long execTime, final String queryString,
            final ModificationStatement statement, final QueryState queryState, final QueryOptions options) {
        return Query.create(startTime, execTime, queryState.getClientState().getRemoteAddress().toString(),
                Query.StatementType.UPDATE, statement.keyspace(), statement.columnFamily(), queryString,
                extractConsistencyLevel(options));
    }

    private Query createGenericQuery(final long startTime, final long execTime, final String queryString,
            final CQLStatement statement, final QueryState queryState, final QueryOptions options) {
        return Query.create(startTime, execTime, queryState.getClientState().getRemoteAddress().toString(),
                Query.StatementType.UNKNOWN, "", "", queryString, extractConsistencyLevel(options));
    }

    private ConsistencyLevel extractConsistencyLevel(final QueryOptions queryOptions) {
        ConsistencyLevel queryConsistencyLevel = ConsistencyLevel.UNKNOWN;

        for (ConsistencyLevel consistencyLevel : ConsistencyLevel.values()) {
            if (consistencyLevel.name().equals(queryOptions.getConsistency().name())) {
                queryConsistencyLevel = consistencyLevel;
                break;
            }
        }

        return queryConsistencyLevel;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy