Blog
Experiment Tracking with Dvc

Experiment Tracking with DVC

Leveraging DVC for Streamlined ML Experimentation

DVC Experiments View

DVC Experiments View

In the realm of machine learning, managing and tracking experiments can swiftly become a complex task, especially as projects scale. The introduction of tools like DVC (Data Version Control)1 has been a game-changer, allowing developers and data scientists to automate and streamline this process. Through a practical example involving a Random Forest Classifier trained on the Iris dataset, we'll explore how DVC facilitates a seamless experiment tracking system.

Join the AI BootCamp!

Ready to dive into the world of AI and Machine Learning? Join the AI BootCamp to transform your career with the latest skills and hands-on project experience. Learn about LLMs, ML best practices, and much more!

Why Track Experiments?

Imagine you're working on a machine learning project. As your experiments grow in number, so does the difficulty of tracking each one's parameters, metrics, and outcomes. Without proper management, this complexity can lead to significant overhead and confusion, potentially stalling project progress.

DVC offers a solution to this problem by integrating code and data version control, making it simpler to reproduce experiments and track changes across project iterations. The dvclive module, in particular, enables automatic metric logging during model training, which is crucial for comparing experiments and understanding model performance over time.

Basic Tracking with DVCLive

DVCLive2 is a lightweight Python library that integrates seamlessly with DVC to log metrics during model training. By adding a few lines of code to your training script, you can automatically capture key metrics such as accuracy, precision, and recall, and visualize them in the DVC Experiments View.

app.py
from dvclive import Live
from sklearn import datasets
from sklearn.ensemble import RandomForestClassifier
from sklearn.metrics import accuracy_score, average_precision_score, recall_score
from sklearn.model_selection import train_test_split
 
 
def evaluate(model: RandomForestClassifier, X, y, split: str, live: Live):
    y_pred = model.predict(X)
    live.log_metric(f"{split}/accuracy", accuracy_score(y, y_pred))
    live.log_metric(
        f"{split}/average_precision",
        average_precision_score(y, model.predict_proba(X), average="macro"),
    )
    live.log_metric(f"{split}/recall", recall_score(y, y_pred, average="macro"))
 
 
X, y = datasets.load_iris(as_frame=True, return_X_y=True)
 
X_train, X_test, y_train, y_test = train_test_split(
    X, y, test_size=0.2, random_state=42
)
 
with Live("experiments") as live:
    model = RandomForestClassifier(n_estimators=5, max_depth=1, random_state=42)
    model.fit(X_train, y_train)
    evaluate(model, X_train, y_train, "train", live)
    evaluate(model, X_test, y_test, "test", live)

We're training a RandomForestClassifier on the Iris dataset, evaluating its performance, and logging the results using the dvclive library. Key components of the code include:

  • Loading the Iris dataset: Utilizes sklearn.datasets.load_iris to fetch the Iris dataset, which is divided into training and testing sets
  • Model Training: A RandomForestClassifier is instantiated with specific hyperparameters (n_estimators=5, max_depth=1) and trained on the training data
  • Evaluation and Logging: The evaluate function is called for both training and test sets. It logs metrics (accuracy, average precision, and recall) using the dvclive library
  • dvclive Integration: The Live context manager from dvclive is used to initialize a logging environment, which automatically tracks and saves the evaluation metrics

You can run this script using the command python app.py and view the results with:

dvc exp show --md
Experimenttrain.accuracytrain.average_precisiontrain.recalltest.accuracytest.average_precisiontest.recall
workspace0.6750.666670.666670.633330.666670.66667
master------
└── c3b1f3b [bitty-walk]0.6750.666670.666670.633330.666670.66667

The table contains our first experiment's results, showing the accuracy, average precision, and recall scores for both the training and test sets.

Let's run another experiment by changing the hyperparameters of the model:

model = RandomForestClassifier(n_estimators=5, max_depth=1, random_state=42)

Let's look at the results:

dvc exp show --md
Experimenttrain.accuracytrain.average_precisiontrain.recalltest.accuracytest.average_precisiontest.recall
workspace0.950.911360.94913111
master------
├── 7397a99 [irony-awns]0.950.911360.94913111
└── c3b1f3b [bitty-walk]0.6750.666670.666670.633330.666670.66667

Our second experiment, with different hyperparameters, shows improved metrics. You can commit these changes to git.

Create a Pipeline

To streamline the experiment tracking process, we can create a DVC pipeline that automates the training and evaluation steps. Another great feature of pipelines is that they allow you to run multiple experiments with different hyperparameters in a single command. This is particularly useful when performing hyperparameter tuning or grid search.

Let's start by creating a params.yaml file to store our model hyperparameters:

params.yaml
model:
  n_estimators: 10
  max_depth: 2

Next, we'll modify our training script to read these hyperparameters from the file:

