Interactive Queries
Learn how to query the state in Kafka Streams applications.
We'll cover the following...
Previously, we saw how to use state stores to persist the intermediate results of stateful operations that require access to previously processed records, such as aggregations and joins. These stores are implemented as key-value stores and can be either in-memory or disk-based. The full state of the application is typically split across many distributed instances of our application and across many state stores managed locally by these application instances. We can only access the state stores through a Kafka topic. For example, in the case of a KTable, we converted it to a KStream, wrote it to a topic, and read it to get the data.
However, this is not the only way. Kafka Streams enables our applications to be queryable through a feature called Interactive Queries. This provides read-only access to the state stores used by Kafka Streams, enabling external applications to query the state of the stream-processing application and retrieve specific information. Let’s suppose we have a Kafka Streams application that processes a stream of sales data and calculates the total sales for each product. With Interactive Queries, external applications can retrieve the total sales for a specific product in real time, allowing them to make data-driven decisions based on the current state of the stream-processing application.
It’s important to understand that Interactive Queries are only a set of APIs in the Kafka Streams library that expose the state store data, but exposing that state (data) to external applications requires additional work. The good thing is that this can be done in various ways, depending on the application requirements. One common way to expose data using Interactive Queries is through a REST API. Kafka Streams provides a built-in REST endpoint that can be used to query the state stores of the stream-processing application. External applications can use this REST API to retrieve specific data from the state stores, and the data can be returned in a JSON format that can be easily consumed by client applications.
Let’s understand how this works in the context of an application. Before we try out the application, we will review its architecture at a high level.
Application architecture
The diagram below provides a high-level overview of the architecture of an application that uses Kafka Streams Interactive Queries.
The application consists of the following two parts:
- The main Kafka Streams topology that runs the stream-processing topology to maintain counts of stream of product views 
- The RPC layer, which in this case is the REST API, that exposes the state store data to external applications 
Sample code
Click the “Run” button in the widget below. This will initiate the build process and start the Kafka Streams application. Once the application has started, you should see the Started kafka streams application message.
After that, follow the steps outlined below:
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
         https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>com.example</groupId>
    <artifactId>spring-kafka-producer</artifactId>
    <version>1.0-SNAPSHOT</version>
    <packaging>jar</packaging>
    <properties>
        <maven.compiler.source>17</maven.compiler.source>
        <maven.compiler.target>17</maven.compiler.target>
    </properties>
    <dependencies>
        <!-- Kafka Streams -->
        <dependency>
            <groupId>org.apache.kafka</groupId>
            <artifactId>kafka-streams</artifactId>
            <version>3.7.0</version>
        </dependency>
        <!-- Jersey + Grizzly for REST API -->
        <dependency>
            <groupId>org.glassfish.jersey.containers</groupId>
            <artifactId>jersey-container-grizzly2-http</artifactId>
            <version>2.41</version>
        </dependency>
        <dependency>
            <groupId>org.glassfish.jersey.core</groupId>
            <artifactId>jersey-server</artifactId>
            <version>2.41</version>
        </dependency>
        <dependency>
            <groupId>org.glassfish.jersey.inject</groupId>
            <artifactId>jersey-hk2</artifactId>
            <version>2.41</version>
        </dependency>
    </dependencies>
    <build>
        <plugins>
            <!-- Compiler plugin -->
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.10.1</version>
                <configuration>
                    <source>${maven.compiler.source}</source>
                    <target>${maven.compiler.target}</target>
                </configuration>
            </plugin>
            <!-- Shade plugin to build executable JAR with dependencies -->
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-shade-plugin</artifactId>
                <version>3.3.0</version>
                <executions>
                    <execution>
                        <phase>package</phase>
                        <goals>
                            <goal>shade</goal>
                        </goals>
                        <configuration>
                            <transformers>
                                <!-- This ensures MANIFEST.MF has Main-Class -->
                                <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
                                    <mainClass>com.example.demo.App</mainClass>
                                </transformer>
                            </transformers>
                        </configuration>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>
</project>
Click the “+” button to open a new terminal tab, and enter the command to start the Kafka consumer ...