Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Python-based backends CI #6466

Merged
merged 21 commits into from
Nov 3, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
e387124
Bumped vllm version
oandreeva-nv Oct 17, 2023
58775bc
Add python-bsed backends testing
pskiran1 Oct 19, 2023
30592aa
Add python-based backends CI
pskiran1 Oct 19, 2023
62b9fb2
Merge branch 'main' of https://github.com/triton-inference-server/ser…
pskiran1 Oct 19, 2023
6bd1c82
Fix errors
pskiran1 Oct 19, 2023
74a872e
Merge branch 'main' of https://github.com/triton-inference-server/ser…
pskiran1 Oct 20, 2023
8c4d4cf
Add vllm backend
pskiran1 Oct 21, 2023
8fde6a8
Fix pre-commit
pskiran1 Oct 23, 2023
40d6a26
Modify test.sh
pskiran1 Oct 23, 2023
6edb28f
Remove vllm_opt qa model
pskiran1 Oct 25, 2023
e048cf7
Remove vLLM ackend tests
pskiran1 Oct 26, 2023
03fbaef
Merge branch 'main' of https://github.com/triton-inference-server/ser…
pskiran1 Oct 27, 2023
90fc93f
Merge branch 'main' of https://github.com/triton-inference-server/ser…
pskiran1 Nov 2, 2023
f5c0041
Resolve review comments
pskiran1 Nov 2, 2023
8a889f0
Merge branch 'spolisetty_vllm_ci' of https://github.com/triton-infere…
pskiran1 Nov 2, 2023
2938b2d
Fix pre-commit errors
pskiran1 Nov 2, 2023
eee97dd
Merge branch 'spolisetty_vllm_ci' of https://github.com/triton-infere…
pskiran1 Nov 2, 2023
e419916
Update qa/L0_backend_python/python_based_backends/python_based_backen…
pskiran1 Nov 2, 2023
7be3387
Merge branch 'spolisetty_vllm_ci' of https://github.com/triton-infere…
pskiran1 Nov 3, 2023
df968c7
Merge branch 'main' of https://github.com/triton-inference-server/ser…
pskiran1 Nov 3, 2023
7c98ef6
Remove collect_artifacts_from_subdir function call
pskiran1 Nov 3, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,4 @@
__pycache__
tmp
*.log
test_results.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
# Copyright 2023, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# * Neither the name of NVIDIA CORPORATION nor the names of its
# contributors may be used to endorse or promote products derived
# from this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
# OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

import sys
import unittest
from random import randint

import numpy as np
import tritonclient.grpc as grpcclient
from tritonclient.utils import *

sys.path.append("../../common")
from test_util import TestResultCollector


class PythonBasedBackendsTest(TestResultCollector):
def setUp(self):
self.triton_client = grpcclient.InferenceServerClient(url="localhost:8001")
self.add_sub_model_1 = "add"
self.add_sub_model_2 = "sub"
self.python_model = "add_sub"
self.pytorch_model = "add_sub_pytorch"

self.triton_client.load_model(
self.add_sub_model_1,
config='{"backend":"add_sub","version_policy":{"latest":{"num_versions":2}}}',
)
self.triton_client.load_model(self.add_sub_model_2)
self.triton_client.load_model(self.python_model)
self.triton_client.load_model(self.pytorch_model)

def test_add_sub_models(self):
self.assertTrue(
self.triton_client.is_model_ready(self.add_sub_model_1, model_version="2")
)
self._test_add_sub_model(
model_name=self.add_sub_model_1, model_version="2", single_output=True
)

self.assertTrue(
self.triton_client.is_model_ready(self.add_sub_model_1, model_version="1")
)
self._test_add_sub_model(
model_name=self.add_sub_model_1, model_version="1", single_output=True
)

self.assertTrue(self.triton_client.is_model_ready(self.add_sub_model_2))
self._test_add_sub_model(model_name=self.add_sub_model_2, single_output=True)

def test_python_model(self):
self.assertTrue(
self.triton_client.is_model_ready(self.python_model, model_version="2")
)
self._test_add_sub_model(
model_name=self.python_model, shape=[16], model_version="2"
)

def test_pytorch_model(self):
self.assertTrue(
self.triton_client.is_model_ready(self.pytorch_model, model_version="1")
)
self._test_add_sub_model(model_name=self.pytorch_model)

