Skip to content

Latest commit

 

History

History
175 lines (122 loc) · 7.91 KB

README.md

File metadata and controls

175 lines (122 loc) · 7.91 KB

FlowerBI

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.

The Playground lets you interactively build a query, see what the JS/TS code for that query would look like, and also the generated SQL, and the tabulated results.

Playground animated GIF

Logo designed by @shep1987

Use Case

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.

Flexible and creative querying at the client

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.

Easy mapping to widely-used visualisation libraries

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.

Lock down the schema

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 Yaml like this:

Customer:
    id:
        Id: [int]
    columns:
        CustomerName: [string]

Bug:
    id:
        Id: [int]
    columns:
        CustomerId: [Customer]
        ReportedDate: [DateReported]

FlowerBI.Tools automatically generates TypeScript and C# files 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.

Automatic joins, grouping and aggregation

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.

Live Demo

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.

Reference Documentation

Gradually appearing:

Contributors

License (MIT)

Copyright © 2020-2022 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.