You can write your code in a language of your choice. But we have preferences for the languages of our stack:
- Go
- Python
- NodeJS
Please organise your code, document it and write relevant unit tests to test your code. Do provide instructions for any specific setup required to run your code.
At Sephora SEA, we are building business schemas on top of the e-commerce source databases. For instance, we have a products table which is a consolidation of 30 other tables. It contains information for each of our products (ids, descriptions, categories...).
Our tables are organised in datasets (folders), eg. final
, raw
, tmp
.
The products table is created from a chain of dependent SQL scripts. The process is as follows:
- Dump the e-commerce databases into a raw dataset (
res/raw
folder) - Clean and consolidate the data on top of these raw tables using SQL scripts stored in the
res/tmp
folder. For instance, the result of theres/tmp/inventory_items.sql
script will be stored in thetmp.inventory_items
table. - Create the
final.products
table (from theres/final/products.sql
script)
This assignment is roughly about recreating this implementation. We will first focus on the logic to implement and run the dependency tree, followed by a deployment and running it against a real data warehouse instance.
Your task is to build parts of the tool that will orchestrate the aforementioned process, in order to create the final.products
table:
- Write a function that that would run the SQL scripts in the correct order. For example, since
tmp.variants
depends ontmp.item_purchase_prices
,res/tmp/item_purchase_prices.sql
needs to be run before we can runres/tmp/variants.sql
. Please provide documentation as of how you are proceeding.
Going further, we would like to parallelize the execution of few of these scripts. If you think of the dependencies as a tree: scripts from different nodes can work simultaneously, but, still, must not be executed before its children's tasks are done.
- Write a function that parallelize the execution of the SQL scripts, ensuring they respect their dependencies. Please provide documentation as of how you are proceeding.
In this section, we will work on deploying an implementation of the previous question.
- Setting up a BigQuery instance. The data warehouse platform that we are using in Sephora is Google Bigquery. Google allows you to try BigQuery for free with a Google account (that you have by default with a gmail address for example)
- Go to https://bigquery.cloud.google.com/welcome, login and create a project.
- You will be redirected to console.developers.google.com where you can give a name to your project.
- Once the project is created, and if not done by default, activate BigQuery API there: https://console.developers.google.com/apis/api/bigquery-json.googleapis.com/ (you might have to select your project in the header)
- Once done, you now have access to bigquery ! https://bigquery.cloud.google.com/
- Let's create our datasets. In the left column of the UI, find the
No datasets found in this project
text. There, you will be able to create 3 datasetsraw
,tmp
andfinal
. - Click
Compose Query
- Click
Show Options
, uncheckUse Legacy SQL
, and clickHide Options
- Run the queries in the
db_init
folder to populate theraw
dataset
The instance is now ready!
-
Update your previous code to run on BigQuery. Instead of having a fake function simulating the execution of the query in section A, implement it to run on BigQuery. Upon execution, the
tmp
andfinal
datasets should be populated based on theraw
data. Google BigQuery has a documented API as well as a bunch of SDKs available. -
Deploy an API on the platform of your choice. The API should obey the following specifications:
POST /run
: Runs the code implemented in B-2. When this endpoint is called, thetmp
andfinal
tables should be recreated.POST /<table_name>?<attribute_1>=<value_1>&<attribute_2>=<value_2>...
: Append a new row in the definedtable_name
in theraw
dataset with the attributes. eg.POST /products?name=banana%20lipstick&category_id=4&external_id=123&type=product
. This API should be able to handle invalid inputs for the table query (eg. return the appropriate response and/or response code if the field does not exist in the table specified).- The endpoints can be public.
- The endpoints should return appropriate status codes, as well as having appropriate response format for handling the JSON responses.
-
The files in the
res/raw
folder represents the available raw data tables in theraw
dataset. -
For point A-1, your code should be flexible enough so that it will still work properly if more scripts are added in the tmp or final folder. The dependencies should not be hardcoded inside the script itself but should be a more flexible implementation (eg. as part of a configuration file).
-
For point A-2, you can use a dummy function such as the following as a placeholder for the function actually running the sql scripts (eg. here in golang):
func() {
// This is execution time of the tmp.inventory_items script
time.Sleep(time.Second * 3)
}
- Source code of the exercise (Sections A and B).
- Documentation of how you are proceeding and how to deploy and run locally (Sections A and B-2)
- Credentials to access the datasets of your BQ instance for the ds@luxola.com user (can be setup here https://console.cloud.google.com/iam-admin/iam. The BigQuery Data Viewer permission is enough)
- URL endpoint of your API
Good luck :)