def _test_add_sub_model(
self, model_name, model_version="1", shape=[4], single_output=False
):
input0_data = np.random.rand(*shape).astype(np.float32)
input1_data = np.random.rand(*shape).astype(np.float32)

inputs = [
grpcclient.InferInput(
"INPUT0", input0_data.shape, np_to_triton_dtype(input0_data.dtype)
),
grpcclient.InferInput(
"INPUT1", input1_data.shape, np_to_triton_dtype(input1_data.dtype)
),
]

inputs[0].set_data_from_numpy(input0_data)
inputs[1].set_data_from_numpy(input1_data)

if single_output:
outputs = [grpcclient.InferRequestedOutput("OUTPUT")]

else:
outputs = [
grpcclient.InferRequestedOutput("OUTPUT0"),
grpcclient.InferRequestedOutput("OUTPUT1"),
]

response = self.triton_client.infer(
model_name=model_name,
inputs=inputs,
model_version=model_version,
request_id=str(randint(10, 99)),
outputs=outputs,
)

if single_output:
if model_name == "add":
self.assertTrue(
np.allclose(input0_data + input1_data, response.as_numpy("OUTPUT"))
)
else:
self.assertTrue(
np.allclose(input0_data - input1_data, response.as_numpy("OUTPUT"))
)
else:
self.assertTrue(
np.allclose(input0_data + input1_data, response.as_numpy("OUTPUT0"))
)
self.assertTrue(
np.allclose(input0_data - input1_data, response.as_numpy("OUTPUT1"))
)

def tearDown(self):
self.triton_client.close()


if __name__ == "__main__":
unittest.main()
113 changes: 113 additions & 0 deletions qa/L0_backend_python/python_based_backends/test.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,113 @@
#!/bin/bash
# Copyright 2023, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# * Neither the name of NVIDIA CORPORATION nor the names of its
# contributors may be used to endorse or promote products derived
# from this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
# OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

source ../../common/util.sh

TRITON_DIR=${TRITON_DIR:="/opt/tritonserver"}
SERVER=${TRITON_DIR}/bin/tritonserver
BACKEND_DIR=${TRITON_DIR}/backends
QA_MODELS_PATH="../../python_models"
MODEL_REPOSITORY="$(pwd)/models"
SERVER_ARGS="--model-repository=${MODEL_REPOSITORY} --backend-directory=${BACKEND_DIR} --model-control-mode=explicit --log-verbose=1"
SERVER_LOG="./python_based_backends_server.log"
CLIENT_LOG="./python_based_backends_client.log"
TEST_RESULT_FILE="./test_results.txt"
CLIENT_PY="./python_based_backends_test.py"
GEN_PYTORCH_MODEL_PY="../../common/gen_qa_pytorch_model.py"
EXPECTED_NUM_TESTS=3
RET=0

rm -rf ${MODEL_REPOSITORY}
pip3 install torch

# Setup add_sub backend and models
mkdir -p ${BACKEND_DIR}/add_sub
cp ${QA_MODELS_PATH}/python_based_backends/add_sub_backend/model.py ${BACKEND_DIR}/add_sub/model.py

mkdir -p ${MODEL_REPOSITORY}/add/1/
echo '{ "operation": "add" }' > ${MODEL_REPOSITORY}/add/1/model.json
echo "backend: \"add_sub\"" > ${MODEL_REPOSITORY}/add/config.pbtxt
cp -r ${MODEL_REPOSITORY}/add/1/ ${MODEL_REPOSITORY}/add/2/

mkdir -p ${MODEL_REPOSITORY}/sub/1/
echo '{ "operation": "sub" }' > ${MODEL_REPOSITORY}/sub/1/model.json
echo "backend: \"add_sub\"" > ${MODEL_REPOSITORY}/sub/config.pbtxt

# Setup python backend model
mkdir -p ${MODEL_REPOSITORY}/add_sub/1
cp ${QA_MODELS_PATH}/add_sub/model.py ${MODEL_REPOSITORY}/add_sub/1/
cp ${QA_MODELS_PATH}/add_sub/config.pbtxt ${MODEL_REPOSITORY}/add_sub/
cp -r ${MODEL_REPOSITORY}/add_sub/1/ ${MODEL_REPOSITORY}/add_sub/2/

