Skip to content

Logo Logo

Cloud deployment


Core Module

We are now returning to using the cloud. In this module, you should have gone through the steps of having your code in your GitHub repository to automatically build into a docker container, store that, store data and pull it all together to make a training run. After the training is completed you should hopefully have a file stored in the cloud with your trained model weights.

Today's exercises will be about serving those model weights to an end user. We focus on two different ways of deploying our model: Google cloud functions and Google cloud run. Both services are serverless, meaning that you do not have to manage the server that runs your code.

Image

GCP in general has 5 core deployment options. We are going to focus on Cloud Functions and Cloud Run, which are two of the serverless options. In contrast to these two, you have the option to deploy to Kubernetes Engine and Compute Engine which are more traditional ways of deploying your code. Here you have to manage the underlying infrastructure.

Cloud Functions

Google Cloud Functions, is the most simple way that we can deploy our code to the cloud. As stated above, it is a serverless service, meaning that you do not have to worry about the underlying infrastructure. You just write your code and deploy it. The service is great for small applications that can be encapsulated in a single script.

❔ Exercises

  1. Go to the start page of Cloud Functions. Can be found in the sidebar on the homepage or you can just search for it. Activate the service in the cloud console or use the following command:

    gcloud services enable cloudfunctions.googleapis.com
    
  2. Click the Create Function button which should take you to a screen like the image below. Make sure it is a 2nd Gen function, give it a name, set the server region to somewhere close by and change the authentication policy to Allow unauthenticated invocations so we can access it directly from a browser. Remember to note down the

    Image

  3. On the next page, for Runtime pick the Python 3.11 option (or newer). This will make the inline editor show both a main.py and requirements.py file. Look over them and try to understand what they do. Especially, take a look at the functions-framework which is a needed requirement of any Cloud function.

    Image

    After you have looked over the files, click the Deploy button.

    Solution

    The functions-framework is a lightweight, open-source framework for turning Python functions into HTTP functions. Any function that you deploy to Cloud Functions must be wrapped in the @functions_framework.http decorator.

  4. Afterwards, the function should begin to deploy. When it is done, you should see ✅. Now let's test it by going to the Testing tab.

    Image

  5. If you know what the application does, it should come as no surprise that it can run without any input. We therefore just send an empty request by clicking the Test The Function button. Does the function return the output you expected? Wait for the logs to show up. What do they show?

    1. What should the Triggering event look like in the testing prompt for the program to respond with

      Hallo General Kenobi!
      

      Try it out.

      Solution

      The default triggering event is a JSON object with a key name and a value. Therefore the triggering event should look like this:

      {
          "name": "General Kenobi"
      }
      
    2. Go to the trigger tab and go to the URL for the application. Execute the API a couple of times. How can you change the URL to make the application respond with the same output as above?

      Solution

      You can change the URL to include a query parameter name with the value General Kenobi. For example

      https://us-central1-my-personal-mlops-project.cloudfunctions.net/function-3?name=General%20Kanobi
      

      where you would need to replace everything before the ? with your URL.

    3. Click on the metrics tab. You should hopefully see it being populated with a few data points. Identify what each panel is showing.

      Solution
      • Invocations/Second: The number of times the function is invoked per second
      • Execution time (ms): The time it takes for the function to execute in milliseconds
      • Memory usage (MB): The memory usage of the function in MB
      • Instance count (instances): The number of instances that are running the function
    4. Check out the logs tab. You should see that your application has already been invoked multiple times. Also, try to execute this command in a terminal:

      gcloud functions logs read
      
  6. Next, we are going to create our own application that takes some input so we can try to send it requests. We provide a very simple script to get started.

    Simple script

    sklearn_cloud_functions.py
    # Load data
    import pickle
    
    import numpy as np
    from sklearn import datasets
    from sklearn.neighbors import KNeighborsClassifier
    
    iris_x, iris_y = datasets.load_iris(return_X_y=True)
    
    # Split iris data in train and test data
    # A random permutation, to split the data randomly
    np.random.seed(0)
    indices = np.random.permutation(len(iris_x))
    iris_x_train = iris_x[indices[:-10]]
    iris_y_train = iris_y[indices[:-10]]
    iris_x_test = iris_x[indices[-10:]]
    iris_y_test = iris_y[indices[-10:]]
    
    # Create and fit a nearest-neighbor classifier
    
    knn = KNeighborsClassifier()
    knn.fit(iris_x_train, iris_y_train)
    knn.predict(iris_x_test)
    
    # save model
    
    with open("model.pkl", "wb") as file:
        pickle.dump(knn, file)
    
    1. Figure out what the script does and run the script. This should create a file with a trained model.

      Solution

      The file trains a simple KNN model on the iris dataset and saves it to a file called model.pkl.

    2. Next, create a storage bucket and upload the model file to the bucket. Try to do this using the gsutil command and check afterward that the file is in the bucket.

      Solution
      gsutil mb gs://<bucket-name>  # mb stands for make bucket
      gsutil cp model.pkl gs://<bucket-name>  # cp stands for copy
      

      where you replace <bucket-name> with the name of your bucket.

    3. Create a new cloud function with the same initial settings as the first one, e.g. Python 3.11 and HTTP. Then implement in the main.py file code that:

      • Loads the model from the bucket
      • Takes a request with a list of integers as input
      • Returns the prediction of the model

      In addition to writing the main.py file, you also need to fill out the requirements.txt file. You need at least three packages to run the application. Remember to also change the Entry point to the name of your function. If your deployment fails, try to go to the Logs Explorer page in gcp which can help you identify why.

      Solution

      The main script should look something like this:

      main.py
      import pickle
      
      import functions_framework
      from google.cloud import storage
      
      BUCKET_NAME = "my_sklearn_model_bucket"
      MODEL_FILE = "model.pkl"
      
      client = storage.Client()
      bucket = client.get_bucket(BUCKET_NAME)
      blob = bucket.get_blob(MODEL_FILE)
      my_model = pickle.loads(blob.download_as_string())
      
      
      @functions_framework.http
      def knn_classifier(request):
          """Simple knn classifier function for iris prediction."""
          request_json = request.get_json()
          if request_json and "input_data" in request_json:
              input_data = request_json["input_data"]
              input_data = [float(in_data) for in_data in input_data]
              input_data = [input_data]
              prediction = my_model.predict(input_data)
              return {"prediction": prediction.tolist()}
          return {"error": "No input data provided."}
      

      And, the requirement file should look like this:

      functions-framework>=3.7.0
      google-cloud-storage>=2.14.0
      scikit-learn>=1.4.0
      

      importantly make sure that you are using the same version of scikit-learn as you used when you trained the model. Else when trying to load the model you will most likely get an error.

    4. When you have successfully deployed the model, try to make predictions with it. What should the request look like?

      Solution

      It depends on how exactly you have chosen to implement the main.py. But for the provided solution, the payload should look like this:

      {
          "data": [1, 2, 3, 4]
      }
      

      with the corresponding curl command:

      curl -X POST \
          https://your-cloud-function-url/knn_classifier \
          -H "Content-Type: application/json" \
          -d '{"input_data": [5.1, 3.5, 1.4, 0.2]}'
      
  7. Let's try to figure out how to do the above deployment using gcloud instead of the console UI. The relevant command is gcloud functions deploy. For this function to work you will need to put the main.py and requirements.txt in a separate folder. Try to execute the command to successfully deploy the function.

    Solution
    gcloud functions deploy <func-name> \
        --gen2 --runtime python311 --trigger-http --source <folder> --entry-point knn_classifier
    

    where you need to replace <func-name> with the name of your function and <folder> with the path to the folder containing the main.py and requirements.txt files.

  8. (Optional) You can finally try to redo the exercises by deploying a PyTorch application. You will essentially need to go through the same steps as the sklearn example, including uploading a trained model to storage and writing a cloud function that loads it and returns some output. You are free to choose whatever PyTorch model you want.

