2017-10-10 02:19:34 +08:00
# sample-controller
This repository implements a simple controller for watching Foo resources as
defined with a CustomResourceDefinition (CRD).
2018-06-20 17:26:29 +08:00
**Note:** go-get or vendor this package as `k8s.io/sample-controller` .
2017-10-22 17:35:37 +08:00
This particular example demonstrates how to perform basic operations such as:
* How to register a new custom resource (custom resource type) of type `Foo` using a CustomResourceDefinition.
* How to create/get/list instances of your new resource type `Foo` .
* How to setup a controller on resource handling create/update/delete events.
2017-10-10 02:19:34 +08:00
It makes use of the generators in [k8s.io/code-generator ](https://github.com/kubernetes/code-generator )
to generate a typed client, informers, listers and deep-copy functions. You can
do this yourself using the `./hack/update-codegen.sh` script.
The `update-codegen` script will automatically generate the following files &
directories:
* `pkg/apis/samplecontroller/v1alpha1/zz_generated.deepcopy.go`
2019-03-25 11:12:53 +08:00
* `pkg/generated/`
2017-10-10 02:19:34 +08:00
Changes should not be made to these files manually, and when creating your own
controller based off of this implementation you should not copy these files and
instead run the `update-codegen` script to generate your own.
2018-04-14 03:32:47 +08:00
## Details
The sample controller uses [client-go library ](https://github.com/kubernetes/client-go/tree/master/tools/cache ) extensively.
The details of interaction points of the sample controller with various mechanisms from this library are
explained [here ](docs/controller-client-go.md ).
2019-06-17 01:53:03 +08:00
## Fetch sample-controller and its dependencies
Like the rest of Kubernetes, sample-controller has used
[godep ](https://github.com/tools/godep ) and `$GOPATH` for years and is
now adopting go 1.11 modules. There are thus two alternative ways to
go about fetching this demo and its dependencies.
### Fetch with godep
When NOT using go 1.11 modules, you can use the following commands.
```sh
go get -d k8s.io/sample-controller
cd $GOPATH/src/k8s.io/sample-controller
godep restore
```
### When using go 1.11 modules
When using go 1.11 modules (`GO111MODULE=on`), issue the following
commands --- starting in whatever working directory you like.
```sh
git clone https://github.com/kubernetes/sample-controller.git
cd sample-controller
```
Note, however, that if you intend to
2021-05-10 23:19:18 +08:00
generate code then you will also need the
2019-06-17 01:53:03 +08:00
code-generator repo to exist in an old-style location. One easy way
to do this is to use the command `go mod vendor` to create and
populate the `vendor` directory.
### A Note on kubernetes/kubernetes
If you are developing Kubernetes according to
https://github.com/kubernetes/community/blob/master/contributors/guide/github-workflow.md
then you already have a copy of this demo in
`kubernetes/staging/src/k8s.io/sample-controller` and its dependencies
--- including the code generator --- are in usable locations
(valid for all Go versions).
2018-04-14 03:32:47 +08:00
2017-10-22 17:35:37 +08:00
## Purpose
2017-10-10 02:19:34 +08:00
This is an example of how to build a kube-like controller with a single type.
2017-10-22 17:35:37 +08:00
## Running
2018-01-31 03:26:18 +08:00
**Prerequisite**: Since the sample-controller uses `apps/v1` deployments, the Kubernetes cluster version should be greater than 1.9.
2017-10-22 17:35:37 +08:00
```sh
# assumes you have a working kubeconfig, not required if operating in-cluster
2019-04-09 20:52:57 +08:00
go build -o sample-controller .
./sample-controller -kubeconfig=$HOME/.kube/config
2017-10-22 17:35:37 +08:00
# create a CustomResourceDefinition
2019-04-09 20:52:57 +08:00
kubectl create -f artifacts/examples/crd.yaml
2017-10-22 17:35:37 +08:00
# create a custom resource of type Foo
2019-04-09 20:52:57 +08:00
kubectl create -f artifacts/examples/example-foo.yaml
2017-10-22 17:35:37 +08:00
# check deployments created through the custom resource
2019-04-09 20:52:57 +08:00
kubectl get deployments
2017-10-22 17:35:37 +08:00
```
## Use Cases
CustomResourceDefinitions can be used to implement custom resource types for your Kubernetes cluster.
These act like most other Resources in Kubernetes, and may be `kubectl apply` 'd, etc.
Some example use cases:
* Provisioning/Management of external datastores/databases (eg. CloudSQL/RDS instances)
* Higher level abstractions around Kubernetes primitives (eg. a single Resource to define an etcd cluster, backed by a Service and a ReplicationController)
## Defining types
Each instance of your custom resource has an attached Spec, which should be defined via a `struct{}` to provide data format validation.
In practice, this Spec is arbitrary key-value data that specifies the configuration/behavior of your Resource.
For example, if you were implementing a custom resource for a Database, you might provide a DatabaseSpec like the following:
``` go
type DatabaseSpec struct {
Databases []string `json:"databases"`
Users []User `json:"users"`
Version string `json:"version"`
}
type User struct {
Name string `json:"name"`
Password string `json:"password"`
}
```
2018-01-08 17:14:48 +08:00
## Validation
2021-05-10 23:19:18 +08:00
To validate custom resources, use the [`CustomResourceValidation` ](https://kubernetes.io/docs/tasks/access-kubernetes-api/extend-api-custom-resource-definitions/#validation ) feature. Validation in the form of a [structured schema ](https://kubernetes.io/docs/tasks/extend-kubernetes/custom-resources/custom-resource-definitions/#specifying-a-structural-schema ) is mandatory to be provided for `apiextensions.k8s.io/v1` .
2018-02-18 21:15:38 +08:00
### Example
2021-05-10 23:19:18 +08:00
The schema in [`crd.yaml` ](./artifacts/examples/crd.yaml ) applies the following validation on the custom resource:
2018-02-18 21:15:38 +08:00
`spec.replicas` must be an integer and must have a minimum value of 1 and a maximum value of 10.
## Subresources
2021-05-10 23:19:18 +08:00
Custom Resources support `/status` and `/scale` [subresources ](https://kubernetes.io/docs/tasks/access-kubernetes-api/custom-resources/custom-resource-definitions/#subresources ). The `CustomResourceSubresources` feature is in GA from v1.16.
2018-01-08 17:14:48 +08:00
### Example
2021-05-10 23:19:18 +08:00
The CRD in [`crd-status-subresource.yaml` ](./artifacts/examples/crd-status-subresource.yaml ) enables the `/status` subresource for custom resources.
This means that [`UpdateStatus` ](./controller.go ) can be used by the controller to update only the status part of the custom resource.
2018-02-18 21:15:38 +08:00
2019-08-30 14:30:59 +08:00
To understand why only the status part of the custom resource should be updated, please refer to the [Kubernetes API conventions ](https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#spec-and-status ).
2018-02-18 21:15:38 +08:00
In the above steps, use `crd-status-subresource.yaml` to create the CRD:
```sh
# create a CustomResourceDefinition supporting the status subresource
2019-04-09 20:52:57 +08:00
kubectl create -f artifacts/examples/crd-status-subresource.yaml
2018-02-18 21:15:38 +08:00
```
2018-01-08 17:14:48 +08:00
2021-05-10 23:19:18 +08:00
## A Note on the API version
The [group ](https://kubernetes.io/docs/reference/using-api/#api-groups ) version of the custom resource in `crd.yaml` is `v1alpha` , this can be evolved to a stable API version, `v1` , using [CRD Versioning ](https://kubernetes.io/docs/tasks/extend-kubernetes/custom-resources/custom-resource-definition-versioning/ ).
2017-10-22 17:35:37 +08:00
## Cleanup
You can clean up the created CustomResourceDefinition with:
2021-05-10 23:19:18 +08:00
```sh
kubectl delete crd foos.samplecontroller.k8s.io
```
2017-10-22 17:35:37 +08:00
## Compatibility
2017-10-10 02:19:34 +08:00
HEAD of this repository will match HEAD of k8s.io/apimachinery and
k8s.io/client-go.
2017-10-22 17:35:37 +08:00
## Where does it come from?
2017-10-10 02:19:34 +08:00
`sample-controller` is synced from
https://github.com/kubernetes/kubernetes/blob/master/staging/src/k8s.io/sample-controller.
Code changes are made in that location, merged into k8s.io/kubernetes and
later synced here.