Please find more info in our Medium article 🙃
MLOps Community 🎉 is an open, free and transparent place for MLOps practitioners can collaborate on experiences and best practices around MLOps (DevOps for ML).
Engineering Labs Initiative is an educational project pursuing three goals:
- have fun 🥳
- learn 🤓
- share 🤝
The first lab was about integration of PyTorch with MLflow. The ML problem to tackle was a free choice.
Our team chose the Review classification problem based on Yelp Dataset. The data consists of the list of reviews on restaurant, museums, hospitals, etc., and the number of stars associated with this review (0-5). We model this task as a binary classification problem: is the review positive (has >=3 stars) or negative (has <3 stars). Following the torchtext tutorial, we implemented a model consisting of 2 layers: EmbeddingBag
and a linear layer, followed by a sigmoid activation function (using PyTorch Lightning). Please find the code in ./src. Many thanks to @paulomaia20 for handling this! 🤓
First of all, we implemented a small Web UI with Streamlit that defined the final goal of the project:
This Web UI is written in somewhat 50 lines of Python code! It uses REST API to access the deployed Model Proxy service, which provides both model inference and statistics calculation. The Web UI is deployed via Streamlit Sharing. Thanks @dmangonakis for the implementation! 😎
We absolutely ❤️ Kubernetes. And for this task, we couldn't resist not to use it. So we created a kubernetes cluster in GCP (thanks Google for $300 free credit), used helm charts to deploy MLflow server backed by managed PostgreSQL database as backend store and GCS bucket as artifact store. All services were exposed via public IP (thanks Neu.ro for adding the A-records into their DNS table for getting cool .mlops.neu.ro
domain names!). See ./gcp for details. Thanks @artem-yushkovsky for setting this up! 🤠
$ kubectl -n mlflow get all
NAME READY STATUS RESTARTS AGE
pod/mlflow-57c5fcd4df-52bbd 1/1 Running 0 16h
NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
service/mlflow LoadBalancer 10.104.0.146 34.91.123.207 5000:31243/TCP 11d
NAME READY UP-TO-DATE AVAILABLE AGE
deployment.apps/mlflow 1/1 1 1 11d
NAME DESIRED CURRENT READY AGE
replicaset.apps/mlflow-57c5fcd4df 1 1 1 16h
replicaset.apps/mlflow-7686588c6b 0 0 0 11d
Unfortunately, GCP free tier account doesn't include GPU resources to train models. Fortunately, there exist many other places where you can get computational resources for free. Thanks to NILG.AI for providing their servers to develop the model, and thanks to Neu.ro with their free $100 GPU quota we used to train the model. Please find the scripts in ./scripts/neuro.
$ sh ./scripts/neuro/gpu_train.sh
...
+ neuro run --name yelp-train --preset gpu-small-p --volume storage:yelp_dataset:/project:rw --volume secret:bucket-sa-key:/opt/developers-key.json --env PYTHONPATH=/project --env GOOGLE_APPLICATION_CREDENTIALS=/opt/developers-key.json --env MLFLOW_TRACKING_URI=http://mlflow.lab1-team3.mlops.neu.ro:5000 --env GIT_PYTHON_REFRESH=quiet --detach gcr.io/mlops-lab1-team3/yelp-dataset/model:v1.0 mlflow run /project --no-conda -P max_epochs=15
√ Job ID: job-1676f810-0d1c-47ff-8b82-00d5a4bb35c2
√ Name: yelp-train
- Status: pending Creating
- Status: pending Scheduling
- Status: pending ClusterScalingUp (Scaling up the cluster to get more resources)
- Status: pending Initializing
- Status: pending ContainerCreating
√ Http URL: https://yelp-train--artemyushkovskiy.jobs.neuro-compute.org.neu.ro
√ Status: running
2021/02/04 20:14:55 INFO mlflow.projects.utils: === Created directory /tmp/tmp80iddg3c for downloading remote URIs passed to arguments of type 'path' ===
2021/02/04 20:14:55 INFO mlflow.projects.backend.local: === Running command 'python3 -u src/train.py --n_grams 1 --batch_size 32 --embed_dim 32 --max_epochs 15' in run with ID '59e1f3764f6f44d9aeea13e10395ea5d' ===
2021-02-04 20:14:58,124 - INFO - utils.py:download_from_url - Downloading from Google Drive; may take a few minutes
2021-02-04 20:14:59,451 - INFO - utils.py:_process_response - Downloading file yelp_review_polarity_csv.tar.gz to ../data/yelp_review_polarity_csv.tar.gz.
yelp_review_polarity_csv.tar.gz: 166MB [00:02, 57.8MB/s]
2021-02-04 20:15:02,330 - INFO - utils.py:_process_response - File ../data/yelp_review_polarity_csv.tar.gz downloaded.
2021-02-04 20:15:02,330 - INFO - utils.py:extract_archive - Opening tar file ../data/yelp_review_polarity_csv.tar.gz.
2021-02-04 20:15:06,128 - INFO - text_classification.py:_setup_datasets - Building Vocab based on ../data/yelp_review_polarity_csv/train.csv
560000lines [01:03, 8808.98lines/s]
2021-02-04 20:16:11,081 - INFO - text_classification.py:_setup_datasets - Vocab has 464400 entries
2021-02-04 20:16:11,082 - INFO - text_classification.py:_setup_datasets - Creating training data
560000lines [01:55, 4868.63lines/s]
2021-02-04 20:18:06,112 - INFO - text_classification.py:_setup_datasets - Creating testing data
38000lines [00:07, 4922.04lines/s]
2021-02-04 20:18:13,834 - INFO - train.py:main - Creating average embedding model
2021-02-04 20:18:17,085 - INFO - train.py:main - Saving vocabulary to /tmp/tmpu_zkipr2/vocab.pkl
GPU available: True, used: True
TPU available: None, using: 0 TPU cores
LOCAL_RANK: 0 - CUDA_VISIBLE_DEVICES: [0]
2021-02-04 20:18:20,847 - INFO - train.py:main - run_id: 59e1f3764f6f44d9aeea13e10395ea5d
2021/02/04 20:18:22 INFO mlflow.utils.autologging_utils: pytorch autologging will track hyperparameters, performance metrics, model artifacts, and lineage information for the current pytorch workflow to the MLflow run with ID '59e1f3764f6f44d9aeea13e10395ea5d'
| Name | Type | Params
-------------------------------------------
0 | embedding | EmbeddingBag | 14.9 M
1 | fc | Linear | 66
-------------------------------------------
14.9 M Trainable params
0 Non-trainable params
14.9 M Total params
Epoch 12: 100% 18688/18688 [02:21<00:00, 132.31it/s, loss=0.194, v_num=5]
Registered model 'yelp-model' already exists. Creating a new version of this model...
2021/02/04 20:49:08 INFO mlflow.tracking._model_registry.client: Waiting up to 300 seconds for model version to finish creation. Model name: yelp-model, version 10
Created version '10' of model 'yelp-model'.
2021/02/04 20:49:12 INFO mlflow.projects: === Run (ID '59e1f3764f6f44d9aeea13e10395ea5d') succeeded ===
Thanks MLflow team such a cool product! It's really useful to have all the information in one place:
The developed models and their metadata is stored in GCS bucket gs://mlops-lab1-team3-mlflow
(public read), the experiments metadata (parameters, metrics, etc.) is stored in PostgreSQL.
To serve the models, we implemented a small FastAPI-based service that loads the Pytorch pickled model from the local FS and serves it via simple REST API:
$ kubectl -n mlflow-model port-forward pod/mlflow-model-server-6f46cdd48f-85zxq 8080:8080
Forwarding from 127.0.0.1:8080 -> 8080
Forwarding from [::1]:8080 -> 8080
...
$ curl -s -X POST -H "Content-Type: application/json" -d '{"text": "very cool restaurant!"}' http://localhost:8080/predict | jq
{
"text": "very cool restaurant!",
"is_positive_review": 1,
"details": {
"mlflow_run_id": "3acade02674549b19044a59186d97db4",
"inference_elapsed": 0.0006744861602783203
}
}
This service is running in Kubernetes as a 1-replica deployment with a service providing load balancing with a static internal IP, so, if needed, it can be easily scaled horizontally. Thanks @artem-yushkovsky 🎉
In order to add some business-logic to the model deployment, we implemented an additional abstraction layer - the model proxy. It's a thicker REST API service with access to a PostgreSQL database to store and serve prediction results. This service accesses the model via REST API over internal network and calculates some small statistics on the prediction correctness:
$ curl -s -X POST -H "Content-Type: application/json" -d '{"text": "very good cafe", "is_positive_user_answered": true}' http://model-proxy.lab1-team3.mlops.mlops.neu.ro/predictions | jq
{
"id": 40,
"text": "very good cafe",
"is_positive": {
"user_answered": true,
"model_answered": true
},
"details": {
"mlflow_run_id": "3acade02674549b19044a59186d97db4",
"inference_elapsed": 0.0009300708770751953,
"timestamp": "2021-02-04T20:46:49.484379"
}
}
$ curl -s http://model-proxy.lab1-team3.mlops.neu.ro/predictions | jq
[
...
{
"text": "very good cafe",
"is_positive_user_answered": true,
"is_positive_model_answered": true,
"mlflow_run_id": "3acade02674549b19044a59186d97db4",
"inference_elapsed": 0.0009300708770751953,
"timestamp": "2021-02-04T20:46:49.484379",
"id": 40
},
...
]
$ curl -s http://model-proxy.lab1-team3.mlops.neu.ro/statistics | jq
{
"statistics": {
"correctness_rate": 0.85
}
}
Though this service does not implement any kind of authentication, and though its statistics calculation is rather straightforward (also, we would consider a distributed logging system based on ELK stack a better solution for adding business-level metadata to the model server), it serves the demo purposes well. Please find the code in ./model_proxy. Kudos @artem-yushkovsky! 🥳
In order to add more MLOps flow to the project, we decided to implement a microservice that implements GitOps for MLflow: meet Model Operator! This service follows the Kubernetes Operator pattern and constantly polls MLflow server to see which model has Production
tag. Once this tag has changed, it changes the Model Server deployment in Kubernates and thus re-deploys the model. To illustrate this process, we exposed the logs of this service via Webtail.
For example, we want to deploy the recently trained model of version 10. In MLflow UI, we change its Stage
to Production
:
The Model Operator notices that the model has changed:
and it automatically triggers the model deployment:
NAME READY UP-TO-DATE AVAILABLE AGE
deployment.apps/mlflow-model-operator 1/1 1 1 2m15s
deployment.apps/mlflow-model-proxy 1/1 1 1 19h
deployment.apps/mlflow-model-server 0/1 1 0 5s
NAME READY STATUS RESTARTS AGE
pod/mlflow-model-operator-6b988bcdbb-hbkjz 3/3 Running 0 2m15s
pod/mlflow-model-proxy-646d4f9cd6-9xtmc 1/1 Running 0 19h
pod/mlflow-model-server-65fd74f845-nvbwp 1/1 Terminating 0 95s
pod/mlflow-model-server-75869b74fb-6cvhx 0/1 Init:1/2 0 5s
NAME DESIRED CURRENT READY AGE
replicaset.apps/mlflow-model-operator-6b988bcdbb 1 1 1 2m15s
replicaset.apps/mlflow-model-proxy-646d4f9cd6 1 1 1 19h
replicaset.apps/mlflow-model-server-75869b74fb 1 1 0 5s
NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
service/mlflow-model-operator LoadBalancer 10.104.2.23 34.91.157.40 80:32328/TCP 16h
service/mlflow-model-proxy LoadBalancer 10.104.14.129 34.91.75.82 80:31813/TCP 20h
service/mlflow-model-server ClusterIP 10.104.5.80 <none> 80/TCP 21h
To be honest, we were surprised not to find this kind of GitOps solution for MLflow+Kubernetes, and we keep believing that it exists but not yet discovered. Also, we need to mention that current solution disables the served model for a few minutes during the re-deployment process. There are other solutions, for example, Seldon, that implement zero-downtime model deployment with many other perks, but this goes beyond current demo project. Please find the Model Operator service code in ./model_operator. Nice job @artem-yushkovsky! 👾
During the work on this lab, we explored multiple awesome and open tools, we managed to create a complete MLOps pipeline from model development to model serving (note: data management and model monitoring are excluded from the scope of current project). We learned many things:
- how to plan an ML project,
- how to use your free tier credits to set up the infrastructure,
- how to spend a few hours to develop a pytorch model and a few days debugging it,
- how to build complex deployment solutions on top of kubernetes,
- how to convert abstract discussions around the tech to a POC product,
- and many other cool things! 🎉