FlowerBI supports querying relational databases through a single POST route, so that clients have enough power to do aggregation and joins, but the server/API can can carefully limit what clients are able to do.
It focuses on supporting very succinct query definitions at the client, and strong-typing via TypeScript inference.
Logo designed by @shep1987
Our users' data is:
- in SQL Server databases,
- arranged in a star schema,
- served up by our own API (dotnet core/C#),
- subject to fine-grained row-level authorisation (that is, only some users are allowed to see some rows).
Our UI is:
- written in TypeScript/React,
- required to show aggregated statistics in nice-looking charts,
- driven by rapid evolution and enhancement from user feedback.
We've tried using a paid no-code BI product, and while it had some severe drawbacks, we liked the way it worked with data:
- In one central place, you describe the schema: a fact table that has several foreign keys to dimension tables, all many-to-one relationships.
- For each chart you want to render, you just specify columns to group by (from any table) and columns to aggregate over (count, sum, etc.)
- Something builds the SQL query for you - this part isn't really that hard, but it's very productive to have it automated.
There are a lot of free libraries for drawing charts: chart.js is really easy to use, and it has a good React wrapper.
But it would be a drag to have to write a separate API route that runs a different specific SQL query to get the data for each chart. FlowerBI takes away that problem, and there's honestly not a lot to it.
Define the shape of the data you need in your client code:
const { records } = useQuery(fetch, {
select: {
customer: Customer.CustomerName,
bugCount: Bug.Id.count()
},
filters: [ Workflow.Resolved.equalTo(true) ]
});
The useQuery
function is a handy React hook, defined in the flowerbi-react
npm package, but the core client code in the flower-bi
package has no dependency on React, so it's not a prerequisite.
You supply the fetch
function to call your API, with your choice of authentication. Inside your API you pass a chunk of JSON to FlowerBI.Engine
and it performs the SQL query.
Render the returned records easily in React, maybe using chart.js:
<Pie data={{
labels: records.map(x => x.customer),
datasets: [{
label: "Bugs",
data: result.records.map(x => x.bugCount)
}]
}} />
The record fields customer
and bugCount
are strongly typed, inferred from the select
in the query.
Obviously it's not safe to allow clients to send raw SQL to an API and get it executed, so that's not happening here. The query refers to tables/columns (such as Customer.CustomerName
) that are defined inside the API, declared in C# like this:
[DbTable("Customer")]
public static class Customer
{
public static readonly PrimaryKey<int> Id = new PrimaryKey<int>("Id");
public static readonly Column<string> CustomerName = new Column<string>("CustomerName");
}
[DbTable("Bug")]
public static class Bug
{
public static readonly PrimaryKey<int> Id = new PrimaryKey<int>("Id");
public static readonly ForeignKey<int> CustomerId = new ForeignKey<int>("CustomerId", Customer.Id);
// other columns
}
FlowerBI.Tools
automatically reflects over this structure and generates a TypeScript file that the client can use to get auto-completion and type inference in its queries. So the client code can query the data in a creative and flexible way, but only within the boundaries set by your API's schema definition. Your API can also easily add extra filters to the query, to impose "row-level security" on a per-user basis.
Our example query casually takes from two different tables:
select: {
customer: Customer.CustomerName,
bugCount: Bug.Id.count()
}
The schema tells us that Bug
has a foreign key CustomerId
to the Customer
table, so we're going to need a join
. Also the bugCount
column uses an aggregation function, count
, which means that we're going to group by
the customer name to get the number of bugs reported per customer.
You can specify any number of plain columns (strings, numbers, dates, booleans) to group by, and any number of numeric columns with an aggregation function (count
, sum
, average
).
You can also optionally supply different filters to each aggregation, so e.g. get the count of all bugs, and the count of resolved bugs, per customer.
select: {
customer: Customer.CustomerName,
countAllBugs: Bug.Id.count(),
countResolvedBugs: Bug.Id.count([Workflow.Resolved.equalTo(true)])
}
The result has records like this:
{
customer: string,
countAllBugs: number,
countResolvedBugs: number
}
Perfect for mapping to a multi-bar chart.
https://earwicker.com/flowerbi/demo/
This runs the whole stack in-browser, using some WASM-based components. This is not part of the real solution; no WASM is needed. It's just a way to run a live demo without having to pay to run real boxes!
- sql.js representing the RDBMS
- The dotnet core engine built in Blazor, representing an application server
- The UI "fetches" data from the Blazor app, which uses
FlowerBI.Engine
to generate SQL queries and runs them against sql.js. It does some ugly hackery to make the queries compatible, as they are currently generated to target Microsoft SQL Server which has a different syntax for many basic things.
Gradually appearing:
Copyright © 2020 Daniel Earwicker
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.