Documentation for version: latest
These are the base repos that project-flotta needs to run as expected:
Dependecy | Repo | Git Version |
---|---|---|
flotta-operator | project-flotta/flotta-operator | main |
flotta-device-worker | project-flotta/flotta-device-worker | main |
yggdrasil | RedHatInsights/yggdrasil | main(since d69d38da3de3c0a473cd0bd6a713e72c2ea842f9) |
By default, project use Makefiles, and all the project has a help section where information can be obtained:
This is the example when running make help in the flotta-operator repo.
$ -> make help
Usage:
make <target>
General
help Display this help.
Development
manifests Generate WebhookConfiguration, ClusterRole and CustomResourceDefinition objects.
generate Generate code containing DeepCopy, DeepCopyInto, and DeepCopyObject method implementations.
fmt Run go fmt against code.
vet Run go vet against code.
gosec Run gosec locally
imports fix and format go imports
lint Check if the go code is properly written, rules are in .golangci.yml
test Run tests.
Build
build Build manager binary.
fast-build Fast build manager binary for local dev.
run Run a controller from your host.
docker-build Build docker image with the manager.
docker-push Push docker image with the manager.
Deployment
install Install CRDs into the K8s cluster specified in ~/.kube/config.
uninstall Uninstall CRDs from the K8s cluster specified in ~/.kube/config.
deploy Deploy controller to the K8s cluster specified in ~/.kube/config.
undeploy Undeploy controller from the K8s cluster specified in ~/.kube/config.
gen-manifests Generates manifests for deploying the operator into $(TARGET)-flotta-operator.yaml
install-router Install openshift router
controller-gen Download controller-gen locally if necessary.
kustomize Download kustomize locally if necessary.
ginkgo Download ginkgo locally if necessary.
client-gen Download client-gen locally if necessary.
lister-gen Download lister-gen locally if necessary.
informer-gen Download client-gen locally if necessary.
k8s-client-gen Generate typed client for flotta project.
validate-swagger Validate swagger
$ ->
For running operator, you need to install the CRD first, the best way to develop is in a small Kubernetes cluster, like Kind or Minikube.
To install Operator you need to think that there are two main parts, the CRD installation and the operator itself.
To install CRD, you need to run the following:
$ -> make install
/home/eloy/dev/upstream/project-flotta/k4e-operator/bin/kustomize build config/crd | kubectl apply -f -
customresourcedefinition.apiextensions.k8s.io/edgeconfigs.management.project-flotta.io configured
customresourcedefinition.apiextensions.k8s.io/edgedevices.management.project-flotta.io configured
customresourcedefinition.apiextensions.k8s.io/edgedevicesets.management.project-flotta.io configured
customresourcedefinition.apiextensions.k8s.io/edgedevicesignedrequest.management.project-flotta.io configured
customresourcedefinition.apiextensions.k8s.io/edgeworkloads.management.project-flotta.io configured
customresourcedefinition.apiextensions.k8s.io/playbookexecutions.management.project-flotta.io configured
To run the operator, some dependencies need to be executed first, like openshift-router and cert-manager:
make install-router
make install-cert-manager
To run operator locally, you can use the following code:
make run
To run the Edge API locally, execute:
make http-api-run
For registering devices, it’s important to download certs and set correct permission:
$ -> sudo rm /tmp/*.pem
$ -> make get-certs
kubectl get secret -n flotta flotta-ca -o go-template='' >/tmp/ca.pem
kubectl get secret -n flotta reg-client-ca-2f5cpqwrhm -o go-template='' > /tmp/cert.pem
kubectl get secret -n flotta reg-client-ca-2f5cpqwrhm -o go-template='' > /tmp/key.pem
$ -> sudo chown root:root /tmp/*.pem
On device, we have two different things to install, yggdrasil and device worker. There are two ways to run this piece:
⚠️ Golang 1.16: By default, device needs to compile in Golang 1.16 mainly because this software needs to be packaged as RPM, so the version for RHEL8 is golang 1.16
This way, means that Yggdrasil already knows where the device-worker is located, and it’s defined on workers.toml.
So for this, on device-worker repo the following command should be run:
sudo --preserve-env=XDG_RUNTIME_DIR,HOME,USER make install
This will create two files:
$ -> cat /usr/local/etc/yggdrasil/workers/device-worker.toml
exec = "/usr/local/libexec/yggdrasil/device-worker"
protocol = "grpc"
env = []
$ -> ls /usr/local/libexec/yggdrasil/device-worker
/usr/local/libexec/yggdrasil/device-worker
And yggdrasil can run with the following command:
sudo ./yggd \
--log-level trace \
--protocol http\
--path-prefix api/flotta-management/v1\
--client-id $(cat /etc/machine-id)\
--cert-file /tmp/cert.pem\
--key-file /tmp/key.pem\
--ca-root /tmp/ca.pem\
--server 127.0.0.1:8043
That you can see that device-worker is running and starting to get data from Flotta Edge API:
[yggd] 2022/05/10 15:55:52 /home/eloy/dev/upstream/project-flotta/yggdrasil/cmd/yggd/main.go:201: starting yggd version 0.2.98
....
[yggd] 2022/05/10 15:55:52 /home/eloy/dev/upstream/project-flotta/yggdrasil/cmd/yggd/main.go:377: starting worker: device-worker
...
[yggd] 2022/05/10 15:55:52 /home/eloy/dev/upstream/project-flotta/yggdrasil/cmd/yggd/worker.go:114: /usr/local/libexec/yggdrasil/device-worker: Data directory: /usr/local/etc/yggdrasil/device
[yggd] 2022/05/10 15:55:52 /home/eloy/dev/upstream/project-flotta/yggdrasil/cmd/yggd/worker.go:114: /usr/local/libexec/yggdrasil/device-worker: device config file: /usr/local/etc/yggdrasil/device/device-config.json
Running device-worker managed by yggdrasil is how the end users run in their devices, so this is the closest way to replicate user issues.
Because separate services are useful when developing, this way is useful to be able to separate logs from yggdrasil and device-worker, and be able to focus only on the device-worker part:
Run yggdrasil in one tab, with the following unixsocket @yggd
sudo ./yggd \
--log-level trace \
--protocol http\
--path-prefix api/flotta-management/v1\
--client-id $(cat /etc/machine-id)\
--cert-file /tmp/cert.pem\
--key-file /tmp/key.pem\
--ca-root /tmp/ca.pem\
--socket-addr @yggd\
--server 127.0.0.1:8043
And in other tab, just start device-worker manually, and making sure that
/usr/local/etc/yggdrasil/workers/device-worker.toml
is not present at all.
sudo -E YGG_LOG_LEVEL=debug \
YGG_CONFIG_DIR="/tmp/device" \
YGG_SOCKET_ADDR="unix:@yggd" \
YGG_CLIENT_ID="$(cat /etc/machine-id)" \
FLOTTA_XDG_RUNTIME_DIR=$XDG_RUNTIME_DIR \
go run cmd/device-worker/main.go
And now, device-worker will register to the API, and start to do all enrolment/registration phase.
If you encounter issue while installing the device, check the following:
[yggd] 2022/05/18 09:19:17 yggdrasil/cmd/yggd/worker.go:114: /usr/local/libexec/yggdrasil/device-worker: Data directory: /usr/local/etc/yggdrasil/device
[yggd] 2022/05/18 09:19:17 yggdrasil/cmd/yggd/worker.go:114: /usr/local/libexec/yggdrasil/device-worker: device config file: /usr/local/etc/yggdrasil/device/device-config.json
Here the PREFIX used by yggdrasil is /usr/local/
so you need to check that the same base path is used in the device-worker, see the make install
output:
install -D -m 755 ./bin/device-worker /usr/local/libexec/yggdrasil/device-worker
If the base path does not match, re-install yggdrasil: ensure the environment variables PREFIX and MAKE_FLAGS are not set then, from yggdrasil folder, run make clean install
and retry to run the device-worker.
${PREFIX}/etc/yggdrasil/device/device-config.json
. If so, delete it and retry to run yggdrasil. By default, PREFIX equals /usr/local
.kubectl edit edsr <id of the device>
. Then, search for approved
and set the value to true
[yggd] 2022/02/21 16:55:00 ./cmd/yggd/main.go:189: starting yggd version 0.2.98
cannot stop workers: cannot stop worker: cannot stop worker: cannot stop process: os: process already finished
Run the following command
sudo rm ${PREFIX}/var/run/yggdrasil/workers/device-worker.pid
By default PREFIX equals /usr/local
.
DEVICE=<device id>
kubectl patch edgedevice/$DEVICE --type=merge -p '{"metadata": {"finalizers":null}}'
kubectl delete edgedevice/$DEVICE
Because Flotta device works uses rootless podman, can be a way that podman pod is listening correctly but the podman user socket is not yet started. For podman be able to run, it’s important to check the status:
Error message:
cannot start Workload Manager. DeviceID: eloy-test; err: workload cannot initialize podman manager: unable to connect to Podman socket: Get "http://d/v3.4.2/libpod/_ping": dial unix ///run/user/1000/podman/podman.sock: connect: no such file or directory
Podman status:
$ -> sudo systemctl status podman
○ podman.service - Podman API Service
Loaded: loaded (/usr/lib/systemd/system/podman.service; disabled; vendor preset: disabled)
Active: inactive (dead) since Thu 2022-06-02 10:04:50 CEST; 5h 2min ago
TriggeredBy: ● podman.socket
Docs: man:podman-system-service(1)
Process: 34417 ExecStart=/usr/bin/podman $LOGGING system service (code=exited, status=0/SUCCESS)
Main PID: 34417 (code=exited, status=0/SUCCESS)
Jun 02 10:04:45 localhost.localdomain systemd[1]: Started Podman API Service.
Jun 02 10:04:45 localhost.localdomain podman[34417]: time="2022-06-02T10:04:45+02:00" level=info msg="/usr/bin/podman filtering at log level info"
Jun 02 10:04:45 localhost.localdomain podman[34417]: time="2022-06-02T10:04:45+02:00" level=info msg="Not using native diff for overlay, this may cause degraded perform>
Jun 02 10:04:45 localhost.localdomain podman[34417]: time="2022-06-02T10:04:45+02:00" level=info msg="Found CNI network podman (type=bridge) at /etc/cni/net.d/87-podman>
Jun 02 10:04:45 localhost.localdomain podman[34417]: time="2022-06-02T10:04:45+02:00" level=info msg="Found CNI network kind (type=bridge) at /etc/cni/net.d/kind.confli>
Jun 02 10:04:45 localhost.localdomain podman[34417]: time="2022-06-02T10:04:45+02:00" level=info msg="Setting parallel job count to 25"
Jun 02 10:04:45 localhost.localdomain podman[34417]: time="2022-06-02T10:04:45+02:00" level=info msg="using systemd socket activation to determine API endpoint"
Jun 02 10:04:45 localhost.localdomain podman[34417]: time="2022-06-02T10:04:45+02:00" level=info msg="using API endpoint: ''"
Jun 02 10:04:45 localhost.localdomain podman[34417]: time="2022-06-02T10:04:45+02:00" level=info msg="API service listening on \"/run/podman/podman.sock\""
Jun 02 10:04:50 localhost.localdomain systemd[1]: podman.service: Deactivated successfully.
Podman user socket status:
$ -> systemctl status --user podman.socket
● podman.socket - Podman API Socket
Loaded: loaded (/usr/lib/systemd/user/podman.socket; disabled; vendor preset: disabled)
Active: active (running) since Thu 2022-06-02 10:17:50 CEST; 4h 50min ago
Triggers: ● podman.service
Docs: man:podman-system-service(1)
Listen: /run/user/1000/podman/podman.sock (Stream)
CGroup: /user.slice/user-1000.slice/user@1000.service/app.slice/podman.socket
Jun 02 10:17:50 localhost.localdomain systemd[13455]: Listening on Podman API Socket.
$ ->