Description: | A MongoDB focused application toolkit |
---|---|
Author: | Bryan Reinero <breinero@gmail.com> |
Firehose is a instrumented, multithreaded execution framework which you can use to benchmark your own applications. It allows you to do sophistated testing easily.
- Firehose includes these major components:
- A simple code instrumentation and reporting library
- A customizable command line interface builder
- A multithreaded worker pool
- An application framework so that you may use all of these components together for your own load testing purposes
- NEW! Circuit Breaker package to protect staturation conditions
You can use Firehose to generate custom load simply by providing a class which implements the Executor interface Firehose calls this Executor's execute method to do a single unit of work. Take a look at the DSV import class to see an example implementation.
Ok, so I want to read a CSV file and import those records into MongoDB as fast as I can. To get that file into MongoDB I will need to execute a three step process on each line of the file;
- Read the next line from the file
- Parse the line, converting it into a object prepped for insertion
- Insert the new object into MongoDB
As a curious and conscientious software engineer, I am very interested to know how much time each of these steps take so that I can establish performance baselines. I can use Firehose's instrumentation library to mark the start and end of each step with use of the Interval class class. For instance, here's how I determine how long and individual insertion takes.
Interval insertDuration = samples.set("insert"); // Set the time marker dao.insert( object ); // perform the actual work insertDuration.mark(); // mark the operation as complete
The insertDuration Interval is collected and averaged automatically by Firehose giving me mean latency of the insertion operation. Firehose will then pretty-print the running average to the console so that I may see how fast my operations are executing in real time.
Firehose pretty prints this output, ( refreshing the console each second)
{ threads: 2, "lines read": 100000, samples: { units: "microseconds", "reporting interval ms": 5, total: { mean: 221.32, median: 181.5, std: 82.68293015354203, count: 50, total: 11066.0 }, readline: { mean: 4.160000000000002, median: 1.0, std: 14.412806097073416, count: 50, total: 208.0 }, build: { mean: 14.291666666666666, median: 15.0, std: 3.439188012365297, count: 48, total: 686.0 }, insert: { mean: 170.28571428571428, median: 158.0, std: 39.36368885152915, count: 49, total: 8344.0 } } }
This output tells me that inserts are taking an average of 170 microseconds, as averaged over a time interval of 5 milliseconds. During this interval I inserted 49 documents. As the output is printed in JSON format I can insert these stats into MongoDB for benchmarking analysis!
You can take a look at how this workload is processed here
Firehose uses Java Management Extensions (JMX) to make it even easier monitor and manage your application. Firehose exposes Interval data with MBeans so you can gather performance metrics from a remote JMX client such as JConsole. Additionally The worker pool can be stopped, started or resized at will from the remote client without having to restart the application. More information on JMX is available in the documentation.
Under the hood, Firehose uses the Apache Commons CLI library to parse command line options passed in at runtime. Firehose wraps the Commons CLI into the framework such that we can configure my own set of command line options easily. Using the CLI framework is a two step process.
- Declare command line options in a properties file
- Assign callback methods to handle the input
As an example let's take a look at the usage for Firehose's DSV Import feature to see how it uses the Commons CLI:
option | long form | type | description |
---|---|---|---|
-cr | --noPretty | print out in CR-delimited lines. Default is console mode pretty printing (when possible) | |
-f, | --file | <filepath> | filename to import |
-fs, | --fsync | write concern: wait for page flush | |
-h, | --headers | <name:type> | ',' delimited list of columns |
-j, | --journal | enable write concern wait for journal commit | |
-m, | --mongos | <host:port> | ',' delimited list of mongodb hosts to connect to. Default localhost:27017 |
-ns, | --namespace | <namespace> | target database and collection this work will use (format: 'db.col') |
-pi, | --printInterval | <seconds> | print output every n seconds |
-ri, | --reportInterval | <seconds> | average stats over a time interval of i milliseconds |
-t, | --threads | <threads> | number of worker threads. Default 1 |
-v, | --verbose | Enable verbose output | |
-wc, | --writeConcern | <concern> | write concern. Default = w:1 |
To generate these options I first declared the options I wanted to use inside my options.json file. Here's a snippet of the file:
{ "application": "Firehose", "options": [ { "op": "m", "longOpt" : "mongos", "name": "hostname:port", "description": "',' delimited list of mongodb host to connect to. Default localhost:27017,", "args": "multi", "separator": "," }, { "op": "f", "longOpt" : "file", "name": "file", "description":"filename to import (full path)", "required": true, "args" : 1 }, { "op": "t", "longOpt" : "threads", "name": "threads", "description": "number of worker threads. Default 1", args: 1 } }
Firehose will read this file at application start up, creating the specific command line options I need to run the application. Now, all I need to do is define a set of callbacks which handle the processing of my command line options when a user actually runs the DSV Import tool. For example, here's the callback for handling input on the "-t" (or number of worker threads in the pool) option.
cli.addCallBack("t", new CallBack() { @Override public void handle(String[] values) { numThreads = Integer.parseInt(values[0]); } });
You can examine more callback examples in the code.
java -jar target/Firehose-0.1.0.one-jar.jar -f test.csv -d , -ns test.firehose -h _id:objectid,count:float,sum:float,name:string -t 2
This command line invokes Firehose with 2 threads, parsing a CSV file of 4 columns. Each column is to be translated into json fields named "_id", "count", "sum" and "name", of types ObjectId, float, float, string respectively.
Firehose's application framework is made for standing up simple load tests quickly. As such, it comes with a set of command line options fully configured for control of the worker pool, instrumentation library, and access to MongoDB. Users of the application framework need only add:
- Any extra command line options specific to their application
- An instance of Executable which the worker pool calls as a unit of work
Let's again use the DSV import tool as an example. The application framework is initialized inside Firehose's constructor. The first step is to define the appropriate command line interface callbacks I need to handle user input.
public Firehose ( String[] args ) throws Exception { Map<String, CallBack> myCallBacks = new HashMap<String, CallBack>(); // custom command line callback for csv conversion myCallBacks.put("h", new CallBack() { @Override public void handle(String[] values) { for (String column : values) { String[] s = column.split(":"); converter.addField( s[0], Transformer.getTransformer( s[1] ) ); } } }); // custom command line callback for delimeter myCallBacks.put("d", new CallBack() { @Override public void handle(String[] values) { converter.setDelimiter( values[0] ); } }); // custom command line callback for the input file parameter myCallBacks.put("f", new CallBack() { @Override public void handle(String[] values) { filename = values[0]; try { br = new BufferedReader(new FileReader(filename)); }catch (Exception e) { e.printStackTrace(); System.exit(-1); } } });
Remember, the Application class has already defined CLI callbacks for the worker pool, instrumentation engine and MongoDB driver. All I needed to add where the callbacks for the input file, value delimiter and column headers. I've defined these callbacks as a collection of anonymous functions which I pass to the Application class' constructor:
worker = Application.ApplicationFactory.getApplication(this, args, myCallBacks);
- The Application class' constructor takes 3 parameters
- A class which implements Executor
- A String array of the command line options
- A list of custom command line callbacks
Bingo. I'm ready to rock and roll. Notice that the 'this' in the first parameter refers to an instance of the Firehose class, which implements Executable. The overridden execute() method is where all the work is done.
Firehose's circuit breakers watch for trigger conditions that trip the breaker automatically. This protects both the downstream server from overwhelming surges in load and the requesting client who has the circuit breaker. This is because once the circuit breaker has been tripped, the client application can respond appropriately. As opposed to simply hanging on its own pending requests to complete while the inbound load exhausts stack, heap, CPU or other resources.
More detail is available here
I've included a CSV file generator called RandomDSVGenerator so that you may test your build and see Firehose in action with minimal effort. Simply run the following commands from the the command line prompt.
$ ./gradlew jar $ java -cp applications/DSVImport/build/libs/DSVImport-<VERSION>-SNAPSHOT.jar com.bryanreinero.dsvload.RandomDSVGenerator -f test.csv -n 10000 $ java -cp applications/DSVImport/build/libs/DSVImport-<VERSION>-SNAPSHOT.jar com.bryanreinero.dsvload.Firehose -f test.csv -d , -h _id:objectid,count.0:float,count.1:float,name:string
As a consultant, I often advise my clients to instrument their application code such that they have a baseline of performance metrics. Getting baselines is extremely useful in identifying bottlenecks, understanding how much concurrency your application can handle, determining what latency is "normal" for the application, and indicating when performance is deviating from those norms.
While most developers will acknowledge the value of instrumentation, few actually implement it. So to help them along, Firehose was designed with some basic instrumentation boiled right into it.
Firehose is supported and somewhat tested on Java 1.7
- Additional dependencies are:
Copyright (C) {2013} {Bryan Reinero}
This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
This software is not supported by MongoDB, Inc. under any of their commercial support subscriptions or otherwise. Any usage of Firehose is at your own risk. Bug reports, feature requests and questions can be posted in the Issues section here on github.
- Accept piped input from stdine
- Write Javadocs
- Accept json input
- Accept mongoexport formated csv's
- fix README formatting