There are many reasons to save user data in a Molnify application. Dynamic product lists, usage logging, issue statuses, and sales data, are just a few example cases where saving to the database is key.
By connecting your application with the database, you can store, modify and retrieve data without updating your application.
When you upload an Excel file or link a Google spreadsheet to Molnify, you provide Molnify with instructions for how to create your application. But Molnify does not update the instruction file once it is in use, it is only used by you when you wish to upload a new version of the application. For instance, two users can use the same application simultaneously, without the risk of their inputs "clashing" and overriding each other. Because the data in the instruction file cannot change over the course of an application's lifespan, we call that data "static", i.e. non-changing.
On the other hand, if a user activates an action to store data in the application's database, then that data will be available even after the user has closed the application (and, depending on your instruction file, other users of your application may be able to access the data) Since the database data may be modified without updating the app, we call the data in the database "dynamic", i.e. changing.
Before you start using Molnify's Database functionalities in your application, you must ensure that you have a static (non-changing) application ID. You have this if your application has the ID metadata set up. If you do not have the ID set up, you will lose access to all your table data whenever you update your application.
Once you have the ID metadata included in your application, then you are ready to unleash the power of Molnify's database!
The database, Molnify autofills, and other instructions that interact with the database use MySQL.
The Data Manager is used to configure your database connection, and you can access it from My Applications, or from your application's sidebar menu. The data manager is accessible only to the application's uploader and its superusers.
The Data Manager consists of two tools: the table management interface and the database query tester.
The table management interface lets you create and edit tables for your application. You can also use it to edit to the data stored in your tables.
To create a table, click on the "Create New Table" button.
A new table will be automatically created and assigned to your application.
The new table will be given a name of the form data_<your-application-id>_
followed by a number, and this name cannot be changed.
The actions that add or modify data to your tables require the columns _molnify_user_ip
,
_molnify_user_email
, _molnify_application_id
and _molnify_timestamp
to exist in the table. So, they are included automatically whenever you create a new table.
Click on the name of a table which you have created to bring up its details in the right portion of the window.
Once you have opened up a table in the detailed view, you can inspect its data in a searchable view. Note that if you just created your database table, then it will not contain any data initially.
You can use the buttons above the table to both import from and export the table's data to a CSV file. You will be prompted to match the columns in your CSV to the columns of the table.
The import of the CSV is not an atomic transaction. That means, if there is an error on one of the rows to insert, the rows without errors will still be imported.
The fields below the table allow you to filter rows by column values. There is also a global search field, which filters over all columns in the table. The filters and search are cumulative, that is if you filter over two columns the table only displays the rows satisfying both conditions.
The search and filtering can be used as follows:
monkey
yields the same results as searching for mONkEy
Apple Banana
is the same as searching for Banana Apple
"Apple Banana"
does not give the same results as "Banana Apple"
. This method is still case-insensitiveboat !houseboat
will find all rows containing "boat" but exclude any containing "houseboat"In addition to reading the data, you can also insert new rows or delete selected rows. You can select a row or several by clicking on the checkbox in the rightmost column on its row. Click on an entry in order to edit it - any changes will be saved to the database as soon as you deselect the entry.
Unlike Excel, there is no undo-functionality in the database. That means that any changes to your tables' structure or data cannot be reverted.
The shown statistics are, unlike the other views on this page, not editable.
Each table has a counter associated with it, the Auto Increment
.
If your table has a column which is the primary key with AUTO_INCREMENT
enabled,
then the value of this counter is the next new value for that column.
You can use this table for adding, editing and removing columns in the same way as you use the data table above it.
A table's columns defines the structure of its data.
For example, if there is a column named variable
, then each
row of data may have a value for variable
.
Column names are ideally descriptive, and we recommend that you avoid
names that do not describe their function.
For example, we consider the name max_acceleration
to be
easier to work with than the abbreviated ma
.
If a table's columns define its data, then a column's data type defines the data in the column. The data type is a label that instructs the database on how it shall interpret the 1s and 0s that constitute the data, how much memory it should reserve for the data, and more.
Data type | Use case |
---|---|
INT UNSIGNED | Non-negative whole numbers |
INT | Whole numbers that may be negative, recordId |
BOOL | Values that are either TRUE/FALSE. The database converts BOOL
columns to TINYINT, where 0 is FALSE and all other values are TRUE |
FLOAT | Decimal numbers |
TIMESTAMP | Date and time of an audit event, or other logging |
VARCHAR | Text strings, up to 255 characters |
TEXT | Text strings longer than a VARCHAR |
The actions that insert data into the database require the columns _molnify_user_ip
(VARCHAR),
_molnify_user_email
(VARCHAR), _molnify_application_id
(VARCHAR)
and _molnify_timestamp
(TIMESTAMP).
They are included in newly created tables by default.
You can set the default value of a column as well. Then, any row of data that is inserted into the table and does
not have a value specified for that column will be assigned the default value. For example, you can use a BOOL column
is_deleted
with a default value of 0
to initialise data as "active", and then use an
add row to database action
to set is_deleted
to 1
when the user wishes to remove an entry.
This is a technique for soft-removing data, as no application can actually remove data from the database.
An index is an instruction for the database to prepare the stored data for lookup, and how.
For example, if your application stores a list of cars, then an index on the column "brand" may make the query
SELECT model FROM `data_<your-application-id>_0` WHERE brand = "VOLVO";
perform better, at the expense
of very slightly slowing down insertion of data. For most tables, the performance benefit is too small,
but it can be a critical optimization for tables with thousands of rows of data.
While it is technically not an index, you can define a "primary key" for your database table by creating an index with the
name PRIMARY
.
A primary key is a constraint on the data, and all rows of the table must have a unique value for the primary key column(s).
We recommend that you define a primary key for your table, as it has the following benefits:
By default, only the application that created a database table can read from and write to it. However, there are scenarios where sharing table access between applications is useful. For example, one application might monitor or analyze data collected by another, or multiple applications might need to access the same product catalog.
This interface allows you to grant or revoke access to a table. The application which created the table may never lose access to it, and you need to have superuser or uploader access to both applications in order to perform a change.
Keep in mind that an application with table access has full control over it, including the ability to delete data and even permanently delete the table.
Sometimes data does not load into your application as you expect, or you need to experiment with different queries to figure out how to retrieve the data your application needs. You can use the query tester to try out and validate your queries before including them in your Molnify application.
Note that the query tester is used for testing queries, and not a mix of Excel functions and queries.
For example, the following query would result in an error, as it is an Excel formula that attempts to read
a value from a spreadsheet cell:
="SELECT value, recordId FROM `data_<your-application-id>_0` WHERE _molnify_user_email = '"&B2&"';"
.
You will need to replace any cell references in your query with data before using it in the tester.
The tester history lets you load previously used queries into the tester. Your query history is saved on your device and stays available between sessions. Each application's tester history is kept separate, so queries tested in one application won’t appear in another application's history.
When writing in the tester area, you can use the keyboard shortcut ctrl/cmd + enter to test the query.
Error | Explanation |
---|---|
General troubleshooting tip for complex queries | Simplify your query, and verify that each individual component of the query works. Then, slowly assemble it back together, and see when it breaks. That is where your issue most likely is. |
"You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ..." | The database does not know how to interpret the query. For example, a table name which includes dashes (-) must be enclosed in backticks (`), or there might be a comma (,) after the last item in the SELECT. But, the database does not give you more detail than on which line it failed. So you can split your query into multiple lines, to narrow down on the issue. |
Unknown column 'column' in 'field list' | You are attempting to use a column that does not exist. Check that you have spelled the column name correctly, and that the column exists in the table. |
User 'you@email.com' and application 'application-id' are not allowed to execute query: | For security reasons, you are only allowed to execute READ-ONLY queries, that only affect the columns in the tables which your application has access to. All of those tables are listed in the Table manager. Contact us if you are not permitted to execute a query which you believe you should be allowed to execute. |
Molnify's Autofill functionality is a powerful tool for using data from the database in your application. It allows you to specify a range of cells, and a query to retrieve data from the database. Molnify will then populate the cells with the data from the query, allowing you to use the data in Excel formulas and in outputs.
Name the range that makes out the table contents (i.e., B2:C3 in this case). Use this reference in your autofill to have the table contents replaced by the data returned from the database table.
This metadata would run the SQL query (="SELECT * from data_myTable"
) and populate the named range (data) with the results.
Here follows some example queries to get you started. More complex queries than these are supported, as well.
Query | Description |
---|---|
SELECT * FROM `data_<your-application-id>_0` | Select all data in all rows from the table |
SELECT `colour`, `value` FROM `data_<your-application-id>_0` | Select only specific columns from the table |
SELECT * FROM `data_<your-application-id>_0` WHERE `colour` = 'red' | Select only rows from a table where a column matches a condition |
SELECT * FROM `data_<your-application-id>_0` WHERE `colour` = 'red' AND `value` > 5 | Select only rows from a table based on multiple conditions |
SELECT * FROM `data_<your-application-id>_0` WHERE `colour` = 'red' ORDER BY `_molnify_timestamp` DESC | Order the result based on the values in a column |
SELECT * FROM `data_<your-application-id>_0` WHERE `colour` = 'red' ORDER BY `_molnify_timestamp` DESC LIMIT 10 | Fetch the 10 most recent rows of data that match the condition |
SELECT COUNT(*) FROM `data_<your-application-id>_0` WHERE `value` < 5 | Count the number of rows that match the condition |
SELECT SUM(`value`) FROM `data_<your-application-id>_0` WHERE YEAR(`_molnify_timestamp`) = 2024 | Count the number of rows that were created during a particular year |
SELECT DISTINCT(`customer`) FROM `data_<your-application-id>_0` | List all different customers that have data in the table |
You can use the following actions in order for your application to add new data or update the data in your database table.
A record is a snapshot of your application's variables. Unlike a scenario, which captures all inputs, a record captures all variables which have corresponding column names in the application's records table, and stores them in a new row.
You must use the RecordTableName metadata to specify the name of the table that will contain the records for your application.
The database table for storing records must also be set up as follows:
_molnify_user_ip
, _molnify_user_email
,
_molnify_application_id
and _molnify_timestamp
must exist in the table.
recordId
, of a data type that can store whole numbers, such as INT
.
This name is, like all column names in the database, case-sensitive.
recordId
column.
recordId
column must also be set to auto increment
.
Once you have your database table set up, and metadata naming the table as your records table, you are ready to include the add record to database action in your application.
When a user activates this action, the current values of its variables are stored into a new row of your application's records table.
But, if your application has a variable named recordId
, and it has a value that is not 0 or empty when the user activates the action,
then the existing record corresponding to that ID will instead be updated.
After a record has been created, the recordId variable will be updated to the recordId of the newly created row.
This way, any subsequent saved records by the same user will update the previously saved record - until the webpage is refreshed or the application
is reset.
If your application also has a variable named recordId
, and you load the application with it set using a URL parameter,
then the data from the corresponding record will be loaded into the application.
The add record to database action does not offer many customization options, but it also does not require extensive configuration.
Option | Value | Description |
---|---|---|
type | addRecord |
This must be set to instruct Molnify to create an add record to database action. This option is required |
title | Any string | This sets the text displayed on the generated action button |
name | Any string | This sets an internal name for the action, and is used to integrate the action with e.g. the multiple action and JavaScript functions |
hidden | TRUE or FALSE |
Set this to TRUE if you want to hide the generated action button, and
instead activate the action with
JavaScript,
using e.g. custom buttons |
Example:
This example creates an add record action, since type
is set to addRecord
. The action button's title will be Add Record to database.
You can use the add row to database action for a configurable method of storing and updating database data. The action allows you to define which values should be stored in which columns and allows for a primary key consisting of several columns when updating data.
The columns _molnify_user_ip
, _molnify_user_email
,
_molnify_application_id
and _molnify_timestamp
are required
to exist in the table. Values for these columns are created automatically by the action.
Option | Value | Description |
---|---|---|
type | insertRow |
This must be set to instruct Molnify to create an add row to database action. This option is required |
tableName | data_<your-application-id> |
This must be set to instruct Molnify which table to insert the data into. This option is required |
.col | Any value | Replace "col" in the option name with the name of the database column to insert data into. A dot (.) must prefix the column name. The action requires at least one ".col" option |
k.keycol | Any value | If a database column is part of the primary key, replace "keycol" in the option name with the name of the column to insert data into. A "k" and a dot (k.) must prefix the column name. |
title | Any string | This sets the text displayed on the generated action button |
name | Any string | This sets an internal name for the action, and is used to integrate the action with e.g. the multiple action and JavaScript functions |
hidden | TRUE or FALSE |
Set this to TRUE if you want to hide the generated action button, and
instead activate the action with
JavaScript,
using e.g. custom buttons |
If you use the add row to database action to insert a row of data into
a database table where the table's primary key is exactly one column and
the key is auto-incremented, then Molnify will update the value of the
recordId
variable (if it exists) to the newly generated value.
Example:
This example creates an add row to database action, since type
is set to insertRow
. The action button's title will be Add Data. The values for the database columns 'key' (also primary key), 'col1', and 'col2' will be inserted into the database table data_my_table.
The download custom data action can be used to allow the users of your application to download the result of a query. The downloaded file can be in either .csv or .json format.
Example:
This example creates a download query action, since type
is set to downloadQuery
. The action button's title will be Download Query.The result of the MySQL query will be downloaded as a csv file, delimited by semicolons ;
).
An entire database table can be downloaded through the download database table action. Unlike the download custom data action, this action does not allow choosing file format, delimiter, or query.
This action is equivalent to the download custom data action with the query SELECT * FROM data_<your-application-id>_0
.
Example:
This example creates a download table action, since type
is set to downloadTable
. The action button's title will be Download Table, and clicking on it will download the table data_my_table.
The list of tables that your database may use is determined by its ID. As such, if you need to change your application's ID, you should either delete all your database tables for the application or contact us for help to migrate your data to your new application ID.
This example will walk you through how to build a logging tool in Molnify.
Logs can easily be abstracted to other concepts where your user stores transactions. For example, a log showing a product name, a quantity, and a price is the same as an order.
We pretend that we have a set of locations, each of which should be inspected regularly. An inspection should determine if a location is in "Good" condition, "Poor" condition (but serviceable), or if action is needed.
First things first: we set up our metadata with an ID for the application, a name, a description, a link to this guide, and a template for some quick styling. We place this in the columns A-B in our sheet. In general, we recommend that you structure your workbook so that you can find your metadata easily.
A | B | |
---|---|---|
1 | ID | database-example-logging |
2 | Name | Warehouse Logging Tool |
3 | Description | This application is a showcase of how a (fairly simple) logging tool can be implemented in Molnify. |
4 | Website | https://app.molnify.com/guides/database |
5 | Template | red dot |
Note that you will have to change the ID if you copy this example, as this ID is taken by the example application.
Now that our application has a static ID and some basic metadata, we can add some inputs to it.
First off, we need a list of names of our inspectors, and a list of warehouse locations. We place them in columns K and L, giving us some space to their left for inputs, outputs and actions. For brevity, we choose to have 5 inspectors and 10 locations, but the application could easily be extended to include more.
The states "Good", "Poor" and "Action needed" go into column M.
K | L | M | |
---|---|---|---|
1 | Names | Warehouses | States |
2 | Ivan Metcalf | Chepstow | Good |
3 | Renee Ellison | Tarrin | Poor |
4 | Aisha Morris | Wigston | Action needed |
5 | Barrie Henderson | Pitmedden | |
6 | Ewa Mistry | Laewaes | |
7 | Kirkwall | ||
8 | Woodpine | ||
9 | Rochdale | ||
10 | Peatsland | ||
11 | Auctermunty |
The names and warehouse locations are randomly generated.
Now, we assign these to inputs. We create a named range "Names" for the names, a named range "Warehouses" for the warehouse locations, and a named range "States" for the states. By enabling data validation for the inputs, and typing "dropdown" in their UI cell, we instruct Molnify that the inputs should be rendered as dropdown lists with options defined by the data validation ranges.
D | E | F | |
---|---|---|---|
1 | Name | dropdown | |
2 | Warehouse | dropdown | |
3 | Status | dropdown |
We use data validation to define the range of values which can be entered into the inputs: limit the accepted values of "Name" to those in "Names", the values of "Warehouse" to those in "Warehouses", and the values of "Status" to those in "States".
We can now upload the app, and we have some working dropdowns. But, there is no way to save a log yet.
In order to save the logs, we need to set up a database table. We do so in the Data manager.
In the Data manager, we click on the button to create a new table. It will be automatically assigned the name "data_<my-app-id>_0". After the table is created, we click on its name to open the configuration for our new table.
To store our log data, we create the following columns in the "Columns" table: recordId
, name
, warehouse
, status
.
We cannot set recordId to auto increment
unless we define it as the primary key first,
so we do that by creating an index with the name PRIMARY
.
After we add the primary key, we set recordId to be auto-incremented,
so new recordIds will be generated for new logs.
And that is all the database setup we will need. Now, we will use the add record to database action to add the ability to save logs to the application.
We add our newly created table as the application's records table in the metadata, and we create the action.
A | B | |
---|---|---|
1 | ID | database-example-logging |
2 | Name | Warehouse Logging Tool |
3 | Description | This application is a showcase of how a (fairly simple) logging tool can be implemented in Molnify. |
4 | Website | https://app.molnify.com/guides/database |
5 | Template | red dot |
6 | RecordTableName | ="data_"&B1&"_0" |
We use an Excel formula to calculate the RecordTableName. This way, if you change the application's ID, the RecordTableName changes as well.
H | I | |
---|---|---|
1 | type | addRecord |
2 | title | Saving log |
2 | name | save-log |
3 | hidden | TRUE |
5 | successUrl | ="https://app.molnify.com/app/"&B1 |
We use the successUrl action option to re-load the
application after we save a value, in order to clear the input fields.
Similarly to when we defined the metadata for RecordTableName, we use an Excel formula to calculate the URL in successUrl
.
For Molnify to know which inputs to save in which database columns,
we assign variable names to the inputs matching the column names. E.g. the input for name should have the ui-string
variable=name;dropdown
.
We also hide the default action button for the action. The default action button is visible on all input tabs, and we are limited in how we style it. So, we create a custom action button using an HTML Panel and place it among inputs.
D | E | F | |
---|---|---|---|
1 | Name | variable=name;dropdown | |
2 | Warehouse | variable=warehouse;dropdown | |
3 | Status | variable=status;dropdown | |
4 | <button class="btn btn-sm btn-success buttonMargin molnifyActionButton" onClick="performActionWithName('save-log');">Save log</button> | html;amonginputs;hidecopy; |
The button element uses the onClick attribute to call the hidden action when clicked on.
The classes btn btn-sm btn-success buttonMargin molnifyActionButton
are for styling.
Now that we're able to save logs, it is time to visualise the results. We want to display two tables. The first table should show the ten most recently made logs. The second table will show an overview of our locations, their last reported state, who made that report, and when the report happened.
First, we prepare a named range in the Excel file to read the most recent logs into.
We will display the 10 most recent logs, so the range should be 10 rows.
For each of the rows, we want to display when the log was made, who made it,
for which warehouse, and which state was logged - that is a total of 4 columns.
We name the cells O2:R11
"Recent_logs", and colour them grey
- the grey has no meaning in Molnify but helps us remember that we have
defined a named range there. We also write the column
headers "timestamp", "name", warehouse", and "status" above the named range,
to help us remember what data will be filled into which column.
Then, we create a table in the cells below the named range, N13:S23
.
In each cell of the table, we read the value of the corresponding cell in the named range,
or leave the cell blank if there is no read value. If we do not include the check for empty values,
then there could be rows displaying 0s if the result of the database query does not fill all rows.
N | O | P | Q | R | S | |
---|---|---|---|---|---|---|
13 | Warehouses | <b>Timestamp</b> | <b>Inspector</b> | <b>Warehouse</b> | <b>Status</b> | table |
14 | =IF(O2 <> "", O2, "") | =IF(P2 <> "", P2, "") | =IF(Q2 <> "", Q2, "") | =IF(R2 <> "", R2, "") | ||
15 | =IF(O3 <> "", O3, "") | =IF(P3 <> "", P3, "") | =IF(Q3 <> "", Q3, "") | =IF(R3 <> "", R3, "") | ||
16 | =IF(O4 <> "", O4, "") | =IF(P4 <> "", P4, "") | =IF(Q4 <> "", Q4, "") | =IF(R4 <> "", R4, "") | ||
17 | =IF(O5 <> "", O5, "") | =IF(P5 <> "", P5, "") | =IF(Q5 <> "", Q5, "") | =IF(R5 <> "", R5, "") | ||
18 | =IF(O6 <> "", O6, "") | =IF(P6 <> "", P6, "") | =IF(Q6 <> "", Q6, "") | =IF(R6 <> "", R6, "") | ||
19 | =IF(O7 <> "", O7, "") | =IF(P7 <> "", P7, "") | =IF(Q7 <> "", Q7, "") | =IF(R7 <> "", R7, "") | ||
20 | =IF(O8 <> "", O8, "") | =IF(P8 <> "", P8, "") | =IF(Q8 <> "", Q8, "") | =IF(R8 <> "", R8, "") | ||
21 | =IF(O9 <> "", O9, "") | =IF(P9 <> "", P9, "") | =IF(Q9 <> "", Q9, "") | =IF(R9 <> "", R9, "") | ||
22 | =IF(O10 <> "", O10, "") | =IF(P10 <> "", P10, "") | =IF(Q10 <> "", Q10, "") | =IF(R10 <> "", R10, "") | ||
23 | =IF(O11 <> "", O11, "") | =IF(P11 <> "", P11, "") | =IF(Q11 <> "", Q11, "") | =IF(R11 <> "", R11, "") |
So, we have set up the table to display data, but we need to supply the range Recent_logs
with information from the database.
To do so, we must write a query. The syntax for this is the MySQL
database query language. In other words, queries are generally structured
to begin with a SELECT
which specifies which columns to fetch,
and a FROM
which determines the database table to fetch from.
The FROM
can be followed by additional clauses, such as
WHERE
which filters the output, ORDER BY
which determines how the output is sorted, and LIMIT N
which only returns the first N rows. In this query, we will use the
ORDER BY
clause. Note that we do not need to use LIMIT N
,
since the named range is already limited in the number of rows.
SELECT
`_molnify_timestamp`,
name,
warehouse,
status
FROM
`data_database-example-logging_0`
ORDER BY `_molnify_timestamp` DESC
Now, the only thing that is left to do is to add the query as an autofill metadata.
A | B | |
---|---|---|
1 | ID | database-example-logging |
2 | Name | Warehouse Logging Tool |
3 | Description | This application is a showcase of how a (fairly simple) logging tool can be implemented in Molnify. |
4 | Website | https://app.molnify.com/guides/database |
5 | Template | red dot |
6 | RecordTableName | ="data_"&B1&"_0" |
7 | autofill.Recent_logs | ="SELECT `_molnify_timestamp`, name, warehouse, status FROM `data_"&B1&"_0` ORDER BY `_molnify_timestamp` DESC" |
Remember that the named range is called Recent_logs
,
so we set autofill.Recent_logs
to our query.
With a view of the most recent logs in place, it is time to show the results of the inspections.
Similarly to when we created the first table, we start by preparing the Excel
file to receive the data for the second table.
Since we have 10 warehouses, we create a named range that is 10 rows and 4 columns.
We let this named range be called Warehouse_data
, and define it to be the cells V2:Y11
.
We write the column headers "warehouse", "status", "name", "date" in V1:Y1
above the range. We also colour the range grey,
and colour them grey, so we remember that the cells will be populated with
data by Molnify.
Next, we create the Molnify table in the cells below the named range,
U13:Z23
. We refer each cell in the table to the corresponding
cell in the named range above it.
U | V | W | X | Y | Z | |
---|---|---|---|---|---|---|
13 | Warehouses | <b>Location</b> | <b>Status</b> | <b>Inspector</b> | <b>Date of report</b> | table |
14 | =IF(V2 <> "", V2, "") | =IF(W2 <> "", W2, "") | =IF(X2 <> "", X2, "") | =IF(Y2 <> "", Y2, "") | ||
15 | =IF(V3 <> "", V3, "") | =IF(W3 <> "", W3, "") | =IF(X3 <> "", X3, "") | =IF(Y3 <> "", Y3, "") | ||
16 | =IF(V4 <> "", V4, "") | =IF(W4 <> "", W4, "") | =IF(X4 <> "", X4, "") | =IF(Y4 <> "", Y4, "") | ||
17 | =IF(V5 <> "", V5, "") | =IF(W5 <> "", W5, "") | =IF(X5 <> "", X5, "") | =IF(Y5 <> "", Y5, "") | ||
18 | =IF(V6 <> "", V6, "") | =IF(W6 <> "", W6, "") | =IF(X6 <> "", X6, "") | =IF(Y6 <> "", Y6, "") | ||
19 | =IF(V7 <> "", V7, "") | =IF(W7 <> "", W7, "") | =IF(X7 <> "", X7, "") | =IF(Y7 <> "", Y7, "") | ||
20 | =IF(V8 <> "", V8, "") | =IF(W8 <> "", W8, "") | =IF(X8 <> "", X8, "") | =IF(Y8 <> "", Y8, "") | ||
21 | =IF(V9 <> "", V9, "") | =IF(W9 <> "", W9, "") | =IF(X9 <> "", X9, "") | =IF(Y9 <> "", Y9, "") | ||
22 | =IF(V10 <> "", V10, "") | =IF(W10 <> "", W10, "") | =IF(X10 <> "", X10, "") | =IF(Y10 <> "", Y10, "") | ||
23 | =IF(V11 <> "", V11, "") | =IF(W11 <> "", W11, "") | =IF(X11 <> "", X11, "") | =IF(Y11 <> "", Y11, "") |
Now that we have set up the table, and range to receive the data, it is time to construct the query which tells the database which data to find.
We will need to select the warehouse, status, name, and date of the single most recent report for each warehouse. We need to break down this query.
To select the most recent report for each warehouse, we use the function MAX(_molnify_timestamp)
,
and GROUP BY warehouse
to aggregate by warehouse. This sub-query becomes:
SELECT
warehouse,
MAX(_molnify_timestamp) AS max_timestamp
FROM
`data_database-example-logging_0`
GROUP BY
warehouse
The GROUP BY
instructs the query to not find the MAX
of all columns, but instead find the MAX
for each warehouse.
It is unlikely that two reports are logged at the exact same second, but we
should not ignore the possibility.
Since the recordId is the primary key, we are guaranteed that it will be unique,
so it is a perfect tie-breaker. So, for each of the warehouses present in
the result of the earlier query, we want to determine the biggest recordId.
We perform this operation with a JOIN
and a
GROUP BY
.
SELECT
MAX(a.recordId) AS recordId,
a.warehouse
FROM
`data_database-example-logging_0` AS a
JOIN (
SELECT
warehouse,
MAX(_molnify_timestamp) AS max_timestamp
FROM
`data_database-example-logging_0`
GROUP BY
warehouse
) AS b
ON
a.warehouse = b.warehouse
AND
a.`_molnify_timestamp` = b.max_timestamp
GROUP BY
a.warehouse
The JOIN
combines the table with the result of the inner query.
It returns only rows where the ON
criteria match.
Now that we have the recordId of the most recent log for each warehouse location,
we can use a second JOIN
to find the status, name, and log date for the
rows that match the recordIds. The final query becomes:
SELECT
t.warehouse,
t.status,
t.name,
DATE(t._molnify_timestamp)
FROM
`data_database-example-logging_0` AS t
JOIN (
SELECT
MAX(a.recordId) AS recordId,
a.warehouse
FROM
`data_database-example-logging_0` AS a
JOIN (
SELECT
warehouse,
MAX(_molnify_timestamp) AS max_timestamp
FROM
`data_database-example-logging_0`
GROUP BY
warehouse
) AS b
ON
a.warehouse = b.warehouse
AND
a.`_molnify_timestamp` = b.max_timestamp
GROUP BY
a.warehouse
) AS s
ON
t.recordId = s.recordId
Now, the only thing that is left to do is to add the query as an autofill metadata.
A | B | |
---|---|---|
1 | ID | database-example-logging |
2 | Name | Warehouse Logging Tool |
3 | Description | This application is a showcase of how a (fairly simple) logging tool can be implemented in Molnify. |
4 | Website | https://app.molnify.com/guides/database |
5 | Template | red dot |
6 | RecordTableName | ="data_"&B1&"_0" |
7 | autofill.Recent_logs | ="SELECT `_molnify_timestamp`, name, warehouse, status FROM `data_"&B1&"_0` ORDER BY `_molnify_timestamp` DESC" |
8 | autofill.Warehouse_data | ="SELECT t.warehouse, t.status, t.name, DATE(t._molnify_timestamp) FROM `data_"&B1&"_0` AS t JOIN ( SELECT MAX(a.recordId) AS recordId, a.warehouse FROM `data_"&B1&"_0` AS a JOIN ( SELECT warehouse, MAX(_molnify_timestamp) AS max_timestamp FROM `data_"&B1&"_0` GROUP BY warehouse ) AS b ON a.warehouse = b.warehouse AND a.`_molnify_timestamp` = b.max_timestamp GROUP BY a.warehouse ) AS s ON t.recordId = s.recordId" |
Remember that the named range is called Warehouse_data
,
so we set autofill.Warehouse_data
to our query.
You can download the final Excel file for the logging tool, but remember that the application will not work unless you change the ID and set up the database tables for the new ID.
The demo application can be tried out below, or at https://app.molnify.com/app/database-example-logging.
This example guides you through building a simple data dashboard in Molnify using data from the previous logging tool example.
Instead of reading all data into the application and performing calculations locally—which can cause performance issues with large datasets—we'll optimize autofill queries to perform aggregation directly in the database.
Following best practices, we start by defining our metadata. Note that we use a different ID than the logging tool.
A | B | |
---|---|---|
1 | ID | database-example-dashboard |
2 | Name | Warehouse Data |
3 | Description | This application is a showcase of how a (fairly simple) dashboard can be implemented in Molnify. |
4 | Website | https://app.molnify.com/guides/database |
5 | Template | red dot |
Note that you will have to change the ID if you copy this example, as this ID is taken by the example application.
Next, we'll set up inputs for filtering by warehouse, inspector, status, or a date range. Dropdown menus will help manage these filters efficiently. We define data for the dropdowns in columns K, L and M.
K | L | M | |
---|---|---|---|
1 | Names | Warehouses | States |
2 | Ivan Metcalf | Chepstow | Good |
3 | Renee Ellison | Tarrin | Poor |
4 | Aisha Morris | Wigston | Action needed |
5 | Barrie Henderson | Pitmedden | |
6 | Ewa Mistry | Laewaes | |
7 | Kirkwall | ||
8 | Woodpine | ||
9 | Rochdale | ||
10 | Peatsland | ||
11 | Auctermunty |
The names and warehouse locations match those used in the logging tool.
A more thorough implementation would use database table(s) for the names of the inspectors and warehouses, and possibly even the different states as well. Then, any staff or warehouse changes would immediately be reflected in both applications, which removes the risk of forgetting to update the dropdown data for one of the two applications.
An even more robust implementation would also use IDs for the employees and warehouses, and compare the ID rather than the names for filters and aggregation of data. The only usage of the display names for the inspectors and warehouses would be when displaying them to the user. That way, any name changes would be reflected in the stored data as well.
When analysing data, we expect to want to select multiple items at once, e.g. to compare two warehouses, or to see the data of all warehouses in state "Poor" or "Action needed". To allow this, we will use dropdowns with the multiple ui string.
For choosing a start- and an end-date, we will use two date-pickers. We set the default interval to start a year ago, and end today.
We use a dividername to make it clear that these inputs are filters.
D | E | F | |
---|---|---|---|
1 | Name | dropdown;multiple;dividername=Filter data | |
2 | Warehouse | dropdown;multiple | |
3 | Status | dropdown;multiple | |
4 | Interval start | =DATE(YEAR(TODAY()) - 1, MONTH(TODAY()), DAY(TODAY())) | date |
5 | Interval end | =TODAY() | date |
The "multiple" ui-string will allow the user not only to select multiple options from the dropdown, but also to de-select a chosen option as any "tag" can be removed by clicking on its "x".
Since we're using an existing database, there's no need to create additional tables. We only need to grant the new application access via the logging tool's database manager.
We will be re-using the query for filtering across several autofill queries. Rather than maintaining duplicate filters across multiple queries, we define them in a single cell and reference that cell when constructing queries.
The filter adds clauses that use the FIND_IN_SET
function to match against the values
selected in the multiple dropdowns. If we had not configured the dropdowns to permit multiple inputs,
the filter could instead have been simplified to use equality =
.
To handle dates correctly, the filter uses the TEXT
Excel function to convert
to the SQL timestamp syntax.
Since we can expect a slight delay between our user opening the page, and the data being made available to the user, the charts might be drawn for the user before data has populated the application. We use the custom javascript function calculateButton to ensure data is displayed once the application is fully loaded.
A | B | |
---|---|---|
7 | JavascriptAfterLoad | calculateButton(); |
8 | Table | `data_database-example-logging_0` |
9 | Filter | =" (_molnify_timestamp BETWEEN '"&TEXT(E4, "yyyy-mm-dd hh:mm:ss")&"' AND '"&TEXT(E5 + 1, "yyyy-mm-dd hh:mm:ss")&"' ) "&IF(E1<>"", " AND FIND_IN_SET(name, REPLACE('"&E1&"',';',',')) ", "")&IF(E2<>"", " AND FIND_IN_SET(warehouse, REPLACE('"&E2&"',';',',')) ", "")&IF(E3<>"", " AND FIND_IN_SET(status, REPLACE('"&E3&"',';',',')) ", "") |
Values in the multiple dropdown inputs are separated by semicolons.
To use them with the SQL function FIND_IN_SET
,
we replace semicolons with commas using the REPLACE
function.
Our first chart will visualise the number of times each warehouse has been inspected.
We define a named range 'Inspection_Counts' spanning O2:P11
to hold data, and color the range grey.
We use the data in "Inspection_Counts" to populate a barchart in O13:Q23
.
The warehouse names are the categories for the chart, so only the column P is coloured blue.
O | P | Q | |
---|---|---|---|
13 | Inspection count | Number of inspections | barchart;noGroupedStackedControls;noGridLines;yAxisTicks=3 |
14 | =IF(O2 <> "", O2, "") | =IF(P2 <> "", P2, "") | |
15 | =IF(O3 <> "", O3, "") | =IF(P3 <> "", P3, "") | |
16 | =IF(O4 <> "", O4, "") | =IF(P4 <> "", P4, "") | |
17 | =IF(O5 <> "", O5, "") | =IF(P5 <> "", P5, "") | |
18 | =IF(O6 <> "", O6, "") | =IF(P6 <> "", P6, "") | |
19 | =IF(O7 <> "", O7, "") | =IF(P7 <> "", P7, "") | |
20 | =IF(O8 <> "", O8, "") | =IF(P8 <> "", P8, "") | |
21 | =IF(O9 <> "", O9, "") | =IF(P9 <> "", P9, "") | |
22 | =IF(O10 <> "", O10, "") | =IF(P10 <> "", P10, "") | |
23 | =IF(O11 <> "", O11, "") | =IF(P11 <> "", P11, "") |
We will need to define an
autofill metadata
to populate the "Inspection_Counts" range with data.
The query will SELECT
the warehouse
column, and COUNT
the number of rows in the table. The output counts should be GROUP BY
the different warehouses, so that a row belonging to the warehouse "Tarrin" is only
counted towards the sum for "Tarrin".
In other words, we will use the following query:
SELECT warehouse, COUNT(*)
FROM `data_database-example-logging_0`
GROUP BY warehouse
ORDER BY warehouse ASC
After we apply the filtering in a WHERE
clause, and
dynamically find the table name, the Excel formula for the query becomes:
A | B | |
---|---|---|
8 | Table | `data_database-example-logging_0` |
9 | Filter | =" (_molnify_timestamp BETWEEN '"&TEXT(E4, "yyyy-mm-dd hh:mm:ss")&"' AND '"&TEXT(E5 + 1, "yyyy-mm-dd hh:mm:ss")&"' ) "&IF(E1<>"", " AND FIND_IN_SET(name, REPLACE('"&E1&"',';',',')) ", "")&IF(E2<>"", " AND FIND_IN_SET(warehouse, REPLACE('"&E2&"',';',',')) ", "")&IF(E3<>"", " AND FIND_IN_SET(status, REPLACE('"&E3&"',';',',')) ", "") |
10 | autofill.Inspection_Counts | ="SELECT warehouse, count(*) FROM "&B8&" WHERE "&B9&" GROUP BY warehouse ORDER BY warehouse ASC" |
The autofill refers back to the cell with the table name and the cell with the filter.
Now, we are interested in visualizing the performance of the inspectors. We will use a bar chart again, but with one data series per warehouse status. Users can leverage the chart's grouped/stacked controls to display the total number of reports per inspector.
We create a new named range "Inspector_Data" over S2:V11
, and colour it grey.
Below it, we colour an area blue for the graph.
While we have only five inspectors, the additional rows will not interfere with our calculations or displays,
and they allow us to easily add additional inspectors in the future.
We will use the inspectors' names as the categories for the chart, and the distribution of statuses for the inspectors will be the chart data.
S | T | U | V | W | |
---|---|---|---|---|---|
13 | Inspector report overview | Good | Poor | Action needed | barchart;noGridLines;yAxisTicks=3 |
14 | =IF(S2 <> "", S2, "") | =IF(T2 <> "", T2, "") | =IF(U2 <> "", U2, "") | =IF(V2 <> "", V2, "") | |
15 | =IF(S3 <> "", S3, "") | =IF(T3 <> "", T3, "") | =IF(U3 <> "", U3, "") | =IF(V3 <> "", V3, "") | |
16 | =IF(S4 <> "", S4, "") | =IF(T4 <> "", T4, "") | =IF(U4 <> "", U4, "") | =IF(V4 <> "", V4, "") | |
17 | =IF(S5 <> "", S5, "") | =IF(T5 <> "", T5, "") | =IF(U5 <> "", U5, "") | =IF(V5 <> "", V5, "") | |
18 | =IF(S6 <> "", S6, "") | =IF(T6 <> "", T6, "") | =IF(U6 <> "", U6, "") | =IF(V6 <> "", V6, "") | |
19 | =IF(S7 <> "",S7, "") | =IF(T7 <> "", T7, "") | =IF(U7 <> "", U7, "") | =IF(V7 <> "", V7, "") | |
20 | =IF(S8 <> "", S8, "") | =IF(T8 <> "", T8, "") | =IF(U8 <> "", U8, "") | =IF(V8 <> "", V8, "") | |
21 | =IF(S9 <> "", S9, "") | =IF(T9 <> "", T9, "") | =IF(U9 <> "", U9, "") | =IF(V9 <> "", V9, "") | |
22 | =IF(S10 <> "", S10, "") | =IF(T10 <> "", T10, "") | =IF(U10 <> "", U10, "") | =IF(V10 <> "", V10, "") | |
23 | =IF(S11 <> "", S11, "") | =IF(T11 <> "", T11, "") | =IF(U11 <> "", U11, "") | =IF(V11 <> "", V11, "") |
To populate the "Inspector_Data" range with data, we use the autofill metadata.
For our query, we will SELECT
the name of the inspector
in the first column. In the second column, we will sum the number of rows,
but only rows if its column status
equals "Good". Similar
logic will sum the rows where status
is equal to "Poor" for the
third column, and where status
is equal to "Action needed" for
the fourth and last column. These sums should also be GROUP BY
the inspector name, so that a report is only counted towards the result for
its inspector. The output should be ORDER BY
the inspector name.
In MySQL, our query is:
SELECT name,
SUM(IF(status='Good', 1, 0)),
SUM(IF(status='Poor', 1, 0)),
SUM(IF(status='Action needed', 1, 0))
FROM WHERE `data_database-example-logging_0`
GROUP BY name
ORDER BY name ASC
We add our filters and dynamic table name resolution to the above query. Additionally, instead of writing the name of the statuses directly in the formula, we dynamically read from the "States" named range. This ensures that if we were to change the name of a state, we would not risk forgetting to change the autofill query. We add the Excel formula which generates this query to our metadata:
A | B | |
---|---|---|
8 | Table | `data_database-example-logging_0` |
9 | Filter | =" (_molnify_timestamp BETWEEN '"&TEXT(E4, "yyyy-mm-dd hh:mm:ss")&"' AND '"&TEXT(E5 + 1, "yyyy-mm-dd hh:mm:ss")&"' ) "&IF(E1<>"", " AND FIND_IN_SET(name, REPLACE('"&E1&"',';',',')) ", "")&IF(E2<>"", " AND FIND_IN_SET(warehouse, REPLACE('"&E2&"',';',',')) ", "")&IF(E3<>"", " AND FIND_IN_SET(status, REPLACE('"&E3&"',';',',')) ", "") |
10 | autofill.Inspection_Counts | ="SELECT warehouse, count(*) FROM "&B8&" WHERE "&B9&" GROUP BY warehouse ORDER BY warehouse ASC" |
11 | autofill.Inspector_Data | ="SELECT name, SUM(IF(status='"&M2&"', 1, 0)), SUM(IF(status='"&M3&"', 1, 0)), SUM(IF(status='"&M4&"', 1, 0)) FROM "&B8&" WHERE "&B9&" GROUP BY name ORDER BY name ASC" |
The status names "Good", "Poor" and "Action needed" are defined in M2:M4.
We will also visualise the proportion of reports which assign each kind of status. Since we are interested in the data as a proportion, we will use a pie chart for visualization.
Once more, we prepare a range to fill our data into: we give the range Y2:Z4
the name "Status_Counts", colour it grey, and colour an area below it blue for the chart.
Note that by explicitly instructing Molnify that the chart is a pie chart through the ui-string, we do not need to ensure that the portions of the pie sum to 1.
Y | Z | AA | |
---|---|---|---|
13 | Status overview | Count | pieChart |
14 | =IF(Y2 <> "", Y2, "") | =IF(Z2 <> "", Z2, "") | |
15 | =IF(Y3 <> "", Y3, "") | =IF(Z3 <> "", Z3, "") | |
16 | =IF(Y4 <> "", Y4, "") | =IF(Z4 <> "", Z4, "") |
The query which we use in the autofill metadata
will SELECT
the statuses and COUNT
the rows. The GROUP BY
operation
groups the rows into the different statuses before the counting.
The query becomes as follows:
SELECT
status,
COUNT(*)
FROM `data_database-example-logging_0`
GROUP BY status
We apply the filter and table name to the query, and the resulting Excel formula becomes as below:
A | B | |
---|---|---|
8 | Table | `data_database-example-logging_0` |
9 | Filter | =" (_molnify_timestamp BETWEEN '"&TEXT(E4, "yyyy-mm-dd hh:mm:ss")&"' AND '"&TEXT(E5 + 1, "yyyy-mm-dd hh:mm:ss")&"' ) "&IF(E1<>"", " AND FIND_IN_SET(name, REPLACE('"&E1&"',';',',')) ", "")&IF(E2<>"", " AND FIND_IN_SET(warehouse, REPLACE('"&E2&"',';',',')) ", "")&IF(E3<>"", " AND FIND_IN_SET(status, REPLACE('"&E3&"',';',',')) ", "") |
10 | autofill.Inspection_Counts | ="SELECT warehouse, count(*) FROM "&B8&" WHERE "&B9&" GROUP BY warehouse ORDER BY warehouse ASC" |
11 | autofill.Inspector_Data | ="SELECT name, SUM(IF(status='"&M2&"', 1, 0)), SUM(IF(status='"&M3&"', 1, 0)), SUM(IF(status='"&M4&"', 1, 0)) FROM "&B8&" WHERE "&B9&" GROUP BY name ORDER BY name ASC" |
12 | autofill.Status_Counts | ="SELECT status, COUNT(*) FROM "&B8&" WHERE "&B9&" GROUP BY status" |
We do not sort the result, since the data is visualised in a pie chart.
We might want to display the dashboard on a TV screen or similar non-interactive display. But, the Molnify application will not refresh the data unless we interact with it, e.g. by modifying a filter. To solve this, we add a Javascript to automatically refresh the data every minute.
However, we may have regular users of the dashboard for whom the automatic update is a waste of bandwidth. So, we will distinguish between use-cases with a URL parameter.
We add the following to our JavascriptAfterLoad:
A | B | |
---|---|---|
7 | JavascriptAfterLoad | calculateButton(); if(getValueForURLParam('autorefresh') === 'true') {setInterval( () => { calculateButton(); }, 60000)}; |
The setInterval performs an update every 60000 milliseconds.
With the javascript included, the application will automatically refresh every minute
if it is accessed on the URL
https://app.molnify.com/app/<your-application-id>?autorefresh=true
.
You can download the final Excel file for the data dashboard. Remember to update the application ID and configure database access for it to function correctly.
The demo dashboard can be tried out below, or at https://app.molnify.com/app/database-example-dashboard.