Table of Contents generated with DocToc
This doc explains how to setup a development environment so you can get started
contributing
to kfserving
. Also take a look at:
Follow the instructions below to set up your development environment. Once you meet these requirements, you can make changes and deploy your own version of kfserving!
Before submitting a PR, see also CONTRIBUTING.md.
You must install these tools:
go
: KFServing controller is written in Go.git
: For source control.dep
: For managing external Go dependencies. You should installdep
using theirinstall.sh
.ko
: For development.kubectl
: For managing development environments.kustomize
To customize YAMLs for different environments
KFServing currently requires Knative Serving
for auto-scaling, canary rollout, Istio
for traffic routing and ingress.
-
You can follow the instructions on Set up a kubernetes cluster and install Knative Serving or Custom Install to install
Istio
andKnative Serving
. Observability plug-ins are good to have for monitoring. -
If you already have
Istio
(e.g. from a Kubeflow install) then simply skip theIstio
steps. For Kubeflow install, you can installKnative Serving
v0.8 via the following commands after downloading repository kubeflow/manifests.kubeflow/manifests/knative/knative-serving-crds/base$ kustomize build . | kubectl apply -f -
kubeflow/manifests/knative/knative-serving-install/base$ kustomize build . | kubectl apply -f -
-
You need follow the instructions on Updating your install to use cluster local gateway to add cluster local gateway to your cluster if you are on knative serving 0.9.0+.
To start your environment you'll need to set these environment variables (we
recommend adding them to your .bashrc
):
GOPATH
: If you don't have one, simply pick a directory and addexport GOPATH=...
$GOPATH/bin
onPATH
: This is so that tooling installed viago get
will work properly.KO_DOCKER_REPO
: The docker repository to which developer images should be pushed (e.g.docker.io/<username>/[project]
).
- Note: Set up a docker repository for pushing images. You can use any container image registry by adjusting the authentication methods and repository paths mentioned in the sections below.
- Note: if you are using docker hub to store your images your
KO_DOCKER_REPO
variable should bedocker.io/<username>
. - Note: Currently Docker Hub doesn't let you create subdirs under your username.
.bashrc
example:
export GOPATH="$HOME/go"
export PATH="${PATH}:${GOPATH}/bin"
export KO_DOCKER_REPO='docker.io/<username>'
The Go tools require that you clone the repository to the
src/github.com/kubeflow/kfserving
directory in your
GOPATH
.
To check out this repository:
- Create your own fork of this repo
- Clone it to your machine:
mkdir -p ${GOPATH}/src/github.com/kubeflow
cd ${GOPATH}/src/github.com/kubeflow
git clone [email protected]:${YOUR_GITHUB_USERNAME}/kfserving.git
cd kfserving
git remote add upstream [email protected]:kubeflow/kfserving.git
git remote set-url --push upstream no_push
Adding the upstream
remote sets you up nicely for regularly
syncing your fork.
Once you reach this point you are ready to do a full build and deploy as described below.
Once you've setup your development environment, you can see things running with:
$ kubectl -n knative-serving get pods
NAME READY STATUS RESTARTS AGE
activator-c8495dc9-z7xpz 2/2 Running 0 6d
autoscaler-66897845df-t5cwg 2/2 Running 0 6d
controller-699fb46bb5-xhlkg 1/1 Running 0 6d
webhook-76b87b8459-tzj6r 1/1 Running 0 6d
make deploy
After above step you can see things running with:
$ kubectl get pods -n kfserving-system -l control-plane=kfserving-controller-manager
NAME READY STATUS RESTARTS AGE
kfserving-controller-manager-0 2/2 Running 0 13m
- Note: By default it installs to
kfserving-system
namespace with the publishedkfserving-controller-manager
image.
make deploy-dev
- Note:
deploy-dev
builds the image from your local code, publishes toKO_DOCKER_REPO
and deploys thekfserving-controller-manager
with the image digest to your cluster for testing. Please also ensure you are logged in toKO_DOCKER_REPO
from your client machine.
kubectl apply -f docs/samples/tensorflow/tensorflow.yaml
You should see model serving deployment running under default or your specified namespace.
$ kubectl get inferenceservices -n default
NAME READY URL DEFAULT TRAFFIC CANARY TRAFFIC AGE
flowers-sample True flowers-sample.default.example.com 100 1h
$ kubectl get pods -n default -l serving.kubeflow.org/inferenceservice=flowers-sample
NAME READY STATUS RESTARTS AGE
flowers-sample-default-htz8r-deployment-8fd979f9b-w2qbv 3/3 Running 0 10s
NOTE: KFServing scales pods to 0 in the absence of traffic. If you don't see any pods, try sending out a query via curl using instructions in the tensorflow sample: https://github.com/kubeflow/kfserving/tree/master/docs/samples/tensorflow
As you make changes to the code-base, there are two special cases to be aware of:
-
If you change an input to generated code, then you must run
make manifests
. Inputs include:- API type definitions in pkg/apis/serving/v1alpha2/,
- Manifests or kustomize patches stored in config.
-
If you change a package's deps (including adding external dep), then you must run
dep ensure
. Dependency changes should be a separate commit and not mixed with logic changes.
These are both idempotent, and we expect that running these at HEAD
to have no
diffs. Code generation and dependencies are automatically checked to produce no
diffs for each pull request.
In some cases, if newer dependencies are required, you need to run "dep ensure -update package-name" manually.
Once the codegen and dependency information is correct, redeploying the controller is simply:
make deploy-dev
You can also use Knative CLI (knctl
) to interact with models deployed on KFServing. It provides a simple set of commands to interact with a Knative installation. You can grab pre-built binaries from the Releases page. Once downloaded, you can run the following commands to get it working.
# compare checksum output to what's included in the release notes
$ shasum -a 265 ~/Downloads/knctl-*
# move binary to your system’s /usr/local/bin -- might require root password
$ mv ~/Downloads/knctl-* /usr/local/bin/knctl
# make the newly copied file executable -- might require root password
$ chmod +x /usr/local/bin/knctl
You can then run a smoke test by running the following command to show the details of tensorflow sample revision.
knctl revision show -r flowers-sample-default-4s74r
Revision 'flowers-sample-default-4s74r'
Name flowers-sample-default-4s74r
Tags -
Image digest index.docker.io/tensorflow/serving@sha256:df3c6fe1fbe5ccc3a916984ff313cc2d17e617f7b8782fc31e762c491325d813
Log URL http://localhost:8001/api/v1/namespaces/knative-monitoring/services/kibana-logging/proxy/app/kibana#/discover?_a=(query:(match:(kubernetes.labels.knative-dev%2FrevisionUID:(query:'1135797e-8585-11e9-adbd-b680f8334647',type:phrase))))
Annotations autoscaling.knative.dev/class: kpa.autoscaling.knative.dev
autoscaling.knative.dev/target: "1"
Age 1h
Conditions
Type Status Age Reason Message
Active False 59m NoTraffic The target is not receiving traffic.
BuildSucceeded True 1h - -
ContainerHealthy True 1h - -
Ready True 1h - -
ResourcesAvailable True 1h - -
Pods conditions
Pod Type Status Age Reason Message
Succeeded
- If you are on kubernetes 1.15+, we highly recommend adding object selector on kfserving pod mutating webhook configuration so that only pods managed by kfserving go through the kfserving pod mutator
kubectl patch mutatingwebhookconfiguration inferenceservice.serving.kubeflow.org --patch '{"webhooks":[{"name": "inferenceservice.kfserving-webhook-server.pod-mutator","objectSelector":{"matchExpressions":[{"key":"serving.kubeflow.org/inferenceservice", "operator": "Exists"}]}}]}'
- When you run make deploy, you may encounter an error like this:
error: error validating "STDIN": error validating data: ValidationError(CustomResourceDefinition.spec.validation.openAPIV3Schema.properties.status.properties.conditions.properties.conditions.items): invalid type for io.k8s.apiextensions-apiserver.pkg.apis.apiextensions.v1beta1.JSONSchemaPropsOrArray: got "map", expected ""; if you choose to ignore these errors, turn validation off with --validate=false
make: *** [deploy] Error 1
To fix it, please ensure you have a matching version of kubectl client as the master. If not, please update accordingly.
kubectl version
Client Version: version.Info{Major:"1", Minor:"13", GitVersion:"v1.13.6", GitCommit:"abdda3f9fefa29172298a2e42f5102e777a8ec25", GitTreeState:"clean", BuildDate:"2019-05-08T13:53:53Z", GoVersion:"go1.11.5", Compiler:"gc", Platform:"darwin/amd64"}
Server Version: version.Info{Major:"1", Minor:"13", GitVersion:"v1.13.6+IKS", GitCommit:"ac5f7341d5d0ce8ea8f206ba5b030dc9e9d4cc97", GitTreeState:"clean", BuildDate:"2019-05-09T13:26:51Z", GoVersion:"go1.11.5", Compiler:"gc", Platform:"linux/amd64"}
- When you run make deploy-dev, you may see an error like the one below:
2019/05/17 15:13:54 error processing import paths in "config/default/manager/manager.yaml": unsupported status code 401; body:
kustomize build config/overlays/development | kubectl apply -f -
Error: reading strategic merge patches [manager_image_patch.yaml]: evalsymlink failure on '/Users/animeshsingh/go/src/github.com/kubeflow/kfserving/config/overlays/development/manager_image_patch.yaml' : lstat /Users/animeshsingh/go/src/github.com/kubeflow/kfserving/config/overlays/development/manager_image_patch.yaml: no such file or directory
It`s a red herring. To resolve it, please ensure you have logged into dockerhub from you client machine.
- When you deploy the tensorflow sample, you may encounter an error like the one blow:
2019-09-28 01:52:23.345692: E tensorflow_serving/sources/storage_path/file_system_storage_path_source.cc:362] FileSystemStoragePathSource encountered a filesystem access error: Could not find base path /mnt/models for servable flowers-sample
Please make sure not to deploy the inferenceservice in the kfserving-system
or other namespaces where namespace has control-plane
as a label. The storage-initializer
init container does not get injected for deployments in those namespaces since they do not go through the mutating webhook.
- You may get one of the following errors after 'make deploy-dev', and while deploying the sample model
kubectl apply -f docs/samples/tensorflow/tensorflow.yaml
Error from server (InternalError): error when creating "docs/samples/tensorflow/tensorflow.yaml":
Internal error occurred: failed calling webhook "inferenceservice.kfserving-webhook-server.defaulter":
Post https://kfserving-webhook-server-service.kfserving-system.svc:443/mutate-inferenceservices?timeout=30s:
context deadline exceeded
unexpected EOF
dial tcp x.x.x.x:443: connect: connection refused
If above errors appear, first thing to check is if the KFServing controller is running
kubectl get po -n kfserving-system
NAME READY STATUS RESTARTS AGE
kfserving-controller-manager-0 2/2 Running 2 13m
If it is, more often than not, it is caused by a stale webhook, since webhooks are immutable. Please delete them, and test again
kubectl delete mutatingwebhookconfigurations inferenceservice.serving.kubeflow.org && kubectl delete validatingwebhookconfigurations inferenceservice.serving.kubeflow.org && kubectl delete po kfserving-controller-manager-0 -n kfserving-system
mutatingwebhookconfiguration.admissionregistration.k8s.io "inferenceservice.serving.kubeflow.org" deleted
validatingwebhookconfiguration.admissionregistration.k8s.io "inferenceservice.serving.kubeflow.org" deleted