Skip to content

Tutorial

Major feature of a Datero data platform is its ability to join data from different sources within a single query. In this tutorial we will demonstrate how Datero could empower your data analytics without any ETL development.

Scenario

Let's assume we work for an imaginary Enterprise company.

As a usual company, it has multiple departments, like Production, Sales, Finance, etc. Each department has its own data sources, like databases, spreadsheets, etc. And there is a need to join data from different sources to get a complete picture of the company sales data.

This company has the following datasources:

  • Customers are stored in MySQL database
  • Products are stored in PostgreSQL database
  • Employees are stored in MSSQL database
  • Orders are stored in MongoDB database
  • Job roles are stored in SQLite database
  • Company departments are stored in CSV file

And we need to figure out in sales data who sold what product to which customer. As well as what job role and department this employee belongs to.

Logical Data Model

Below is a logical structure of data sources and their relations. Sales data is stored in the orders table. Customers place an orders which contain products. Employees serve the orders. Employees belong to job roles and job roles belong to departments.

erDiagram
  customers ||--o{ orders : places
  orders ||--|{ products : contains
  employees ||--o{ orders : serves
  employees o{--|| job_roles : belongs
  job_roles o{--|| departments : belongs

In terms of datasources, this diagram looks like this:

flowchart BT
  customers[("customers<br />MySQL")]
  orders[("orders<br />MongoDB")]
  products[(products<br />PostgreSQL)]
  employees[(employees<br />MSSQL)]
  job_roles[(job_roles<br />SQLite)]
  departments[(departments<br />CSV)]

  customers --> orders
  products --> orders
  employees --> orders
  job_roles --> employees
  departments --> job_roles

Test Data

The data that are stored in each data source are as follow.

id name
1 Tom
2 Kate
3 John
id name price
1 apple 1
2 banana 2.3
3 orange 3.5
id customer_id product_id employee_id quantity
1 1 1 1 10
2 1 2 2 2
3 1 3 3 5
4 2 1 3 5
5 2 3 2 3
6 3 1 3 8
id name job_id
1 John 1
2 Bob 2
3 Lisa 3
id name department_id
1 owner 1
2 manager 2
3 salesman 3
id name
1 management
2 finance
3 sales

Infrastructure Setup

source files

All the sources used in this tutorial are available in the demo folder of this documentation repository.

To emulate multiple data sources, we will use docker containers orchestrated by docker-compose. It will spin up datero container and several other containers for relevant datasources.

docker-compose
services:
  datero:
    image: docker.io/chumaky/datero:latest
    container_name: datero_main
    environment:
      POSTGRES_USER: postgres       # if value is different from default `postgres` then it must be specified in Datero config.yaml
      POSTGRES_PASSWORD: postgres   # if value is different from default `postgres` then it must be specified in Datero config.yaml
    ports:
      - 8080:80
      - 4444:5432
    volumes:
      # Datero config. Must be specified if postgres image default values are changed.
      # Also allows to specify servers explicitly thus avoiding manual creation
      #- ./config.yaml:/home/instance/config.yaml              
      - ./sqlite:/data/sqlite
      - ../data/tutorial/departments.csv:/data/departments.csv

  mysql:
    image: docker.io/library/mysql:latest
    container_name: datero_mysql
    environment:
      MYSQL_ROOT_PASSWORD: root
      MYSQL_DATABASE: finance
      MYSQL_USER: mysql
      MYSQL_PASSWORD: mysql
    ports:
      - 3306:3306
    volumes:
      - ./mysql/customers.sql:/docker-entrypoint-initdb.d/setup.sql

  postgres:
    image: docker.io/library/postgres:alpine
    container_name: datero_postgres
    environment:
      POSTGRES_DB: factory
      POSTGRES_USER: postgres
      POSTGRES_PASSWORD: postgres
    ports:
      - 5432:5432
    volumes:
      - ./postgres/products.sql:/docker-entrypoint-initdb.d/setup.sql

  mongo:
    image: docker.io/library/mongo:latest
    container_name: datero_mongo
    environment:
      MONGO_INITDB_DATABASE: sales
      MONGO_INITDB_ROOT_USERNAME: mongo
      MONGO_INITDB_ROOT_PASSWORD: mongo
    ports:
      - 27017:27017
    volumes:
      - ./mongo/orders.js:/docker-entrypoint-initdb.d/setup.js

  mssql:
    image: mcr.microsoft.com/mssql/server:2019-latest
    container_name: datero_mssql
    environment:
      ACCEPT_EULA: Y
      MSSQL_SA_PASSWORD: Mssql_2019
    ports:
      - 1433:1433
    entrypoint:
      - /usr/config/entrypoint.sh
    volumes:
      - ./mssql/entrypoint.sh:/usr/config/entrypoint.sh
      - ./mssql/configure_db.sh:/usr/config/configure-db.sh
      - ./mssql/employees.sql:/usr/config/setup.sql

Here is per-service breakdown of a compose file.

  datero:
    image: docker.io/chumaky/datero:latest
    container_name: datero_main
    environment:
      POSTGRES_USER: postgres       # if value is different from default `postgres` then it must be specified in Datero config.yaml
      POSTGRES_PASSWORD: postgres   # if value is different from default `postgres` then it must be specified in Datero config.yaml
    ports:
      - 8080:80
      - 4444:5432
    volumes:
      # Datero config. Must be specified if postgres image default values are changed.
      # Also allows to specify servers explicitly thus avoiding manual creation
      #- ./config.yaml:/home/instance/config.yaml              
      - ./sqlite:/data/sqlite
      - ../data/tutorial/departments.csv:/data/departments.csv

Main datero container

Web ui is available on port 8080. This is done intentionally to allow run on linux hosts where port 80 might require root permissions. To distinguish from postgres datasource we run datero database on port 4444. For sqlite and csv datasources we must mount them into the file system of this datero container. See corresponding service sections for detals.

  mysql:
    image: docker.io/library/mysql:latest
    container_name: datero_mysql
    environment:
      MYSQL_ROOT_PASSWORD: root
      MYSQL_DATABASE: finance
      MYSQL_USER: mysql
      MYSQL_PASSWORD: mysql
    ports:
      - 3306:3306
    volumes:
      - ./mysql/customers.sql:/docker-entrypoint-initdb.d/setup.sql

Customers data source

There is finance database is created. It will contain customers table defined via mysql_customers.sql setup script. Connection will be done over port 3306 under mysql/mysql user/password key pair.

  postgres:
    image: docker.io/library/postgres:alpine
    container_name: datero_postgres
    environment:
      POSTGRES_DB: factory
      POSTGRES_USER: postgres
      POSTGRES_PASSWORD: postgres
    ports:
      - 5432:5432
    volumes:
      - ./postgres/products.sql:/docker-entrypoint-initdb.d/setup.sql

Products data source

There is factory database is created. It will contain products table defined via postgres_products.sql setup script. Connection will be done over port 5432 under postgres/postgres user/password key pair.

  mongo:
    image: docker.io/library/mongo:latest
    container_name: datero_mongo
    environment:
      MONGO_INITDB_DATABASE: sales
      MONGO_INITDB_ROOT_USERNAME: mongo
      MONGO_INITDB_ROOT_PASSWORD: mongo
    ports:
      - 27017:27017
    volumes:
      - ./mongo/orders.js:/docker-entrypoint-initdb.d/setup.js

Orders data source

There is sales database is created. It will contain orders collection defined via mongo_orders.js setup script. Connection will be done over port 27017 under mongo/mongo user/password key pair.

  mssql:
    image: mcr.microsoft.com/mssql/server:2019-latest
    container_name: datero_mssql
    environment:
      ACCEPT_EULA: Y
      MSSQL_SA_PASSWORD: Mssql_2019
    ports:
      - 1433:1433
    entrypoint:
      - /usr/config/entrypoint.sh
    volumes:
      - ./mssql/entrypoint.sh:/usr/config/entrypoint.sh
      - ./mssql/configure_db.sh:/usr/config/configure-db.sh
      - ./mssql/employees.sql:/usr/config/setup.sql

Employess data source

There is hr database is created. It will contain employees table defined via mssql_employees.sql setup script. Connection will be done over port 1433 under sa/Mssql_2019 user/password key pair.

SQL Server initialization takes some time. To handle this gracefuly there are mssql_entrypoint.sh and mssql_configure_db.sh supplementary scripts are used.

  datero:
    image: docker.io/chumaky/datero:latest
    container_name: datero_main
    environment:
      POSTGRES_USER: postgres       # if value is different from default `postgres` then it must be specified in Datero config.yaml
      POSTGRES_PASSWORD: postgres   # if value is different from default `postgres` then it must be specified in Datero config.yaml
    ports:
      - 8080:80
      - 4444:5432
    volumes:
      # Datero config. Must be specified if postgres image default values are changed.
      # Also allows to specify servers explicitly thus avoiding manual creation
      #- ./config.yaml:/home/instance/config.yaml              
      - ./sqlite:/data/sqlite
      - ../data/tutorial/departments.csv:/data/departments.csv

Job roles data source

SQLite database is single file based. It doesn't have any listener over some port to connect to. Hence, we must mount it inside the datero container to enable access to it through its file system.

Database job_roles.db will be mounted to the /data/sqlite folder of the container. This directory is made writable by other users to allow datero container to write to it. The database contains job_roles table defined via job_roles.sql setup script from the demo/sqlite folder.

  datero:
    image: docker.io/chumaky/datero:latest
    container_name: datero_main
    environment:
      POSTGRES_USER: postgres       # if value is different from default `postgres` then it must be specified in Datero config.yaml
      POSTGRES_PASSWORD: postgres   # if value is different from default `postgres` then it must be specified in Datero config.yaml
    ports:
      - 8080:80
      - 4444:5432
    volumes:
      # Datero config. Must be specified if postgres image default values are changed.
      # Also allows to specify servers explicitly thus avoiding manual creation
      #- ./config.yaml:/home/instance/config.yaml              
      - ./sqlite:/data/sqlite
      - ../data/tutorial/departments.csv:/data/departments.csv

Departments data source

This is a file based resource. It doesn't have any listener over some port to connect to. To read the file, it must be accessible from local file system of the datero container.

File departments.csv from the sibling ../data/tutorial directory will be mounted to the /data folder of the container.

To spin-up all the containers, clone this docs repository and run the following command. It will first fetch all the images if they are absent on your local registry and then start all the containers.

docker compose -f demo/docker-compose.yml up -d

After execution of this command, you should see output similar to below.

$ docker ps
CONTAINER ID  IMAGE                                       COMMAND               CREATED             STATUS             PORTS                                         NAMES
db495bdfe319  docker.io/chumaky/datero:latest             supervisord -c /e...  About a minute ago  Up About a minute  0.0.0.0:4444->5432/tcp, 0.0.0.0:8080->80/tcp  datero_main
590a6adbaa2c  docker.io/library/mysql:latest              mysqld                About a minute ago  Up About a minute  0.0.0.0:3306->3306/tcp                        datero_mysql
5c3cba74112b  docker.io/library/postgres:alpine           postgres              About a minute ago  Up About a minute  0.0.0.0:5432->5432/tcp                        datero_postgres
d92ba333ba28  docker.io/library/mongo:latest              mongod                About a minute ago  Up About a minute  0.0.0.0:27017->27017/tcp                      datero_mongo
a797fbb12392  mcr.microsoft.com/mssql/server:2019-latest                        About a minute ago  Up About a minute  0.0.0.0:1433->1433/tcp                        datero_mssql

Checking data sources

Let's check all our datasources to make sure they are running and have expected seed data.

$ docker exec datero_mysql mysql -umysql -pmysql -e "select * from finance.customers"
id      name
1       Tom
2       Kate
3       John
$ docker exec datero_postgres psql -d factory -U postgres -c "select * from products"
id |  name  | price
----+--------+-------
  1 | apple  |     1
  2 | banana |   2.3
  3 | orange |   3.5
(3 rows)
$ docker exec datero_mongo mongo -u mongo -pmongo --authenticationDatabase admin --eval "db.getSiblingDB('sales').orders.find()"
MongoDB shell version v5.0.10
connecting to: mongodb://127.0.0.1:27017/?authSource=admin&compressors=disabled&gssapiServiceName=mongodb
Implicit session: session { "id" : UUID("21c19e50-0a03-4e92-aa99-126da29607d8") }
MongoDB server version: 5.0.10
{ "_id" : ObjectId("64cf8a76bee1f91da11ce665"), "id" : 1, "customer_id" : 1, "product_id" : 1, "employee_id" : 1, "quantity" : 10 }
{ "_id" : ObjectId("64cf8a76bee1f91da11ce666"), "id" : 2, "customer_id" : 1, "product_id" : 2, "employee_id" : 2, "quantity" : 2 }
{ "_id" : ObjectId("64cf8a76bee1f91da11ce667"), "id" : 3, "customer_id" : 1, "product_id" : 3, "employee_id" : 3, "quantity" : 5 }
{ "_id" : ObjectId("64cf8a76bee1f91da11ce668"), "id" : 4, "customer_id" : 2, "product_id" : 1, "employee_id" : 3, "quantity" : 5 }
{ "_id" : ObjectId("64cf8a76bee1f91da11ce669"), "id" : 5, "customer_id" : 2, "product_id" : 3, "employee_id" : 2, "quantity" : 3 }
{ "_id" : ObjectId("64cf8a76bee1f91da11ce66a"), "id" : 6, "customer_id" : 3, "product_id" : 1, "employee_id" : 3, "quantity" : 8 }
$ docker exec datero_mssql /opt/mssql-tools/bin/sqlcmd -S localhost -U sa -P Mssql_2019 -d hr -Q "select * from employees"
id          name      job_id
----------- --------- -----------
          1 John                1
          2 Bob                 2
          3 Lisa                3
$ sqlite3 -header -column demo/sqlite_job_roles.db "select * from job_roles"
id  name      department_id
--  --------  -------------
1   owner     1
2   manager   2
3   salesman  3
$ cat data/tutorial/departments.csv
id,name
1,management
2,finance
3,sales

We successfully connected to all the data sources and all of them have expected data.

Datero setup

Now, when all the data sources are up and running, we can configure them in Datero ui. Navigate to the http://localhost:8080 and you should see default Datero dashboard. For more details about it, please refer to the overview section.

Default Datero dashboard

Datero dashboard

As of now, we don't have any data sources defined yet. Let's add them one by one.

Detailed instructions how to add data sources for each supported connector could be found in the connectors section. We will not repeat them here, but instead will show how to add server and import schema whenever applicable for each of our datasources.

Servers creation

In the left navigation pane of the dashboard in the Connectors section click on the corresponding connector and fill out connection details as per screenshots below. The passwords for all the datasources could be found in the docker-compose.yml file from which we spinned up all the containers.

MySQL server creation
MySQL server creation

Postgres server creation
Postgres server creation

Mongo server creation
Mongo server creation

MSSQL server creation
MSSQL server creation

SQLite server creation
SQLite server creation

Zero config

File connector is zero params connector, so the user friendly name is the only thing we need.

File based server
File based server

Automatic servers creation

The previous step is a manual process. But we can do better. Datero supports automatic servers creation based on the configuration file which could be mounted to the /home/instance/config.yaml file. If you would uncomment the line 13 in the docker-compose.yml file and reinstantinate all the containers, then servers will be created automatically.

docker compose -f demo/docker-compose.yml down

# uncomment the line 13 in the docker-compose.yml file and save it.
#- ./config.yaml:/home/instance/config.yaml

docker compose -f demo/docker-compose.yml up -d

Datero supports fully fledged configuration file with all the possible FDW options. For more details, please refer to the configuration section.

  datero:
    image: docker.io/chumaky/datero:latest
    container_name: datero_main
    environment:
      POSTGRES_USER: postgres       # if value is different from default `postgres` then it must be specified in Datero config.yaml
      POSTGRES_PASSWORD: postgres   # if value is different from default `postgres` then it must be specified in Datero config.yaml
    ports:
      - 8080:80
      - 4444:5432
    volumes:
      # Datero config. Must be specified if postgres image default values are changed.
      # Also allows to specify servers explicitly thus avoiding manual creation
      #- ./config.yaml:/home/instance/config.yaml              
      - ./sqlite:/data/sqlite
      - ../data/tutorial/departments.csv:/data/departments.csv
# overrides default values for underlying postgres connection
postgres:
  hostname: localhost
  port: 5432
  database: postgres
  username: postgres
  password: postgres

servers:
  mysql_customers:
    description: MySQL - Customers
    fdw_name: mysql_fdw
    foreign_server:
      host: mysql
      port: 3306
    user_mapping:
      username: mysql
      password: mysql

  postgres_products:
    description: Postgres - Products
    fdw_name: postgres_fdw
    foreign_server:
      host: postgres
      port: 5432
      dbname: factory
    user_mapping:
      user: postgres
      password: postgres

  # using system generated name to allow re-use for the manually and automatically created servers
  mongo_fdw_1:
    description: Mongo - Orders
    fdw_name: mongo_fdw
    foreign_server:
      address: mongo
      port: 27017
      authentication_database: admin
    user_mapping:
      username: mongo
      password: mongo

  mssql_employees:
    description: MSSQL - Employees
    fdw_name: tds_fdw
    foreign_server:
      servername: mssql
      port: 1433
      database: hr
    user_mapping:
      username: sa
      password: Mssql_2019

  sqlite_job-roles:
    description: SQLite - Job Roles
    fdw_name: sqlite_fdw
    foreign_server:
      database: /data/sqlite/job_roles.db

  # using system generated name to allow re-use for the manually and automatically created servers
  file_fdw_1:
    description: CSV - Departments
    fdw_name: file_fdw

Import schemas

Once all the servers are created, we can import schemas/databases from them.

MySQL import schema
MySQL import schema

Postgres import schema
Postgres import schema

Mongo collections import

Mongo doesn't have schemas. It operates on terms databases and collections. Automatic fetch of collections is under development. For now, we need to manually create foreign table pointed to the collection.

To do this, open Query Editor and execute the following query:

Mongo foreign table creation
-- create foreign tables. mongo_fdw doesn't support IMPORT FOREIGN SCHEMA
DROP SCHEMA IF EXISTS mongo CASCADE;
CREATE SCHEMA IF NOT EXISTS mongo;

CREATE FOREIGN TABLE mongo.orders
( _id          name
, id           int
, customer_id  int
, product_id   int
, employee_id  int
, quantity     int
)
SERVER mongo_fdw_1
OPTIONS (database 'sales', collection 'orders')
;

Mongo import collection
Mongo import collection

MSSQL import schema
MSSQL import schema

SQLite import schema
SQLite import schema

File based import

File represents a single table. Automatic fetch of the list of available files is under development. For now, we need to manually create foreign table pointed to the file.

To do this, open Query Editor and execute the following query:

File based foreign table creation
DROP SCHEMA IF EXISTS csv CASCADE;
CREATE SCHEMA IF NOT EXISTS csv;

CREATE FOREIGN TABLE csv.departments
( id        int
, name      varchar
)
SERVER file_fdw_1
OPTIONS (filename '/data/departments.csv', format 'csv', header 'true')
;

Important

Schema import doesn't physically copy any data. For every source table and view it creates an object of a special type in a local schema. This object type is called foreign table. It implements data virtualization pattern.

Querying foreign table will automatically fetch data from the source database. If supported by connector, any filtering, sorting, grouping, etc. will be pushed down to the source database. This means that only the data that is needed will be fetched.

If you change the schema in the source database, you will need to re-import it in Datero to reflect the changes. Thus, schema evolution is handled automatically just by re-importing the schema.

Query Data

Having all servers set up you should see such dashboard view. In the navigation pane on the left, in the Servers section on the top there are listed all our created servers. Dashboard graphs now show servers distribution by vendor.

Datero dashboard with servers

Datero dashboard with servers

Now open Query Editor by clicking on the corresponding icon on the left toolbar. You will see all our datasource imported to the schemas we specified during schema import.

SQL Editor - Datasources

SQL Editor - Datasources

Finally, we can run our query to join data from all the datasources. Let's get all the sales data with customer, product, employee, job role and department details.

Sales data query
select c.name                          as customer_name
     , p.name                          as product
     , round(o.quantity * p.price, 2)  as total_amount
     , e.name                          as employee_name
     , j.name                          as employee_position
     , d.name                          as employee_department
  from mongo.orders      o
  join mysql.customers   c on c.id = o.customer_id
  join postgres.products p on p.id = o.product_id
  join mssql.employees   e on e.id = o.employee_id
  join sqlite.job_roles  j on j.id = e.job_id
  join csv.departments   d on d.id = j.department_id
;

And this is what we get as a result

Sales data details

Sales data details

Just stop for a second and think what we got. We joined data from 6 different datasources with a single query! There is a mix of relational and non-relational databases. Also, some of the datasources are even not databases, but files. And we didn't write a single line of ETL code to achieve this.

Think about how much time it would take to develop such capability without Datero! 😉

Reverse ETL

But it's not all. Datero is not only about querying data. It also allows to write data back to the datasources.

Assume, we want to add new pineapple product and update price for the apple product. Current products table looks like this.

Current products table

Current products table

Let's execute the following sql to add new product and update existing one.

Add pineapple and update apple
insert into postgres.products values (4, 'pineapple', 6.4);
update postgres.products set price = 1.2 where name = 'apple';

select * from postgres.products order by id;

Checking table again

Products table after update

Products table after update

So, what just happened? We didn't just update a local table in a database. We updated a table in a remote source database!

Let's check it. Connect to the postgres source database container and check the products table.

$ docker exec datero_postgres psql -d factory -U postgres -c "select * from products order by id"
 id |   name    | price
----+-----------+-------
  1 | apple     |   1.2
  2 | banana    |   2.3
  3 | orange    |   3.5
  4 | pineapple |   6.4
(4 rows)

Isn't it cool? 😎

Summary

Dealing with multiple datasources is a common situation in data analytics. Usual approach to combine the data is to develop ETL pipelines. Datero provides quick and easy way to join data from different datasources without any ETL development.

Thanks to the containerized approach, it's easy to be plugged in into any existing infrastructure. It could even be used as a part of some processing pipeline. Instead of writing data processing in python or java, you could spin up Datero container and write data processing in SQL.

Finally, Datero is not only about querying data. It also allows to write data back to the datasources. Which makes it a perfect tool for reverse ETL.