Cloud Run

Cloud functions are great for simple deployments, that can be encapsulated in a single script with only simple requirements. However, they do not scale with more advanced applications that may depend on multiple programming languages. We are already familiar with how we can deal with this through containers and Cloud Run is the corresponding service in GCP for deploying containers.

❔ Exercises

  1. We are going to start locally by developing a small app that we can deploy. We provide two small examples to choose from: first is a small FastAPI app consisting of a single Python script and a docker file. The second is a small Streamlit app (which you can learn more about in this module) consisting of a single docker file. You can choose which one you want to work with.

    Simple Fastapi app
    simple_fastapi_app.py
    from fastapi import FastAPI
    
    app = FastAPI()
    
    
    @app.get("/")
    def read_root():
        """Root endpoint."""
        return {"Hello": "World"}
    
    
    @app.get("/items/{item_id}")
    def read_item(item_id: int):
        """Get an item by id."""
        return {"item_id": item_id}
    
    simple_fastapi_app.dockerfile
    FROM python:3.11-slim
    
    EXPOSE $PORT
    
    WORKDIR /app
    
    RUN apt-get update && apt-get install -y \
        build-essential \
        software-properties-common \
        git \
        && rm -rf /var/lib/apt/lists/*
    
    RUN pip install fastapi
    RUN pip install pydantic
    RUN pip install uvicorn
    
    COPY simple_fastapi_app.py simple_fastapi_app.py
    
    CMD exec uvicorn simple_fastapi_app:app --port $PORT --host 0.0.0.0 --workers 1
    
    Simple Streamlit app
    streamlit_app.dockerfile
    FROM python:3.9-slim
    
    EXPOSE $PORT
    
    WORKDIR /app
    
    RUN apt-get update && apt-get install -y \
        build-essential \
        software-properties-common \
        git \
        && rm -rf /var/lib/apt/lists/*
    
    RUN git clone https://github.com/streamlit/streamlit-example.git .
    
    RUN pip3 install -r requirements.txt
    
    ENTRYPOINT ["streamlit", "run", "streamlit_app.py", "--server.port=$PORT", "--server.address=0.0.0.0"]
    
    1. Start by going over the files belonging to your choice app and understand what it does.

    2. Next, build the docker image belonging to the app

      docker build -f <dockerfile> . -t gcp_test_app:latest
      
    3. Next tag and push the image to your artifact registry

      docker tag gcp_test_app <region>-docker.pkg.dev/<project-id>/<registry-name>/gcp_test_app:latest
      docker push <region>-docker.pkg.dev/<project-id>/<registry-name>/gcp_test_app:latest
      

      Afterward, check your artifact registry contains the pushed image.

  2. Next, go to Cloud Run in the cloud console and enable the service or use the following command:

    gcloud services enable run.googleapis.com
    
  3. Click the Create Service button which should bring you to a page similar to the one below

    Image

    Do the following:

    • Click the select button, which will bring up all build containers and pick the one you want to deploy. In the future, you probably want to choose the Continuously deploy new revisions from a source repository such that a new version is always deployed when a new container is built.

    • Hereafter, give the service a name and select the region. We recommend choosing a region close to you.

    • Set the authentication method to Allow unauthenticated invocations such that we can call it without providing credentials. In the future, you may only set that authenticated invocations are allowed.

    • Expand the Container, Connections, Security tab and edit the port such that it matches the port exposed in your chosen application. If your docker file exposes the env variable $PORT you can set the port to anything.

    Finally, click the create button and wait for the service to be deployed (may take some time).

    Common problems

    If you get an error saying The user-provided container failed to start and listen on the port defined by the PORT environment variable. there are two common reasons for this:

    1. You need to add an EXPOSE statement in your docker container:

      EXPOSE 8080
      CMD exec uvicorn my_application:app --port 8080 --workers 1 main:app
      

      and make sure that your application is also listening on that port. If you hard code the port in your application (as in the above code) it is best to set it 8080 which is the default port for cloud run. Alternatively, a better approach is to set it to the $PORT environment variable which is set by cloud run and can be accessed in your application:

      EXPOSE $PORT
      CMD exec uvicorn my_application:app --port $PORT --workers 1 main:app
      

      If you do this and then want to run locally you can run it as:

      docker run -p 8080:8080 -e PORT=8080 <image-name>:<image-tag>
      
    2. If you are serving a large machine-learning model, it may also be that your deployed container is running out of memory. You can try to increase the memory of the container by going to the Edit container and the Resources tab and increasing the memory.

  4. If you manage to deploy the service you should see an image like this:

    Image

    You can now access your application by clicking the URL. This will access the root of your application, so you may need to add / or /<path> to the URL depending on how the app works.

  5. Everything we just did in the console UI we can also do with the gcloud run deploy. How would you do that?

    Solution

    The command should look something like this

    gcloud run deploy <service-name> \
        --image <image-name>:<image-tag> --platform managed --region <region> --allow-unauthenticated
    

    where you need to replace <service-name> with the name of your service, <image-name> with the name of your image and <region> with the region you want to deploy to. The --allow-unauthenticated flag is optional but is needed if you want to access the service without providing credentials.

  6. After deploying using the command line, make sure that the service is up and running by using these two commands

    gcloud run services list
    gcloud run services describe <service-name> --region <region>
    
  7. Instead of deploying our docker container using the UI or command line, which is a manual operation, we can do it continuously by using cloudbuild.yaml file we learned about in the previous section. This is called continuous deployment, and it is a way to automate the deployment process.

    Image
    Image credit

    Let's revise the cloudbuild.yaml file from the artifact registry exercises in this module which will build and push a specified docker image.

    cloudbuild.yaml

    cloudbuild.yaml
    steps:
    - name: 'gcr.io/cloud-builders/docker'
      id: 'Build container image'
      args: [
        'build',
        '.',
        '-t',
        'europe-west1-docker.pkg.dev/$PROJECT_ID/<registry-name>/<image-name>',
        '-f',
        '<path-to-dockerfile>'
      ]
    - name: 'gcr.io/cloud-builders/docker'
      id: 'Push container image'
      args: [
        'push',
        'europe-west1-docker.pkg.dev/$PROJECT_ID/<registry-name>/<image-name>'
      ]
    options:
      logging: CLOUD_LOGGING_ONLY
    

    Add a third step to the cloudbuild.yaml file that deploys the container image to Cloud Run. The relevant service you need to use is called 'gcr.io/cloud-builders/gcloud' and the command is 'gcloud run deploy'. Afterwards, reuse the trigger you created in the previous module or create a new one to build and deploy the container image continuously. Confirm that this works by making a change to your application and pushing it to GitHub and see if the application is updated continuously.

    Solution

    The full cloudbuild.yaml file should look like this:

    steps:
    - name: 'gcr.io/cloud-builders/docker'
      id: 'Build container image'
      args: [
        'build',
        '.',
        '-t',
        'europe-west1-docker.pkg.dev/$PROJECT_ID/<registry-name>/<image-name>',
        '-f',
        '<path-to-dockerfile>'
      ]
    - name: 'gcr.io/cloud-builders/docker'
      id: 'Push container image'
      args: [
        'push',
        'europe-west1-docker.pkg.dev/$PROJECT_ID/<registry-name>/<image-name>'
      ]
    - name: 'gcr.io/cloud-builders/gcloud'
      id: 'Deploy to Cloud Run'
      args: [
        'run',
        'deploy',
        '<service-name>',
        '--image',
        'europe-west1-docker.pkg.dev/$PROJECT_ID/<registry-name>/<image-name>',
        '--region',
        'europe-west1',
        '--platform',
        'managed',
      ]
    options:
      logging: CLOUD_LOGGING_ONLY
    

🧠 Knowledge check

  1. In the previous module on using the cloud you learned about the Secrets Manager in GCP. How can you use this service in combination with Cloud Run?

    Solution

    In the cloud console, secrets can be set in the Container(s), Volumes, Networking, Security tab under the Variables & Secrets section, see image below.

    Image

    In the gcloud command, you can set the secret by using the --update-secrets flag.

    gcloud run deploy <service-name> \
        --image <image-name>:<image-tag> --platform managed \
        --region <region> --allow-unauthenticated \
        --update-secrets <secret-name>=<secret-version>
    

That ends the exercises on deployment. The exercises above are just a small taste of what deployment has to offer. In both sections, we have explicitly chosen to work with serverless deployments. But what if you wanted to do the opposite e.g. being the one in charge of the management of the cluster that handles the deployed services? If you are interested in taking deployment to the next level should get started on Kubernetes which is the de-facto open-source container orchestration platform that is being used in production environments. If you want to deep dive we recommend starting here which describes how to make pipelines that are a necessary component before you start to create your own Kubernetes cluster.