app.py
from pathlib import Path
 
from box import ConfigBox
from dvclive import Live
from ruamel.yaml import YAML
from sklearn import datasets
from sklearn.ensemble import RandomForestClassifier
from sklearn.metrics import accuracy_score, average_precision_score, recall_score
from sklearn.model_selection import train_test_split
 
config = ConfigBox(YAML(typ="safe").load(Path("params.yaml").open(encoding="utf-8")))
 
 
def evaluate(model: RandomForestClassifier, X, y, split: str, live: Live):
    y_pred = model.predict(X)
    live.log_metric(f"{split}/accuracy", accuracy_score(y, y_pred))
    live.log_metric(
        f"{split}/average_precision",
        average_precision_score(y, model.predict_proba(X), average="macro"),
    )
    live.log_metric(f"{split}/recall", recall_score(y, y_pred, average="macro"))
 
 
X, y = datasets.load_iris(as_frame=True, return_X_y=True)
 
X_train, X_test, y_train, y_test = train_test_split(
    X, y, test_size=0.2, random_state=42
)
 
with Live("experiments") as live:
    model = RandomForestClassifier(
        n_estimators=config.model.n_estimators,
        max_depth=config.model.max_depth,
        random_state=42,
    )
    model.fit(X_train, y_train)
    evaluate(model, X_train, y_train, "train", live)
    evaluate(model, X_test, y_test, "test", live)
 

The script now reads the hyperparameters from the params.yaml file and uses them to train the model. To run the pipeline, we need to define a stage in a dvc.yaml file:

dvc.yaml
stages:
  train_model:
    cmd: python app.py
    deps:
      - app.py
    params:
      - model

This stage runs the app.py script, which trains the model using the hyperparameters specified in params.yaml. To execute the pipeline, run:

dvc exp run
Experimenttrain.accuracytrain.average_precisiontrain.recalltest.accuracytest.average_precisiontest.recallmodel.n_estimatorsmodel.max_depth
workspace0.950.987840.94955111102
master------102
└── 04301de [funny-cyma]0.950.987840.94955111102

Note that the model hyperparameters are now stored in the experiment metadata, making it easier to track and compare different experiments. Let's try to run another experiment with different hyperparameters:

dvc exp run -S model.n_estimators=30 -S model.max_depth=3
Experimenttrain.accuracytrain.average_precisiontrain.recalltest.accuracytest.average_precisiontest.recallmodel.n_estimatorsmodel.max_depth
workspace0.966670.99720.96665111303
master------102
├── 83c1c6b [choky-door]0.966670.99720.96665111303
└── 04301de [funny-cyma]0.950.987840.94955111102

Grid Search

Another powerful feature of DVC is the ability to run grid search experiments. You can run multiple experiments in parallel using the --queue flag. Here's an example:

dvc exp run --queue -S model.n_estimators=8,16,32 -S model.max_depth=2,3,5

This command will queue nine experiments with different combinations of n_estimators and max_depth. To run all the queued experiments, use:

dvc exp run --run-all

Here's a summary of the grid search results:

Experimenttrain.accuracytrain.average_precisiontrain.recalltest.accuracytest.average_precisiontest.recallmodel.n_estimatorsmodel.max_depth
workspace0.966670.99720.96665111303
master------102
├── d37b551 [above-play]111111325
├── 00d0021 [nicer-cere]0.9750.99680.97478111323
├── 2a77fa6 [wacky-bind]0.958330.993280.9581111322
├── 004a3e9 [podgy-arcs]111111165
├── 8236073 [oared-snow]0.966670.997190.96623111163
├── e3edf66 [flown-nosh]0.950.988850.94955111162
├── db393d5 [lento-bice]0.9916710.9914511185
├── 861f1f0 [sedgy-over]0.958330.995620.9585211183
├── 9503b13 [fetal-grub]0.958330.985930.9585211182
├── 83c1c6b [choky-door]0.966670.99720.96665111303
└── 04301de [funny-cyma]0.950.987840.94955111102

You can read about other approaches to hyperparameter tuning, such as ranges as values, on the official DVC documentation (opens in a new tab).

Conclusion

The shift towards using tools like DVC for experiment tracking represents a significant advancement in the field of machine learning. It not only simplifies the management of experiments but also enhances reproducibility and collaboration among team members. By integrating these tools into your workflow, you can dramatically reduce the time spent on experiment management and focus more on innovation and model improvement.

3,000+ people already joined

Join the The State of AI Newsletter

Every week, receive a curated collection of cutting-edge AI developments, practical tutorials, and analysis, empowering you to stay ahead in the rapidly evolving field of AI.

I won't send you any spam, ever!

References

Footnotes

  1. DVC (opens in a new tab)

  2. DVCLive (opens in a new tab)