{ "cells": [ { "cell_type": "markdown", "id": "453524a2-0d79-4512-8d97-0240ec1ea08e", "metadata": {}, "source": [ "" ] }, { "cell_type": "markdown", "id": "c57b540f-18e8-42c3-a688-c30103c5f462", "metadata": {}, "source": [ "# Fundamentals of Accelerated Data Science # " ] }, { "cell_type": "markdown", "id": "61aa05f5-febf-487c-bd59-d489efd98709", "metadata": {}, "source": [ "## 07 - Triton ##\n", "\n", "**Table of Contents**\n", "
\n", "This notebook covers the process of deploying a model with Triton Inference Server and walks through the various tools available within Triton. This notebook covers the below sections: \n", "1. [Background](#s1)\n", "2. [Preparing the Model](#s1)\n", " * [Load the Model](#s1)\n", " * [Creating the Folder Structure](#s1)\n", " * [Creating the Configuration File](#s1)\n", "3. [Loading the Model in Triton](#s1)\n", " * [Starting Triton](#s1)\n", " * [Check Status of Triton Server](#s1)\n", "4. [Testing Inference](#s1)\n", " * [Triton Client](#s1)\n", " * [Verifying Results for Local Model and Triton](#s1)\n", "5. [Analyzing Performance](#s1)\n", " * [Customizing Perf Analyzer](#s1)\n", " * [Exercise #1 - Testing Perf Analyzer](#s1)\n", "6. [Model Analyzer](#s1)" ] }, { "cell_type": "markdown", "id": "d4d08487-8161-4d8c-b402-cdeab3fcbe2a", "metadata": {}, "source": [ "# Background\n", "NVIDIA offers a framework for deploying ML models called Triton. Triton will automatically handle all inference requests that comes to the server. Triton supports multiple backends including PyTorch, TensorFlow, Forest Inference Library (FIL), etc. In this notebook, we will focus on the FIL backend using the xgboost model that was trained previously. " ] }, { "cell_type": "markdown", "id": "aa0a7d94-c0a9-4b86-92bb-fafffa1ed021", "metadata": {}, "source": [ "## Preparing the Model\n", "### Load the Model\n", "Let's start with loading the previous XGBoost model." ] }, { "cell_type": "code", "execution_count": 1, "id": "b6b2149c-3f83-46cd-a1eb-70db7067cf5b", "metadata": {}, "outputs": [], "source": [ "import xgboost as xgb\n", "model = xgb.Booster({'nthread': 4}) # init model\n", "model.load_model('xgboost_model.json') # load model data" ] }, { "cell_type": "markdown", "id": "a7896106-8410-40c3-8b6d-e0bdb94f69c3", "metadata": {}, "source": [ "### Creating the Folder Structure\n", "In the previous notebook, we just saved the XGBoost model to the working directory. Triton expects the model to be in a particular structure: Model name should be the top level directory, and the version number should be next. This allows for multiple models and versions of those models to be hosted simulatenously. Let's create that folder structure and save the model!" ] }, { "cell_type": "code", "execution_count": 2, "id": "28c42b97-2691-46d1-9b3a-7d19685d1d7f", "metadata": {}, "outputs": [], "source": [ "import os\n", "\n", "# Create the model repository directory. The name of this directory is arbitrary.\n", "REPO_PATH = os.path.abspath('models')\n", "os.makedirs(REPO_PATH, exist_ok=True)\n", "\n", "# The name of the model directory determines the name of the model as reported by Triton\n", "model_dir = os.path.join(REPO_PATH, \"virus_prediction\")\n", "\n", "# We can store multiple versions of the model in the same directory. In our case, we have just one version, so we will add a single directory, named '1'.\n", "version_dir = os.path.join(model_dir, '1')\n", "os.makedirs(version_dir, exist_ok=True)\n", "\n", "# The default filename for XGBoost models saved in json format is 'xgboost.json'.\n", "# It is recommended that you use this filename to avoid having to specify a name in the configuration file.\n", "model_file = os.path.join(version_dir, 'xgboost.json')\n", "model.save_model(model_file)" ] }, { "cell_type": "markdown", "id": "900e5837-4807-4fd4-af21-0973a4942f1b", "metadata": {}, "source": [ "### Creating the Configuration File\n", "Triton also requires a configuration file that provides details about the model and deployment. For this notebook, we will assume default parameters." ] }, { "cell_type": "code", "execution_count": null, "id": "61d898fb-a8d2-4d1c-a13f-2c4be6c18969", "metadata": {}, "outputs": [], "source": [ "config_text = f\"\"\"backend: \"fil\"\n", "max_batch_size: 32768\n", "input [ \n", " {{ \n", " name: \"input__0\"\n", " data_type: TYPE_FP32\n", " dims: [ 4 ] \n", " }} \n", "]\n", "output [\n", " {{\n", " name: \"output__0\"\n", " data_type: TYPE_FP32\n", " dims: [ 1 ]\n", " }}\n", "]\n", "instance_group [{{ kind: KIND_GPU }}]\n", "parameters [\n", " {{\n", " key: \"model_type\"\n", " value: {{ string_value: \"xgboost_json\" }}\n", " }},\n", " {{\n", " key: \"output_class\"\n", " value: {{ string_value: \"false\" }}\n", " }},\n", " {{\n", " key: \"storage_type\"\n", " value: {{ string_value: \"AUTO\" }}\n", " }}т\n", "]\n", "\n", "dynamic_batching {{\n", " max_queue_delay_microseconds: 100\n", "}}\"\"\"\n", "config_path = os.path.join(model_dir, 'config.pbtxt')\n", "with open(config_path, 'w') as file_:\n", " file_.write(config_text)" ] }, { "cell_type": "markdown", "id": "f262d4d9-819a-4d28-bb26-7eb737c3387a", "metadata": {}, "source": [ "Now the model is ready to be loaded in Triton! The model repository should look like this.\n", "\n", "```\n", "model/\n", "`-- virus_prediction\n", " |-- 1\n", " | `-- xgboost.json\n", " `-- config.pbtxt\n", "```\n" ] }, { "cell_type": "markdown", "id": "b5876d7a-3416-4ac9-827a-53796a1f7f93", "metadata": {}, "source": [ "## Loading the Model in Triton\n", "Next, we will need to start the Triton server. For this course, the server has already been started, but we will briefly go the steps required." ] }, { "cell_type": "markdown", "id": "709cf499-9faa-4746-9252-16f41fa548e6", "metadata": {}, "source": [ "### Starting Triton\n", "Triton is available as buildable source code or a pre-built docker image. For simplicity, we recommend that most users start with the docker images. This is how you would start the docker container in the console." ] }, { "cell_type": "code", "execution_count": 4, "id": "1283515c-c812-41c5-a100-e4dc5788781e", "metadata": {}, "outputs": [], "source": [ "#!docker run --gpus=1 --rm -p 8000:8000 -p 8001:8001 -p 8002:8002 -v /full/path/to/docs/examples/model_repository:/models nvcr.io/nvidia/tritonserver:-py3 tritonserver --model-repository=/models" ] }, { "cell_type": "markdown", "id": "2ded6e36-0a1d-488e-8b10-f482ced8ea07", "metadata": {}, "source": [ "Wow! That's a lot of inputs. Let's break it down a little bit more.\n", "\n", "- **gpus=1** : Passes through the first GPU to the Triton Inference Server\n", "- **rm** : Removes the container after completing execution\n", "- **p 8000:8000** : Forwards the GTPCInferenceService Port\n", "- **p 8001:8001** : Forwards the HTTPService Port\n", "- **p 8002:8002** : Forwards the Metrics Port\n", "- **v /full/path/to/docs/examples/model_repository:/models** : Mounts the path to the models folder on the host machine to the Triton Inference Server container\n", "- **nvcr.io/nvidia/tritonserver:-py3** : Name of the Triton Inference Server image. The version number will change depending on the latest release\n", "- **tritonserver --model-repository=/models**: The command to run in the container. In this case, we start the Triton Inference Server and point to the models folder\n" ] }, { "cell_type": "markdown", "id": "42967a82-fdfd-413e-8006-eb78a3ca805b", "metadata": {}, "source": [ "As we mentioned before, the server has already been started for this lab. Let's check our connection to the server! We are using \"triton\" as the hostname because the default docker network will resolve \"triton\" to the ip address of the Triton Inference Server." ] }, { "cell_type": "code", "execution_count": 5, "id": "ac240cdc-f0dc-45ad-9838-58066a85696a", "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "* Trying 172.18.0.4:8000...\n", "* Connected to triton (172.18.0.4) port 8000 (#0)\n", "> GET /v2/health/ready HTTP/1.1\n", "> Host: triton:8000\n", "> User-Agent: curl/7.81.0\n", "> Accept: */*\n", "> \n", "* Mark bundle as not supporting multiuse\n", "< HTTP/1.1 200 OK\n", "< Content-Length: 0\n", "< Content-Type: text/plain\n", "< \n", "* Connection #0 to host triton left intact\n" ] } ], "source": [ "!curl -v triton:8000/v2/health/ready" ] }, { "cell_type": "markdown", "id": "d92b9b35-51c6-4eba-a9c0-9a92d19e0a0b", "metadata": {}, "source": [ "Now let's see if the model has been loaded!" ] }, { "cell_type": "code", "execution_count": 6, "id": "d80b37c5-1242-489f-8463-387e657264a1", "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "[{\"name\":\"virus_prediction\"}]" ] } ], "source": [ "!curl -X POST http://triton:8000/v2/repository/index" ] }, { "cell_type": "markdown", "id": "06902d34-7763-4693-b919-f70ee4d60dfa", "metadata": {}, "source": [ "If all went well, we should be able to see the model \"virus prediction\" show up as ready." ] }, { "cell_type": "markdown", "id": "2429b1bf-9372-42a5-86ea-be07cabf30ed", "metadata": {}, "source": [ "## Testing Inference" ] }, { "cell_type": "markdown", "id": "0c2616f6-2de8-4e84-96d1-ad92b5f4b732", "metadata": {}, "source": [ "### Triton Client\n", "To test the deployment we will use the Triton Client library. Let's see how we can create an instance of the client. " ] }, { "cell_type": "code", "execution_count": 7, "id": "ec652ce4-a2d6-4b3a-8e2f-faac51da5487", "metadata": {}, "outputs": [], "source": [ "import time\n", "import tritonclient.grpc as triton_grpc\n", "from tritonclient import utils as triton_utils\n", "HOST = \"triton\"\n", "PORT = 8001\n", "TIMEOUT = 60" ] }, { "cell_type": "code", "execution_count": 8, "id": "22af3af5-1d15-406f-91cb-2208c938d7ca", "metadata": {}, "outputs": [], "source": [ "client = triton_grpc.InferenceServerClient(url=f'{HOST}:{PORT}')" ] }, { "cell_type": "markdown", "id": "84c3a594-e377-47c9-b3e7-0da321138bf5", "metadata": {}, "source": [ "Now let's make sure the Triton server is ready by sending a sample inference request. First let's load the training data. We will only load 32768 rows, as that's the max batch size we specified." ] }, { "cell_type": "code", "execution_count": 9, "id": "b57396e8-bc06-43dd-bf4c-2e34b24f5962", "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ " infected\n", "4506149 0.0\n", "1456192 0.0\n", "2065740 0.0\n", "1072379 0.0\n", "424532 0.0\n", "... ...\n", "1204134 0.0\n", "3330908 0.0\n", "2066864 0.0\n", "1643995 0.0\n", "4661849 0.0\n", "\n", "[32768 rows x 1 columns]\n" ] } ], "source": [ "import cudf \n", "import numpy as np\n", "df = cudf.read_csv('./data/clean_uk_pop_full.csv', usecols=['age', 'sex', 'northing', 'easting', 'infected'], nrows=5000000)\n", "df = df.sample(32768)\n", "input_data = df.drop('infected', axis=1)\n", "target = df[['infected']]\n", "print(target)" ] }, { "cell_type": "markdown", "id": "d24f3792-a541-4b9f-acdc-e72c64897804", "metadata": {}, "source": [ "Now we convert it to a numpy array and force the type to be float32 (the same type we specified in the config file)" ] }, { "cell_type": "code", "execution_count": 10, "id": "1ab43811-f273-48cf-b473-ea3d4786cab4", "metadata": {}, "outputs": [], "source": [ "converted_df = input_data.to_numpy(dtype='float32')" ] }, { "cell_type": "markdown", "id": "ed49f912-5e35-4fe8-b6fa-b91f025cc07e", "metadata": {}, "source": [ "Since we limited our batch size to 32768, let's splice the array and attempt the inference." ] }, { "cell_type": "code", "execution_count": 15, "id": "77e90afc-9ea6-45ad-83bb-2ac7f154b3f8", "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "CPU times: user 2.61 ms, sys: 1.34 ms, total: 3.96 ms\n", "Wall time: 6.44 ms\n" ] } ], "source": [ "%%time\n", "batched_data = converted_df[:32768]\n", "# Prepare the input tensor\n", "input_tensor = triton_grpc.InferInput(\"input__0\", batched_data.shape, 'FP32')\n", "input_tensor.set_data_from_numpy(batched_data)\n", "\n", "# Prepare the output\n", "output = triton_grpc.InferRequestedOutput(\"output__0\")\n", "\n", "# Send inference request\n", "response = client.infer(\"virus_prediction\", [input_tensor], outputs=[output])\n", "\n", "# Get the output data\n", "output_data = response.as_numpy(\"output__0\")" ] }, { "cell_type": "markdown", "id": "923c2923-acc1-48cb-bc41-3adafbf7eca2", "metadata": {}, "source": [ "Let's make sure the results we got were the same as with the local model." ] }, { "cell_type": "code", "execution_count": 16, "id": "f809c2db-11f9-48de-9913-3e032e14a553", "metadata": {}, "outputs": [], "source": [ "from sklearn.metrics import mean_squared_error, r2_score\n", "from sklearn.metrics import roc_curve\n", "from sklearn.metrics import auc\n", "\n", "xgb_data = xgb.DMatrix(input_data)\n", "y_test = model.predict(xgb_data)" ] }, { "cell_type": "code", "execution_count": 17, "id": "52ada8b8-0f47-435b-90df-57c5056bfc26", "metadata": {}, "outputs": [ { "data": { "image/png": "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", "text/plain": [ "
" ] }, "metadata": {}, "output_type": "display_data" } ], "source": [ "# Check that we got the same accuracy as previously\n", "#target = target.to_numpy()\n", "import matplotlib.pyplot as plt\n", "\n", "false_pos_rate, true_pos_rate, thresholds = roc_curve(target.to_numpy(), y_test)\n", "auc_result = auc(false_pos_rate, true_pos_rate)\n", "\n", "fig, ax = plt.subplots(figsize=(5, 5))\n", "ax.plot(false_pos_rate, true_pos_rate, lw=3,\n", " label='AUC = {:.2f}'.format(auc_result))\n", "ax.plot([0, 1], [0, 1], 'k--', lw=2)\n", "ax.set(\n", " xlim=(0, 1),\n", " ylim=(0, 1),\n", " title=\"ROC Curve\",\n", " xlabel=\"False Positive Rate\",\n", " ylabel=\"True Positive Rate\",\n", ")\n", "ax.legend(loc='lower right');\n", "plt.show()\n" ] }, { "cell_type": "code", "execution_count": 18, "id": "30562baa-f7c9-40d8-ab5d-40b3c642b666", "metadata": {}, "outputs": [ { "data": { "image/png": "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", "text/plain": [ "
" ] }, "metadata": {}, "output_type": "display_data" } ], "source": [ "# Check that we got the same accuracy as previously\n", "#target = target.to_numpy()\n", "import matplotlib.pyplot as plt\n", "\n", "false_pos_rate, true_pos_rate, thresholds = roc_curve(target.to_numpy(), output_data)\n", "auc_result = auc(false_pos_rate, true_pos_rate)\n", "\n", "fig, ax = plt.subplots(figsize=(5, 5))\n", "ax.plot(false_pos_rate, true_pos_rate, lw=3,\n", " label='AUC = {:.2f}'.format(auc_result))\n", "ax.plot([0, 1], [0, 1], 'k--', lw=2)\n", "ax.set(\n", " xlim=(0, 1),\n", " ylim=(0, 1),\n", " title=\"ROC Curve\",\n", " xlabel=\"False Positive Rate\",\n", " ylabel=\"True Positive Rate\",\n", ")\n", "ax.legend(loc='lower right');\n", "plt.show()" ] }, { "cell_type": "markdown", "id": "b00984cb-6e04-4951-a864-a67cc4a22b6f", "metadata": {}, "source": [ "As we can see, our AUC score is the same between both inference options!" ] }, { "cell_type": "markdown", "id": "a459c429-b902-4056-ad2e-05d53f82537c", "metadata": {}, "source": [ "## Analyzing Performance" ] }, { "cell_type": "markdown", "id": "e724bd09-057b-408a-896a-b48a7a4362d1", "metadata": {}, "source": [ "Earlier, we tested a *relatively* small inference request. What if we want to see the max throughput of the model? Luckily, Triton offers a performance analysis tool that generates synthetic data to collect latency and throughput numbers. Let's try it out." ] }, { "cell_type": "code", "execution_count": 19, "id": "15129aef-3d3f-4c65-b2ff-1d0b2599ffd7", "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "*** Measurement Settings ***\n", " Batch size: 1\n", " Service Kind: TRITON\n", " Using \"time_windows\" mode for stabilization\n", " Stabilizing using average latency and throughput\n", " Measurement window: 5000 msec\n", " Using synchronous calls for inference\n", "\n", "Request concurrency: 1\n", " Client: \n", " Request count: 39308\n", " Throughput: 2173.85 infer/sec\n", " Avg latency: 456 usec (standard deviation 107 usec)\n", " p50 latency: 428 usec\n", " p90 latency: 608 usec\n", " p95 latency: 664 usec\n", " p99 latency: 689 usec\n", " Avg HTTP time: 452 usec (send/recv 41 usec + response wait 411 usec)\n", " Server: \n", " Inference count: 39309\n", " Execution count: 39309\n", " Successful request count: 39309\n", " Avg request latency: 319 usec (overhead 2 usec + queue 219 usec + compute input 27 usec + compute infer 48 usec + compute output 22 usec)\n", "\n", "Inferences/Second vs. Client Average Batch Latency\n", "Concurrency: 1, throughput: 2173.85 infer/sec, latency 456 usec\n" ] } ], "source": [ "!perf_analyzer -m virus_prediction -u \"triton:8000\"" ] }, { "cell_type": "markdown", "id": "12fe9a4c-e5db-4474-bc9f-76e0f3485506", "metadata": {}, "source": [ "That's a lot of information to take in. Let's break it down.\n", "\n", "**Measurement window**: Timeframe that measurements are taken in \n", "\n", "**Batch Size**: Number of inputs in each request\n", "\n", "\n", "**Concurrency**: Number of simulatenous connections\n", "**Latency**: Time taken to recieve results\n", "**p50/90/95/99**: Different percentiles for latency\n", "\n", "Based on these results, we can see that our throughput is roughly ~2300 inferences per second with a single concurrent connection, and the average latency for each requst is 434 usec." ] }, { "cell_type": "markdown", "id": "27e8c5c0-7f3e-41e3-881d-38f9d0aa5d3e", "metadata": {}, "source": [ "# Customizing Perf Analyzer\n", "The Performance Analyzer tool for Triton has many knobs that can be turned to analyze results. Let's enable GPU metric collection, and increase the batch size and concurrency range values!" ] }, { "cell_type": "code", "execution_count": 20, "id": "6353c5bb-ebd5-492e-ad62-90e8929849d4", "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "*** Measurement Settings ***\n", " Batch size: 8\n", " Service Kind: TRITON\n", " Using \"time_windows\" mode for stabilization\n", " Stabilizing using average latency and throughput\n", " Measurement window: 5000 msec\n", " Latency limit: 0 msec\n", " Concurrency limit: 8 concurrent requests\n", " Using synchronous calls for inference\n", "\n", "Request concurrency: 2\n", " Client: \n", " Request count: 82618\n", " Throughput: 36421.1 infer/sec\n", " Avg latency: 436 usec (standard deviation 94 usec)\n", " p50 latency: 429 usec\n", " p90 latency: 582 usec\n", " p95 latency: 628 usec\n", " p99 latency: 707 usec\n", " Avg HTTP time: 432 usec (send/recv 37 usec + response wait 395 usec)\n", " Server: \n", " Inference count: 660960\n", " Execution count: 41392\n", " Successful request count: 82620\n", " Avg request latency: 294 usec (overhead 2 usec + queue 197 usec + compute input 25 usec + compute infer 44 usec + compute output 25 usec)\n", "\n", " Server Prometheus Metrics: \n", " Avg GPU Utilization:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 0.842105%\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 0.842105%\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 0.842105%\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 0.842105%\n", " Avg GPU Power Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 25.5052 watts\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 25.6625 watts\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 25.5276 watts\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 25.1561 watts\n", " Max GPU Memory Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 832569344 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 230686720 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 230686720 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 230686720 bytes\n", " Total GPU Memory:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 16106127360 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 16106127360 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 16106127360 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 16106127360 bytes\n", "Request concurrency: 4\n", " Client: \n", " Request count: 170691\n", " Throughput: 74290.5 infer/sec\n", " Avg latency: 427 usec (standard deviation 100 usec)\n", " p50 latency: 433 usec\n", " p90 latency: 572 usec\n", " p95 latency: 648 usec\n", " p99 latency: 716 usec\n", " Avg HTTP time: 424 usec (send/recv 34 usec + response wait 390 usec)\n", " Server: \n", " Inference count: 1365536\n", " Execution count: 48705\n", " Successful request count: 170691\n", " Avg request latency: 272 usec (overhead 2 usec + queue 167 usec + compute input 24 usec + compute infer 45 usec + compute output 33 usec)\n", "\n", " Server Prometheus Metrics: \n", " Avg GPU Utilization:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 1.10526%\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 1.10526%\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 1.10526%\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 1.10526%\n", " Avg GPU Power Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 25.517 watts\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 25.6462 watts\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 25.522 watts\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 25.1323 watts\n", " Max GPU Memory Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 832569344 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 230686720 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 230686720 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 230686720 bytes\n", " Total GPU Memory:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 16106127360 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 16106127360 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 16106127360 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 16106127360 bytes\n", "Request concurrency: 6\n", " Client: \n", " Request count: 329197\n", " Throughput: 137543 infer/sec\n", " Avg latency: 346 usec (standard deviation 82 usec)\n", " p50 latency: 339 usec\n", " p90 latency: 438 usec\n", " p95 latency: 458 usec\n", " p99 latency: 621 usec\n", " Avg HTTP time: 343 usec (send/recv 31 usec + response wait 312 usec)\n", " Server: \n", " Inference count: 2633624\n", " Execution count: 86121\n", " Successful request count: 329203\n", " Avg request latency: 212 usec (overhead 1 usec + queue 129 usec + compute input 16 usec + compute infer 36 usec + compute output 29 usec)\n", "\n", " Server Prometheus Metrics: \n", " Avg GPU Utilization:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 2.05%\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 2.05%\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 2.05%\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 2.05%\n", " Avg GPU Power Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 25.5291 watts\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 25.6947 watts\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 25.5563 watts\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 25.1661 watts\n", " Max GPU Memory Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 832569344 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 230686720 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 230686720 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 230686720 bytes\n", " Total GPU Memory:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 16106127360 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 16106127360 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 16106127360 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 16106127360 bytes\n", "Request concurrency: 8\n", " Client: \n", " Request count: 508448\n", " Throughput: 184400 infer/sec\n", " Avg latency: 344 usec (standard deviation 82 usec)\n", " p50 latency: 348 usec\n", " p90 latency: 386 usec\n", " p95 latency: 400 usec\n", " p99 latency: 445 usec\n", " Avg HTTP time: 341 usec (send/recv 31 usec + response wait 310 usec)\n", " Server: \n", " Inference count: 4067600\n", " Execution count: 122220\n", " Successful request count: 508450\n", " Avg request latency: 192 usec (overhead 1 usec + queue 116 usec + compute input 14 usec + compute infer 35 usec + compute output 26 usec)\n", "\n", " Server Prometheus Metrics: \n", " Avg GPU Utilization:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 2.83333%\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 2.83333%\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 2.83333%\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 2.83333%\n", " Avg GPU Power Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 25.5151 watts\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 25.7038 watts\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 25.6153 watts\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 25.1973 watts\n", " Max GPU Memory Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 832569344 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 230686720 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 230686720 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 230686720 bytes\n", " Total GPU Memory:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 16106127360 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 16106127360 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 16106127360 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 16106127360 bytes\n", "Inferences/Second vs. Client Average Batch Latency\n", "Concurrency: 2, throughput: 36421.1 infer/sec, latency 436 usec\n", "Concurrency: 4, throughput: 74290.5 infer/sec, latency 427 usec\n", "Concurrency: 6, throughput: 137543 infer/sec, latency 346 usec\n", "Concurrency: 8, throughput: 184400 infer/sec, latency 344 usec\n" ] } ], "source": [ "!perf_analyzer --collect-metrics -m virus_prediction -u \"triton:8000\" -b 8 --concurrency-range 2:8:2" ] }, { "cell_type": "markdown", "id": "6d0340a9-da5a-400f-9127-b3f4fe1facad", "metadata": {}, "source": [ "The results show that our model configuration gives a throughput of about ~156171 inferences per second. Notice how there are significant throughput gains as we increase the number of concurrent connections. With low concurrency values, Triton is idle during the time when the response is returned to the client and the next request is received at the server. Throughput increases when we increase concurrency values because Triton overlaps the processing of one request with the communication of the other. " ] }, { "cell_type": "markdown", "id": "ea13ffda-b03d-435e-a01e-5f8ceaa076de", "metadata": {}, "source": [ "# Exercise \n", "Please take this time to experiment with the perf analyzer tool. A full list of parameters can be seen with the --help argument. Some parameters that we recommend trying out are listed below: \n", "* -b : batch size\n", "* --concurrency-range : range of concurrency values to test\n", "* --collect-metrics : enable collection of GPU metrics" ] }, { "cell_type": "code", "execution_count": null, "id": "e8bb02a7-92c1-4373-afb6-c4a507ddf20d", "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "*** Measurement Settings ***\n", " Batch size: 8\n", " Service Kind: TRITON\n", " Using \"time_windows\" mode for stabilization\n", " Stabilizing using average latency and throughput\n", " Measurement window: 5000 msec\n", " Latency limit: 0 msec\n", " Concurrency limit: 16 concurrent requests\n", " Using synchronous calls for inference\n", "\n", "Request concurrency: 1\n", " Client: \n", " Request count: 40512\n", " Throughput: 17945.4 infer/sec\n", " Avg latency: 442 usec (standard deviation 103 usec)\n", " p50 latency: 425 usec\n", " p90 latency: 596 usec\n", " p95 latency: 648 usec\n", " p99 latency: 705 usec\n", " Avg HTTP time: 438 usec (send/recv 40 usec + response wait 398 usec)\n", " Server: \n", " Inference count: 324112\n", " Execution count: 40514\n", " Successful request count: 40514\n", " Avg request latency: 309 usec (overhead 0 usec + queue 215 usec + compute input 26 usec + compute infer 46 usec + compute output 20 usec)\n", "\n", " Server Prometheus Metrics: \n", " Avg GPU Utilization:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 0.947368%\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 0.842105%\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 0.842105%\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 0.947368%\n", " Avg GPU Power Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 25.4854 watts\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 25.6408 watts\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 25.5326 watts\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 25.1241 watts\n", " Max GPU Memory Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 832569344 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 230686720 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 230686720 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 230686720 bytes\n", " Total GPU Memory:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 16106127360 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 16106127360 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 16106127360 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 16106127360 bytes\n", "Request concurrency: 3\n", " Client: \n", " Request count: 123389\n", " Throughput: 54124.1 infer/sec\n", " Avg latency: 440 usec (standard deviation 103 usec)\n", " p50 latency: 429 usec\n", " p90 latency: 606 usec\n", " p95 latency: 650 usec\n", " p99 latency: 731 usec\n", " Avg HTTP time: 436 usec (send/recv 38 usec + response wait 398 usec)\n", " Server: \n", " Inference count: 987144\n", " Execution count: 41396\n", " Successful request count: 123393\n", " Avg request latency: 286 usec (overhead 1 usec + queue 178 usec + compute input 27 usec + compute infer 47 usec + compute output 32 usec)\n", "\n", " Server Prometheus Metrics: \n", " Avg GPU Utilization:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 1%\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 1%\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 1%\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 1%\n", " Avg GPU Power Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 25.5275 watts\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 25.6826 watts\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 25.5457 watts\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 25.1294 watts\n", " Max GPU Memory Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 832569344 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 230686720 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 230686720 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 230686720 bytes\n", " Total GPU Memory:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 16106127360 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 16106127360 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 16106127360 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 16106127360 bytes\n", "Request concurrency: 5\n", " Client: \n", " Request count: 248663\n", " Throughput: 106417 infer/sec\n", " Avg latency: 373 usec (standard deviation 88 usec)\n", " p50 latency: 356 usec\n", " p90 latency: 454 usec\n", " p95 latency: 520 usec\n", " p99 latency: 698 usec\n", " Avg HTTP time: 369 usec (send/recv 32 usec + response wait 337 usec)\n", " Server: \n", " Inference count: 1989328\n", " Execution count: 77638\n", " Successful request count: 248666\n", " Avg request latency: 221 usec (overhead 1 usec + queue 136 usec + compute input 18 usec + compute infer 38 usec + compute output 27 usec)\n", "\n", " Server Prometheus Metrics: \n", " Avg GPU Utilization:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 1.84211%\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 1.73684%\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 1.84211%\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 1.84211%\n", " Avg GPU Power Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 25.5205 watts\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 25.7027 watts\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 25.5786 watts\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 25.1956 watts\n", " Max GPU Memory Usage:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 832569344 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 230686720 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 230686720 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 230686720 bytes\n", " Total GPU Memory:\n", " GPU-3b4a3ef9-2229-9b49-0526-7070d90a8f2d : 16106127360 bytes\n", " GPU-99aab0d2-4243-f5eb-c3c7-8fe633a47ee2 : 16106127360 bytes\n", " GPU-cc28cec7-3383-f02d-7890-3b72a424a18d : 16106127360 bytes\n", " GPU-f3f9894c-c591-94f2-4d37-8292251c337e : 16106127360 bytes\n", "Request concurrency: 7\n" ] } ], "source": [ "!perf_analyzer -u \"triton:8000\" -m virus_prediction -b 8 --concurrency-range 1:16:2 --collect-metrics" ] }, { "cell_type": "markdown", "id": "e6ccd1e7-989b-49b5-8ece-0583e9890c66", "metadata": {}, "source": [ "## Model Analyzer\n", "Although it is out of scope for this course, we would like to introduce the Model Analyzer tool that is available as part of Triton. This tool searches through different parameter configurations to find optimal parameters that maximize inference throughput. With some minor processing, results can be viewed in a PDF format as well. An example of the output is shown below. More information about the Model Analyzer can be found [here](https://docs.nvidia.com/deeplearning/triton-inference-server/user-guide/docs/user_guide/model_analyzer.html). " ] }, { "cell_type": "markdown", "id": "a0c0c90f-d8e6-4ce5-a9c9-197a46e1d8a2", "metadata": {}, "source": [ "![image](images/model_analyzer.png)" ] }, { "cell_type": "code", "execution_count": null, "id": "91558f00-41d2-493d-8517-9468efcae64e", "metadata": {}, "outputs": [], "source": [ "import IPython\n", "app = IPython.Application.instance()\n", "app.kernel.do_shutdown(True)" ] }, { "cell_type": "markdown", "id": "c7e473f9-9ccf-43c6-aa86-eda25150a5ca", "metadata": {}, "source": [ "**Well Done!** Let's move to the [next notebook](3-08_k-means_dask.ipynb). " ] }, { "cell_type": "markdown", "id": "832c59bf-517c-4703-9161-1f9997214e19", "metadata": {}, "source": [ "" ] } ], "metadata": { "kernelspec": { "display_name": "Python 3 (ipykernel)", "language": "python", "name": "python3" }, "language_info": { "codemirror_mode": { "name": "ipython", "version": 3 }, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.10.15" } }, "nbformat": 4, "nbformat_minor": 5 }