-
Notifications
You must be signed in to change notification settings - Fork 3
Structure of the codebase
There are three distinct layers to Storm's codebase.
First, Storm was designed from the very beginning to be compatible with multiple languages. Nimbus is a Thrift service and topologies are defined as Thrift structures. The usage of Thrift allows Storm to be used from any language.
Second, all of Storm's interfaces are specified as Java interfaces. So even though there's a lot of Clojure in Storm's implementation, all usage must go through the Java API. This means that every feature of Storm is always available via Java.
Third, Storm's implementation is largely in Clojure. Line-wise, Storm is about half Java code, half Clojure code. But Clojure is much more expressive, so in reality the great majority of the implementation logic is in Clojure.
The following sections explain each of these layers in more detail.
The first place to look to understand the structure of Storm's codebase is the storm.thrift file.
Storm uses this fork of Thrift to produce the generated code. This "fork" is actually Thrift 7 with all the Java packages renamed to be org.apache.thrift7
. Otherwise, it's identical to Thrift 7. This fork was done because of the lack of backwards compatibility in Thrift and the need for many people to use other versions of Thrift in their Storm topologies.
Every spout or bolt in a topology is given a user-specified identifier called the "component id". The component id is used to specify subscriptions from a bolt to the output streams of other spouts or bolts. A StormTopology structure contains a map from component id to component for each type of component (spouts and bolts).
Spouts and bolts have the same Thrift definition, so let's just take a look at the Thrift definition for bolts. It contains a ComponentObject
struct and a ComponentCommon
struct.
The ComponentObject
defines the implementation for the bolt. It can be one of three types:
- A serialized java object (that implements IBolt)
- A
ShellComponent
object that indicates the implementation is in another language. Specifying a bolt this way will cause Storm to instantiate a ShellBolt object to handle the communication between the JVM-based worker process and the non-JVM-based implementation of the component. - A
JavaObject
structure which tells Storm the classname and constructor arguments to use to instantiate that bolt. This is useful if you want to define a topology in a non-JVM language. This way, you can make use of JVM-based spouts and bolts without having to create and serialize a Java object yourself.
ComponentCommon
defines everything else for this component. This includes:
- What streams this component emits and the metadata for each stream (whether it's a direct stream, the fields declaration)
- What streams this component consumes (specified as a map from component_id:stream_id to the stream grouping to use)
- The parallelism for this component
- The component-specific configuration for this component
Note that the structure spouts also have a ComponentCommon
field, and so spouts can also have declarations to consume other input streams. Yet the Storm Java API does not provide a way for spouts to consume other streams, and if you put any input declarations there for a spout you would get an error when you tried to submit the topology. The reason that spouts have an input declarations field is not for users to use, but for Storm itself to use. Storm adds implicit streams and bolts to the topology to set up the acking framework, and two of these implicit streams are from the acker bolt to each spout in the topology. The acker sends "ack" or "fail" messages along these streams whenever a tuple tree is detected to be completed or failed. The code that transforms the user's topology into the runtime topology is located here.
The interfaces for Storm are generally specified as Java interfaces. The main interfaces are:
The strategy for the majority of the interfaces is to:
- Specify the interface using a Java interface
- Provide a base class that provides default implementations when appropriate
You can see this strategy at work with the BaseRichSpout class.
Spouts and bolts are serialized into the Thrift definition of the topology as described above.
One subtle aspect of the interfaces is the difference between IBolt
and ISpout
vs. IRichBolt
and IRichSpout
. The main difference between them is the addition of the declareOutputFields
method in the "Rich" versions of the interfaces. The reason for the split is that the output fields declaration for each output stream needs to be part of the Thrift struct (so it can be specified from any language), but as a user you want to be able to declare the streams as part of your class. What TopologyBuilder
does when constructing the Thrift representation is call declareOutputFields
to get the declaration and convert it into the Thrift structure. The conversion happens at this portion of the TopologyBuilder
code.
Specifying all the functionality via Java interfaces ensures that every feature of Storm is available via Java. Moreso, the focus on Java interfaces ensures that the user experience from Java-land is pleasant as well.
The implementation of Storm, on the other hand, is primarily in Clojure. While the codebase is about 50% Java and 50% Clojure in terms of LOC, most of the implementation logic is in Clojure. There are two notable exceptions to this, and that is the DRPC and transactional topologies implementations. These are implemented purely in Java. This was done to serve as an illustration for how to implement a higher level abstraction on Storm. The DRPC and transactional topologies implementations are in the backtype.storm.coordination, backtype.storm.drpc, and backtype.storm.transactional packages.
Here's a summary of the purpose of the main Java packages and Clojure namespace:
backtype.storm.coordination: Implements the pieces required to coordinate batch-processing on top of Storm, which both DRPC and transactional topologies use. CoordinatedBolt
is the most important class here.
backtype.storm.drpc: Implementation of the DRPC higher level abstraction
backtype.storm.generated: The generated Thrift code for Storm (generated using this fork of Thrift, which simply renames the packages to org.apache.thrift7 to avoid conflicts with other Thrift versions)
backtype.storm.grouping: Contains interface for making custom stream groupings
backtype.storm.hooks: Interfaces for hooking into various events in Storm, such as when tasks emit tuples, when tuples are acked, etc. User guide for hooks is here.
backtype.storm.serialization: Implementation of how Storm serializes/deserializes tuples. Built on top of Kryo.
backtype.storm.spout: Definition of spout and associated interfaces (like the SpoutOutputCollector
). Also contains ShellSpout
which implements the protocol for defining spouts in non-JVM languages.
backtype.storm.task: Definition of bolt and associated interfaces (like OutputCollector
). Also contains ShellBolt
which implements the protocol for defining bolts in non-JVM languages. Finally, TopologyContext
is defined here as well, which is provided to spouts and bolts so they can get data about the topology and its execution at runtime.
backtype.storm.testing: Contains a variety of test bolts and utilities used in Storm's unit tests.
backtype.storm.topology: Java layer over the underlying Thrift structure to provide a clean, pure-Java API to Storm (users don't have to know about Thrift). TopologyBuilder
is here as well as the helpful base classes for the different spouts and bolts. The slightly-higher level IBasicBolt
interface is here, which is a simpler way to write certain kinds of bolts.
backtype.storm.transactional: Implementation of transactional topologies.
backtype.storm.tuple: Implementation of Storm's tuple data model.
backtype.storm.utils: Data structures and miscellaneous utilities used throughout the codebase.
backtype.storm.bootstrap: Contains a helpful macro to import all the classes and namespaces that are used throughout the codebase.
backtype.storm.clojure: Implementation of the Clojure DSL for Storm.
backtype.storm.cluster: All Zookeeper logic used in Storm daemons is encapsulated in this file. This code manages how cluster state (like what tasks are running where, what spout/bolt each task runs as) is mapped to the Zookeeper "filesystem" API.
backtype.storm.command.*: These namespaces implement various commands for the storm
command line client. These implementations are very short.
backtype.storm.config: Implementation of config reading/parsing code for Clojure. Also has utility functions for determining what local path nimbus/supervisor/daemons should be using for various things. e.g. the master-inbox
function will return the local path that Nimbus should use when jars are uploaded to it.
backtype.storm.daemon.acker: Implementation of the "acker" bolt, which is a key part of how Storm guarantees data processing.
backtype.storm.daemon.common: Implementation of common functions used in Storm daemons, like getting the id for a topology based on the name, mapping a user's topology into the one that actually executes (with implicit acking streams and acker bolt added - see system-topology!
function), and definitions for the various heartbeat and other structures persisted by Storm.
backtype.storm.daemon.drpc: Implementation of the DRPC server for use with DRPC topologies.
backtype.storm.daemon.nimbus: Implementation of Nimbus.
backtype.storm.daemon.supervisor: Implementation of Supervisor.
backtype.storm.daemon.task: Implementation of an individual task for a spout or bolt. Handles message routing, serialization, stats collection for the UI, as well as the spout-specific and bolt-specific execution implementations.
backtype.storm.daemon.worker: Implementation of a worker process (which will contain many tasks within). Implements message transferring and task launching.
backtype.storm.event: Implements a simple asynchronous function executor. Used in various places in Nimbus and Supervisor to make functions execute in serial to avoid any race conditions.
backtype.storm.log: Defines the functions used to log messages to log4j.
backtype.storm.messaging.*: Defines a higher level interface to implementing point to point messaging. In local mode Storm uses in-memory Java queues to do this; on a cluster, it uses ZeroMQ. The generic interface is defined in protocol.clj.
backtype.storm.stats: Implementation of stats rollup routines used when sending stats to ZK for use by the UI. Does things like windowed and rolling aggregations at multiple granularities.
backtype.storm.testing: Implementation of facilities used to test Storm topologies. Includes time simulation, complete-topology
for running a fixed set of tuples through a topology and capturing the output, tracker topologies for having fine grained control over detecting when a cluster is "idle", and other utilities.
backtype.storm.thrift: Clojure wrappers around the generated Thrift API to make working with Thrift structures more pleasant.
backtype.storm.timer: Implementation of a background timer to execute functions in the future or on a recurring interval. Storm couldn't use the Timer class because it needed integration with time simulation in order to be able to unit test Nimbus and the Supervisor.
backtype.storm.ui.*: Implementation of Storm UI. Completely independent from rest of code base and uses the Nimbus Thrift API to get data.
backtype.storm.util: Contains generic utility functions used throughout the code base.
backtype.storm.zookeeper: Clojure wrapper around the Zookeeper API and implements some "high-level" stuff like "mkdirs" and "delete-recursive".