--- title: "Developer guide" layout: default sort: 5 --- # Developer guide {: .no_toc} ## Table of contents {: .no_toc .text-delta} 1. TOC {:toc} --- ## Building from source ### Download the source code ```bash git clone https://github.com/kubernetes-sigs/node-feature-discovery cd node-feature-discovery ``` ### Docker build #### Build the container image See [customizing the build](#customizing-the-build) below for altering the container image registry, for example. ```bash make ``` #### Push the container image Optional, this example with Docker. ```bash docker push ``` ### Docker multi-arch builds with buildx The default set of architectures enabled for mulit-arch builds are `linux/amd64` and `linux/arm64`. If more architectures are needed one can override the `IMAGE_ALL_PLATFORMS` variable with a comma separated list of `OS/ARCH` tuples. #### Build the manifest-list with a container image per arch ```bash make image-all ``` Currently `docker` does not support loading of manifest-lists meaning the images are not shown when executing `docker images`, see: [buildx issue #59](https://github.com/docker/buildx/issues/59). #### Push the manifest-list with container image per arch ```bash make push-all ``` The resulting container image can be used in the same way on each arch by pulling e.g. `node-feature-discovery:{{ site.release }}` without specifying the architecture. The manifest-list will take care of providing the right architecture image. #### Change the job spec to use your custom image (optional) To use your published image from the step above instead of the `registry.k8s.io/nfd/node-feature-discovery` image, edit `image` attribute in the spec template(s) to the new location (`/[:]`). ### Deployment The `yamls` makefile generates a `kustomization.yaml` matching your locally built image and using the `deploy/overlays/default` deployment. See [build customization](#customizing-the-build) below for configurability, e.g. changing the deployment namespace. ```bash K8S_NAMESPACE=my-ns make yamls kubectl apply -k . ``` You can use alternative deployment methods by modifying the auto-generated kustomization file. ### Building locally You can also build the binaries locally ```bash make build ``` This will compile binaries under `bin/` ### Customizing the build There are several Makefile variables that control the build process and the name of the resulting container image. The following are targeted targeted for build customization and they can be specified via environment variables or makefile overrides. | Variable | Description | Default value | | -------------------------- | ----------------------------------------------------------------- | ------------- | | HOSTMOUNT_PREFIX | Prefix of system directories for feature discovery (local builds) | / (*local builds*) /host- (*container builds*) | | IMAGE_BUILD_CMD | Command to build the image | docker build | | IMAGE_BUILD_EXTRA_OPTS | Extra options to pass to build command | *empty* | | IMAGE_BUILDX_CMD | Command to build and push multi-arch images with buildx | DOCKER_CLI_EXPERIMENTAL=enabled docker buildx build --platform=${IMAGE_ALL_PLATFORMS} --progress=auto --pull | | IMAGE_ALL_PLATFORMS | Comma separated list of OS/ARCH tuples for mulit-arch builds | linux/amd64,linux/arm64 | | IMAGE_PUSH_CMD | Command to push the image to remote registry | docker push | | IMAGE_REGISTRY | Container image registry to use | registry.k8s.io/nfd | | IMAGE_TAG_NAME | Container image tag name | <nfd version> | | IMAGE_EXTRA_TAG_NAMES | Additional container image tag(s) to create when building image | *empty* | | K8S_NAMESPACE | nfd-master and nfd-worker namespace | node-feature-discovery | For example, to use a custom registry: ```bash make IMAGE_REGISTRY= ``` Or to specify a build tool different from Docker, It can be done in 2 ways: 1. via environment ```bash IMAGE_BUILD_CMD="buildah bud" make ``` 1. by overriding the variable value ```bash make IMAGE_BUILD_CMD="buildah bud" ``` ### Testing Unit tests are automatically run as part of the container image build. You can also run them manually in the source code tree by running: ```bash make test ``` End-to-end tests are built on top of the e2e test framework of Kubernetes, and, they required a cluster to run them on. For running the tests on your test cluster you need to specify the kubeconfig to be used: ```bash make e2e-test KUBECONFIG=$HOME/.kube/config ``` There are several environment variables that can be used to customize the e2e-tests: | Variable | Description | Default value | | -------------------------- | ----------------------------------------------------------------- | ------------- | | KUBECONFIG | Kubeconfig for running e2e-tests | *empty* | | E2E_TEST_CONFIG | Parameterization file of e2e-tests (see [example][e2e-config-sample]) | *empty* | | E2E_PULL_IF_NOT_PRESENT | True-ish value makes the image pull policy IfNotPresent (to be used only in e2e tests) | false | | E2E_TEST_FULL_IMAGE | Run e2e-test also against the Full Image tag | false | | E2E_GINKGO_LABEL_FILTER | Ginkgo label filter to use for running e2e tests | *empty* | | OPENSHIFT | Non-empty value enables OpenShift specific support (only affects e2e tests) | *empty* | ### NFD-Master For development and debugging it is possible to run nfd-master as a stand-alone binary outside the cluster. The `-no-publish` flag can be used to prevent nfd-master making changes to the nodes. If `-no-publish` is not set, nfd-master also requires the `NODE_NAME` environment variable to be set for cleaning up stale annotations. ```bash make build NODE_NAME= ./nfd-master -no-publish -kubeconfig ~/.kube/config ``` ### NFD-Worker For development and debugging it is possible to run nfd-worker as a stand-alone binary outside the cluster. The `-no-publish` flag can be used to prevent nfd-worker from creating NodeFeature objects in the target cluster. If the `-no-publish` is not set, nfd-worker also requires the `NODE_NAME` and `KUBERNETES_NAMESPACE` environment variables to be defined to create the NodeFeature object in the target cluster. ```bash make build KUBERNETES_NAMESPACE=default NODE_NAME=nonexistent-node ./bin/nfd-worker -kubeconfig ~/.kube/config ``` > **NOTE:** Running nfd-worker locally this way discovers and publishes > features of the local development system you're running nfd-worker on. ### NFD-Topology-Updater For development and debugging it is possible to run nfd-topology-updater as a stand-alone binary outside the cluster. However, it requires access to the kubelet's local pod-resources socket and the kubelet http api so in practice it needs to be run on a host acting as a Kubernetes node and thus running kubelet. Running kubelet with `--read-only-port=10255` (or `readOnlyPort: 10255` in config) makes it possible to connect to kubelet without auth-token (never do this in a production cluster). Also, the `-no-publish` flag can be used to prevent nfd-topology-updater from creating NodeResourceTopology objects in the target cluster. If the `-no-publish` is not set, nfd-topology-updater also requires the `NODE_NAME` and `KUBERNETES_NAMESPACE` environment variables to be defined. ```bash make build KUBERNETES_NAMESPACE=default NODE_NAME=nonexistent-node ./bin/nfd-topology-updater -kubeconfig ~/.kube/config -kubelet-config-uri http://127.0.0.1:10255 ``` ## Running with Tilt Another option for building NFD locally is via Tilt tool, which can build container images, push them to a local registry and reload your Kubernetes pods automatically. When using Tilt, you don't have to build container images and re-deploy your pods manually but instead let the Tilt take care of it. Tiltfile is a configuration file for the Tilt and is located at the root directory. To develop NFD with Tilt, follow the steps below. ### Prerequisites 1. Install [Docker](https://docs.docker.com/engine/install/) 1. Setup Docker as a non-root user. 1. Install [kubectl](https://kubernetes.io/docs/tasks/tools/) 1. Install [kustomize](https://github.com/kubernetes-sigs/kustomize) 1. Install [tilt](https://docs.tilt.dev/install.html) 1. Create a local Kubernetes cluster - Create image registry first - Create a Kubernetes cluster. Please note that docker containers will be served as controller node and worker nodes, and NFD-worker will run as a DaemonSet in nested container. Therefore, to make sure the NFD-worker can discover the host features, the host folders "/boot" and "/lib" should be mounted into worker node docker containers when creating the Kubernetes cluster. 1. Start up node feature discovery development environment To start up your Tilt development environment, run at the root of your local NFD codebase. ```shell tilt up ``` Tilt will start a web interface in the localhost and port 10350. From the web interface, you are able to see how NFD worker and master are progressing, watch their build and runtime logs. Once your code changes are saved locally, Tilt will notice it and re-build the container image from the current code, push the image to the registry and re-deploy NFD pods with the latest container image. ### Environment variables To override environment variables used in the Tiltfile during image build, export them in your current terminal before starting Tilt. ```shell export IMAGE_TAG_NAME="v1" tilt up ``` This will override the default value(`master`) of `IMAGE_TAG_NAME` variable defined in the Tiltfile. ## Documentation All documentation resides under the [docs](https://github.com/kubernetes-sigs/node-feature-discovery/tree/{{site.release}}/docs) directory in the source tree. It is designed to be served as a html site by [GitHub Pages](https://pages.github.com/). Building the documentation is containerized to fix the build environment. The recommended way for developing documentation is to run: ```bash make site-serve ``` This will build the documentation in a container and serve it under [localhost:4000/](http://localhost:4000/) making it easy to verify the results. Any changes made to the `docs/` will automatically re-trigger a rebuild and are reflected in the served content and can be inspected with a browser refresh. To just build the html documentation run: ```bash make site-build ``` This will generate html documentation under `docs/_site/`. [e2e-config-sample]: https://github.com/kubernetes-sigs/node-feature-discovery/blob/{{site.release}}/test/e2e/e2e-test-config.exapmle.yaml [podresource-api]: https://kubernetes.io/docs/concepts/extend-kubernetes/compute-storage-net/device-plugins/#monitoring-device-plugin-resources [feature-gate]: https://kubernetes.io/docs/reference/command-line-tools-reference/feature-gates