# Setup pytorch backend model
cp ${GEN_PYTORCH_MODEL_PY} ./gen_qa_pytorch_model.py
GEN_PYTORCH_MODEL_PY=./gen_qa_pytorch_model.py

set +e
python3 ${GEN_PYTORCH_MODEL_PY} -m ${MODEL_REPOSITORY}

if [ $? -ne 0 ]; then
echo -e "\n***\n*** Running ${GEN_PYTORCH_MODEL_PY} FAILED. \n***"
exit 1
fi
set -e

run_server
if [ "$SERVER_PID" == "0" ]; then
cat $SERVER_LOG
echo -e "\n***\n*** Failed to start $SERVER\n***"
exit 1
fi

set +e
python3 $CLIENT_PY -v >$CLIENT_LOG 2>&1

if [ $? -ne 0 ]; then
echo -e "\n***\n*** Running $CLIENT_PY FAILED. \n***"
RET=1
else
check_test_results $TEST_RESULT_FILE $EXPECTED_NUM_TESTS
if [ $? -ne 0 ]; then
echo -e "\n***\n*** Test Result Verification FAILED.\n***"
RET=1
fi
fi
set -e

kill $SERVER_PID
wait $SERVER_PID
rm -rf ${MODEL_REPOSITORY} ${GEN_PYTORCH_MODEL_PY}

if [ $RET -eq 1 ]; then
cat $CLIENT_LOG
cat $SERVER_LOG
echo -e "\n***\n*** Python-based Backends test FAILED. \n***"
else
echo -e "\n***\n*** Python-based Backends test PASSED. \n***"
fi

exit $RET
2 changes: 1 addition & 1 deletion qa/L0_backend_python/test.sh
Original file line number Diff line number Diff line change
Expand Up @@ -395,7 +395,7 @@ fi
# Disable variants test for Jetson since already built without GPU Tensor support
# Disable decoupled test because it uses GPU tensors
if [ "$TEST_JETSON" == "0" ]; then
SUBTESTS="ensemble io bls decoupled variants"
SUBTESTS="ensemble io bls decoupled variants python_based_backends"
for TEST in ${SUBTESTS}; do
# Run each subtest in a separate virtual environment to avoid conflicts
# between dependencies.
Expand Down
124 changes: 124 additions & 0 deletions qa/common/gen_qa_pytorch_model.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,124 @@
# Copyright 2023, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# * Neither the name of NVIDIA CORPORATION nor the names of its
# contributors may be used to endorse or promote products derived
# from this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
# OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.


import argparse
import os

import torch
from torch import nn


class AddSubNet(nn.Module):
def __init__(self):
super(AddSubNet, self).__init__()

def forward(self, input0, input1):
return (input0 + input1), (input0 - input1)


def generate_model(model_dir):
model = AddSubNet()

traced_model = torch.jit.trace(
model,
(torch.rand(1, 4, dtype=torch.float), torch.rand(1, 4, dtype=torch.float)),
)

os.makedirs(model_dir, exist_ok=True)
model_path = os.path.join(model_dir, "model.pt")

traced_model.save(model_path)


def generate_config(config_path):
pskiran1 marked this conversation as resolved.
Show resolved Hide resolved
with open(f"{config_path}/config.pbtxt", "w") as f:
f.write(
"""
backend: "pytorch"
input [
{
name: "INPUT0"
data_type: TYPE_FP32
dims: [ 4 ]
}
]
input [
{
name: "INPUT1"
data_type: TYPE_FP32
dims: [ 4 ]
}
]
output [
{
name: "OUTPUT0"
data_type: TYPE_FP32
dims: [ 4 ]
}
]
output [
{
name: "OUTPUT1"
data_type: TYPE_FP32
dims: [ 4 ]
}
]
"""
)


if __name__ == "__main__":
parser = argparse.ArgumentParser()
parser.add_argument(
"-m",
"--model-directory",
type=str,
required=True,
help="The path to the model repository.",
)
parser.add_argument(
"--model-name",
type=str,
required=False,
default="add_sub_pytorch",
help="Model name",
)
parser.add_argument(
"--version",
type=str,
required=False,
default="1",
help="Model version",
)

args = parser.parse_args()

model_directory = os.path.join(args.model_directory, args.model_name)
os.makedirs(model_directory, exist_ok=True)

generate_model(model_dir=os.path.join(model_directory, args.version))
generate_config(model_directory)
Loading
Loading