Bonsai Store - F# based reporting
Bonsai Store is a project with the purpose of providing efficient reporting over sets of items of user defined data types, via an F# interface. A Bonsai Store can be constructed programmatically over any data type. If the data type contains indexes (annotated member methods), a tree-based (hence the name)data structure is constructed internally, enabling efficient filtering.
Build
Unix:
$ sh build.sh All
Windows:
$ build.cmd All
Rationale
The motivation behind the project is to provide a simple interface for constructing aggregation based reports while maintaining good performance.
Instead of using an external query language (such as SQL
), queries are constructed in F#
and the data is stored in memory.
The following features are emphasized:
- Simple semantics - A BonsaiStore is conceptually just a set of items.
- Reports may be constructed by composing well known operations over sequences such as
filter
,map
andfold
. - Queries are statically type checked.
- Reusability - Query templates can be extracted and packaged as libraries.
Bonsai Store is particularly suited for scenarios where you have a large enough data set to make required filtering expensive and small enough to fit in memory. Between 1 and 100 million data points with a diverse set of filtering criteria would be a good candidate.
An example
To give a concrete example of how Bonsai Store can be used, consider the following data types:
open FSharp.BonsaiStore
type Employee = { EmployeeId : int ; Name : string; Team : string }
type Product = { ProductId : int; Category : string ; Name : string }
type SalesItem =
{
Product : Product
Quantity : int
Price : float
Date : System.DateTime
Employee : Employee
}
[<Index; Level(0)>]
member private this.DateIx() =
let span = this.Date - DateTime(1900,1,1)
span.Days / 100
[<Index; Level(1)>]
member private this.PriceIx() =
int (this.Price / 300.)
[<Index; Level(2)>]
member private this.PersonNameIx() =
this.Employee.Name.GetHashCode() % 100
[<Index; Level(3)>]
member private this.CategoryIx() =
this.Product.Category.ToString().GetHashCode()
The SalesItem
type represents an sales event and contains some basic information such as the employee responsible for the sale, the price, quantity and product.
The type is also enhanced with some private methods annotated with an Index
attribute. These methods define the indexes to be used when constructing a bonsai store over the data type. For instance the DateIx
method signals that items should be partitioned into groups where each items in a group are at most 100 days apart considering their Date
property. The Level(0)
flag means that this condition will constitute the top level in a hierarchical set of partitions. The second level will split on price intervals of 300.
Constructing BonsaiStores
Given a list of sales items, a Bonsai Store can be constructed:
module SB = FSharp.BonsaiStore.StoreBuilder
let store = SB.buildDefaultStore items
The SB.buildDefaultStore
examines the index structure and builds a hierarchal tree where each level corresponds to an index (and the order defined by the level attribute if given).
Basic reporting
Reports may be defined in a map-reduce fashion. Here's a very simple example counting the total number of elements given an arbitrary filter:
module R = FSharp.BonsaiStore.Reporting
let totalNumSalesItems filter = R.report store filter 0 (fun _ -> 1) Array.sum
Here's the type of the report
function:
val report<'T,'R> (store: IStore<'T>)
(filter: Expr<'T -> bool>)
(map: 'T -> 'R)
(reduce: 'R [] -> 'R)
: 'R =
The first argument is a store object (any type implementing the IStore
interface will do).
The second argument is a filter condition in the form of an Expr<'T -> bool>
.The reason for using an expression rather rather than a plain function is that expressions can be dissected and mapped to existing indexes. From a users perspective it's sufficient to think of it as a simple predicate.
The third argument is a map
function constructing a report from a single item.
The last parameter is the a reduce function folding an array of reports into a single report.
In order for the semantics of report generation to consistent, the type of a report ('R
in the signature above) needs to form a commutative monoid; That is an algebraic structure consisting of a binary operation which is commutative and associative, along with an identity element.
Given a report type R
, with identity id
and operator *
, the following conditions must hold:
a * id = a , for every a in 'R (identity)
a * b = b * a , for every a,b in 'R (commutativity)
a * (b * c) = (a * b) * c, for every a,b,c in 'R (associativity)
The reduce function encodes the identity element as well as the binary operator:
id = reduce [||]
reduce = Array.fold (*) id
The signature of reduce also enables a more efficient implementation of folding a sequence of elements.
Considering the example above: R.report store filter 0 (fun _ -> 1) Array.sum
The reduce function is Array.sum
which can be rewritten into a fold:
Array.sum xs = Array.fold (+) 0
unveiling the corresponding id
and (*) operator. We also need to verify that the conditions for commutativity, associativity and identity are valid:
a + 0 = a
a + b = b + a
a + (b + c) = (a + b) + c
There are many other useful report types that meet the above criteria. A few includes, numbers, tables and unordered lists.
totalNumSalesItems
can now be used to count the elements using any filter condition. Here is an example of counting the number of items with price greater than 100 during the year 2010:
let res =
totalNumSalesItems <@ fun item -> item.Price > 100. && item.Date.Year = 2010 @>
Note that the filter expression is given as an F# quotation (code snipptes surrounded by <@ and @>). The reason why a quotation is required is that it can be deconstructed and transformed into a more efficient instruction for how to select elements from the store.
Here is another example of a slightly more interesting table based report template:
let tableReport pick salesStore filter =
SB.report salesStore filter Table.empty (fun item -> Table.fromSeq [pick item]) Table.merge
tableReport
accepts an arbitrary function for selecting a key-value pair from an item and will generate a table aggregating the values with identical keys. The function may be used for defining a top teams reporting template.
let topTeams =
tableReport (fun item -> item.Person.Team, float item.Quantity * item.Price)
Finally the topTeams
report may be instantiated with a filter:
let topTeamsRes =
topTeams salesStore <@ fun item ->
item.Date.Year = 2011 && item.Date.Month = 7 @>
Next steps
The current interface support the construction of immutable data-stores. A service interfaces for propagating changes between multiple clients will be introduced.