Gradient Workflows Tutorial
Welcome to Gradient Workflows. In this tutorial we'll cover everything you need to know to start automating machine learning tasks and creating reproducible and production-ready pipelines.
Last updated
Welcome to Gradient Workflows. In this tutorial we'll cover everything you need to know to start automating machine learning tasks and creating reproducible and production-ready pipelines.
Last updated
Gradient Workflows provides a simple way to automate machine learning tasks. Workflows brings together powerful Gradient GPU instances with an expressive syntax to generate production-ready machine learning pipelines with a few lines of code.
Gradient Workflows is based on Argo Workflows, which is an open-source and container-native delivery tool for Kubernetes.
Several MLOps platforms on the market promise the ability to define pipelines with code. What Gradient Workflows does best is allow us to manage multiple layers of complexity -- from machine learning code, to development and production dependencies, to compute infrastructure such as GPU nodes and storage providers -- all from a single pane of glass.
Here are some other features that distinguish Gradient Workflows from other MLOps solutions:
Use any Gradient instance (including free instances) within a Workflow
Define a type of compute instance for each step of a Workflow to improve performance and conserve costs
Connect Workflows to GitHub and trigger Workflow runs from a git commit
Write your own jobs for use within Gradient Workflows
Don't worry if this doesn't make sense yet. Workflows has a steeper learning curve than Notebooks due to the natural complexities of shipping machine learning applications to production.
As we progress through the tutorial, we'll start with a basic use case and add complexity gradually. This way we can get a full understanding of Workflows from a simple toy pipeline through to a production application that is deterministic, reproducible, and fit for external consumption.
Let's get started!
The first task in this tutorial will be to create and run a sample workflow on Gradient. We will use one of the starter workflows that Gradient provides in the console.
By the end of this section, we will have used a workflow to train a model to generate a realistic human face using StyleGAN2, which is a popular library for generative image modeling from NVIDIA.
Here is what we will do:
Create a new project in Gradient
Connect this Gradient project to GitHub so that we can version the workflow in source control
Create a new workflow that copies and runs a simple StyleGAN2 demo
Inspect the results and confirm that we find machine-generated images of human faces
If we haven't already created a project in the Gradient console, we need to do that first. From the Gradient console, select Create A Project
and give your project a name.
Here we have named our project Face Generator. (In Part 2 we will rename the project as we generate more than just faces.)
From the Workflows
tab we'll now select Create
to initialize our first Workflow.
At this point Gradient will prompt us to grant access to GitHub if we have not already done so.
We need to give Gradient access to at least one GitHub repository during this step. The reason for this is that Gradient Workflows is designed to keep our code version-controlled during the application development process and during Part 1 of this tutorial we'll be writing our Gradient Workflow to a new repository in GitHub automatically.
In Part 2, we'll be using the GitHub integration to trigger workflow runs. This is an extremely powerful MLOps capability that we'll explore when we push code from our local machine to our remote GitHub repository.
Let's go ahead and make the GitHub connection. When we connect GitHub, we'll unlock the Workflows UI which will allow us to proceed.
Please note that Gradient does not require access to an entire GitHub organization or account. Granting access to a single repo is enough to grant write privileges to Gradient for the purposes of this tutorial.
When prompted, we'll grant access to a single GitHub repo. We may also grant access to multiple repositories or all repositories at this time.
Once we connect GitHub, we will unlock the Get started with workflows
page.
We now have the option to run either a sample workflow or to set up our own custom workflow.
We will make a custom workflow later in this tutorial -- for now we will select the StyleGAN2 template tile.
We'll then select our GitHub account under Select account or organization
.
We'll also give our workflow a Repository name
for writing to GitHub. In our case we will name our repository "Gradient Workflows Tutorial."
We can now select the Create Workflow
button after which we should see the new workflow begin to initialize:
As the sample workflow initializes, we'll notice that the workflow consists of QTY 2 jobs:
CloneRepo
StyleGan2
Each job represents a discrete action or step in the workflow syntax (which we will inspect shortly). In the UI, each job is represented by a light green box. We can reveal more information about the job -- including logs, data, and so forth -- by clicking into it.
The workflow has completed successfully when the message reads Succeeded
in the left sidebar. To confirm this, we can see that the boxes containing each of our jobs (CloneRepo
and StyleGan2
) now display the checkmark symbol.
Each job within Gradient Workflows has its own set of logs. To reach the logs click into each of the green boxes and tab over to the Logs
section.
It is extremely useful for debugging to have discrete logs for each step of a workflow. This level of granularity means that we can isolate errors to a single job. Since ML pipelines have historically been difficult to debug, logging each step of a workflow independently is helpful.
If we examine the logs of the second job, StyleGan2
, we should see something like this:
It appears that our face generator has succeeded in generating an image. Fantastic!
Let's leave the Logs
tab for the moment and view the Data
tab within the StyleGAN2
job. Here we'll find the seed images that we have just generated.
Amazing! If our training succeeded, we should find the seed image named seed6600.png
should look exactly like the image above. Reproducibility for the win!
Here are a few more images that we were able to generate in a short amount of time. Each of these faces does not exist in real life -- we've just used Gradient Workflows to perform some generative image modeling to create them!
Now that we've successfully run our first workflow, let's check in on the GitHub repository that we created when we kicked off this workflow.
Our entire Workflow is visible in GitHub, which comes in handy. In the next part of this tutorial we'll learn how to work with the GitHub repository.
At this point here is everything we've done:
We created a new project and workflow
We connected our GitHub account to Gradient allowing us to write our workflow to GitHub
We ran a starter workflow that contained two jobs
We successfully ran both jobs and generated a number of images of lifelike human faces
We confirmed that the contents of the workflow is available in GitHub
Up next we're going to customize the workflow to support a more complex use case. We will be creating QTY 7 jobs and we will produce more interesting results -- this time by generating images of cats instead of images of people!
Now that we've successfully generated our first StyleGAN2 workflow, we're going to create a more advanced implementation that increases the complexity of the workflow.
We'll be running QTY 3 jobs to start while demonstrating how to use the GitHub integration to trigger workflow runs. We'll also scale-up the workflow to QTY 5 jobs.
This time we'll be modifying the YAML file that was created automatically in Part 1. To verify the existence of the YAML file, let's visit the repository we created in Part 1 and navigate to the file located at Gradient-Workflows-Tutorial/.gradient/workflows/stylegan2.yaml
.
Excellent! As we'll come to learn, this YAML file contains instructions for each workflow job in a format that Gradient Workflows can understand. We'll cover that syntax and more in this part of the tutorial.
Here are the steps we'll be taking to modify our workflow:
Clone our existing connected GitHub repository to our local machine
Create empty datasets for the workflow
Copy some new text into the YAML file
Push the updated repository to GitHub to trigger a new workflow run
Repeat the process once again to add another step
The first thing we're going to do is clone the repo that we created in Part 1 onto our local machine. While it's certainly possible to modify code files directly in the GitHub user interface, we greatly prefer to work with files locally and then push them to source control.
We named our Workflow Gradient-Workflows-Tutorial
previously so our git clone command will look like this. Yours will be different depending on the GitHub account you connected to Gradient and the name you gave to your first Workflow.
Our particular command looks like this:
However yours may looks like this:
Let's go ahead and clone the repo locally. If you need a refresher on how cloning from source control works, we recommend checking out GitHub's excellent documentation.
After we clone the repo locally, we should cd
into the local directory and ls
to reveal a directory structure like this:
This directory contains a number of files that we created during Part 1 of the tutorial.
If we use the ls -a
command, we will reveal a number of hidden directories including .git
, .gitignore
, and .gradient
directories.
Our YAML file is located within the .gradient
directory and we can cd
into it and ls
to confirm that it contains a workflows
directory that contains our YAML file.
Now that we know we have the right files, let's open the repo in whatever local IDE we prefer.
In this tutorial we'll be using VSCode:
The file that we're looking at in this image is the YAML file responsible for orchestrating the workflow we ran in Part 1. The file is located in the directory .gradient/workflows
within the repo.
If we inspect the file, we see that there are QTY 2 jobs specified in this YAML file. One is called CloneRepo
and the other is called StyleGAN2
. This is exactly what we expected to see since we can recall these two jobs running in Part 1.
What we will do next is modify this YAML file to add a number of steps to our workflow. Whenever we make a change to this repo and push it to the main
branch we will trigger a new workflow run.
In the next step we're going to do exactly that.
Next up we're going to be copying a new YAML over to our local repo and then pushing our repo to GitHub to trigger a new workflow run.
The new YAML file that we need to copy and paste into our workflow project is available at this URL:
Let's go ahead and copy that text and paste it into the file on our local machine located at .gradient/workflows/stylegan2.yaml
:
As you can see, we have copied the contents of the YAML file linked above and pasted it in its entirety into the YAML file located at .gradient/workflows/stylegan2.yaml
.
Before we push this modified YAML to GitHub, let's quickly understand what it's going to do.
We're now going to be talking about the new 5-step YAML file that we've pasted into our repo on our local machine. After we understand what's in it, we're going to push it to GitHub to trigger a workflow run in Gradient.
The first thing we can see is that our on
block is commented out. We will need to uncomment this block before we push the file to GitHub if we're going to run the new workflow.
The rest of the file specifies QTY 5 distinct jobs. This is a substantial increase from the first tutorial in which we created a total of QTY 2 jobs.
Each job is specified under the jobs
heading in the file.
cloneStyleGAN2Repo
- uses a C4 instance type (a basic CPU machine) to clone the StyleGAN2 source code from NVIDIA's GitHub repo into a managed storage provider on Gradient
getPretrainedModelCats
- uses a C4 instance type to copy a pretrained model of cats (a large .pkl
file) over to Gradient
generateCatImagesPretrainedModel
- uses a P4000 instance type to display images of cats generated using the pretrained cats model
[Commented out to start] getPretrainedModelCars
- uses a C4 instance type to copy a pretrained model of cars (a large .pkl
file) over to Gradient
[Commented out to start] generateCarImagesPretrainedModel
- uses a P4000 instance type to display images of cars generated using the pretrained cars model
One of the best parts of Gradient Workflows is that we can schedule an arbitrary number of jobs on different compute instances -- here we are doing just that. As we can see we are starting with jobs #4 and #5 commented out -- so our first run will only run QTY 3 jobs and we can make additions from there.
We can run a job with no dependencies, or external dependencies, or we can require that a job wait for a previous job to complete before running. Each job can consume different types of inputs, produce different types of outputs, and accept different arguments at runtime. Workflows even allows each job to load a distinct Docker image from a public or private repo!
Our next step is to push our local repo to the GitHub remote repository to kick off a new training run. Let's do that now.
One of the things we may have noticed in the YAML file is the appearance of Gradient Datasets. Here's an example from #2 Get pretrained cat model
:
In this code block, Gradient is expecting to see a Gradient Dataset called stylegan2-wsp-cats-pretrained-network
-- so we'll have to create it!
We'll quickly create QTY 4 datasets now, which should only take a minute or two.
First we'll navigate over to the Data
tab in Gradient. It should look like this:
Next, we'll create our new empty dataset. If we read through the YAML file, we can see that we'll need a total of QTY 4 empty datasets.
Those are as follows:
We create the empty dataset like so:
All we're doing here is using each of the QTY 4 references listed above as the Name
parameter when creating a new dataset.
For the Storage Provider
parameter, we're using Gradient Managed, which is the name of the default Gradient cluster.
Once we have these empty datasets created, it's time to kickoff the workflow!
Let's go ahead and run the Workflow now.
First, in our local text editor, we'll uncomment the command at the top of the new YAML file. If we do not uncomment the on
block our workflow will not run.
It should now look like this:
After we save the file, we'll push the change to our remote repository on GitHub.
We use the terminal to push the change using these commands -- although of course we can use any technique we like to push code to remote source control:
After we push to GitHub we should confirm that our remote repo received the changes.
Next we're going to swap back from our GitHub repo to our Gradient console. If everything is working as expected, we should see a new workflow run commence.
Fantastic! We can see that we have successfully pushed code to our remote repo, and we can see that pushing this code has automatically created a new workflow run in Gradient. Perfect!
When our training is done (it took us less than 2 minutes but times may vary), we should see that all QTY 3 jobs have succeeded as represented by the green boxes:
What we'll do next is inspect the results of our step called generateCatImagesPretrainedModel
-- our expectation is to find images of cats within the dataset.
To inspect the generated images, we select the box in the DAG that represents the generation step and we select the DATA
tab to explore the dataset that we've just generated. Inside we should find a number of seed images of cats.
Because this is a deterministic process, the cats you generate should look exactly like these cats. Try it out for yourself!
Excellent!
If we look at a number of our seed images we should see some cats that look exactly like this:
Our next step is to expand this workflow from QTY 3 steps to QTY 5 steps. We'll do that by uncommenting steps 4 and 5 from our YAML file to trigger a new workflow run.
Let's return to our YAML file locally and uncomment steps 4 and 5. This is going to add an addition QTY 2 steps to our workflow and trigger a re-run and the result is that we're going to also generate some cars.
Next we'll push the changes from our local machine back to our remote repository. Here we give the commit the unimaginative title uncomment steps 4 and 5
.
We should then see this commit hit our GitHub repository. In the GIF below we can see that our checks pass in Gradient, the workflow is triggered, and then swapping over to Gradient we see that we now have a workflow running with QTY 5 steps in the DAG!
We've just kicked off a 5-step workflow run -- fantastic!
Let's inspect the results once the workflow has finished. This time it may take a little bit longer since we've added two extra steps.
If we've succeeded, in addition to our cats, we should now see that we've generated a number of cars!
Success once again!
In Part 2 of this tutorial, we've managed to do the following:
Pull our workflow repo locally so that we can make changes to the YAML file
Made one change to the YAML to create QTY 3 workflow jobs to produce a dataset of cats
Made another change to the YAML to create QTY 5 workflow jobs to produce a dataset of cats and of cars
We've now seen how to take a workflow and make changes to the YAML file to add complexity. From here, as you might imagine, we can get as complicated as we would like.
From here, we should feel like we have the tools and resources to get started building modifying the YAML spec to build our own Gradient Workflows.
As we develop our own Gradient Workflows, we might notice that a git commit does not trigger a workflow run. When this happens, we'll want to pay attention to the errors generated in GitHub when making a commit.
If we are successful triggering a workflow run, but the workflow itself is erroring, we'll want to read through the logs generated in the Gradient console. We should remember to inspect each job in the DAG and read the logs independently in order to troubleshoot.
If you have any questions or need a hand getting something to work -- please let us know! You can file a support ticket or reach us on Twitter @hellopaperspace.
Read through the Gradient Workflows documentation
In particular, read through the Gradient Workflows spec
Browse a large number of Gradient tutorials and walkthroughs available on the Paperspace blog
End-to-end StyleGAN2 tutorial blogpost and GitHub repo