Skip to content

Commit

Permalink
tinkerbell/charts: introduce showcase chart
Browse files Browse the repository at this point in the history
- `showcase` is a chart that, based on values.yaml dictionaries:
    - generates Tinkerbell CRs (Hardware/Template/Workflow) for both standard (UEFI) & exotic (supported by Armbian) devices
    - generates download/process jobs for multiple Hook flavors (see tinkerbell/hook#205)
    - generates download/process jobs for a few OS images (Ubuntu Cloud Images, Armbian, etc)
    - should be independent of how one deployed Tinkerbell itself (stack chart, individual components, etc)
- A few features:
    - validates values.yaml for common mistakes; arch must match, etc.
    - validates & handles rootDisk differences (re-invents "formatPartition()" a bit)
    - avoids re-downloading Hooks and Images that are already on disk, even if Job re-runs
    - allows easy way to use
        - custom Hooks
        - custom Kernel cmdline parameters at both the Hook & device level
            - for example `acpi=off` at Hook level and `console=ttyS0` at board level
        - custom OS images for deployment
        - reboot or kexec to finish deployment
        - different partition numbers for OS image's rootfs (some images have ESP, some have a separate `/boot`, etc)
        - control if growpart and/or ssh/user setup is done during provisioning or not
        - conversion of OS images (`qemu-to-raw-gzip` and `xz-to-gz`)
    - has a "merge" mechanism with a common way to set parameters like net gateway, UEFI, etc (also easy to override per-device)
    - default values have everything `enabled: false` thus showcase should produce nothing by default.
        - Hooks & Images can be forced `enabled: true` in values.yaml, or
            - `enabled: true` Devices automatically enable their Hook & Image
- Probably missing:
    - More validations
    - Currently pointing to my Tinkerbell Actions, which I haven't PR'ed yet
- How to use:
    - Clone it, edit the values.yaml to your liking, and deploy.

Signed-off-by: Ricardo Pardini <[email protected]>
  • Loading branch information
rpardini committed Jun 3, 2024
1 parent 35b7539 commit b1caf88
Show file tree
Hide file tree
Showing 5 changed files with 793 additions and 0 deletions.
6 changes: 6 additions & 0 deletions tinkerbell/showcase/Chart.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
apiVersion: v2
name: showcase
description: Generates Tinkerbell CR's for a plethora of standard and exotic hardware; downloads & prepares Hook and OS images for provisioning
type: application
version: "0.0.1"

251 changes: 251 additions & 0 deletions tinkerbell/showcase/templates/devices.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,251 @@
{{- range $deviceId, $dev := .Values.hardware.devices }}

{{- if not $dev.enabled }}
# Device not enabled: {{$deviceId}}
{{- else }}
# Device enabled: {{$deviceId}}
{{- $common := $.Values.hardware.common -}}
{{- $mergedDevice := merge $dev $common }}
{{- $hookObj := index $.Values.provision.hook $mergedDevice.hookRef }}
{{- if not $hookObj }}{{- fail (printf "Device '%s', hookRef '%s': %s" $deviceId $mergedDevice.hookRef "hookRef not found") }}{{- end }}
{{- $imageObj := index $.Values.provision.images $mergedDevice.imageRef }}
{{- if not $imageObj }}{{- (printf "Device '%s', imageRef '%s': %s" $deviceId $mergedDevice.imageRef "imageRef not found") }}{{- end }}
# Check sanity of arch across device / image / hook - they all must match
# Device arch: {{$dev.arch}} Image arch: {{$imageObj.arch}} Hook arch: {{$hookObj.arch}}
{{- if ne $imageObj.arch $hookObj.arch }}{{- fail (printf "Device '%s': '%s'" $deviceId "image and hook arch mismatch") }}{{- end }}
{{- if ne $dev.arch $imageObj.arch }}{{- fail (printf "Device '%s': '%s'" $deviceId "device and image arch mismatch") }}{{- end }}
{{- if ne $dev.arch $hookObj.arch }}{{- fail (printf "Device '%s': '%s'" $deviceId "device and hook arch mismatch") }}{{- end }}
{{- $rootDiskDevice := required (printf "Device '%s' - %s" $deviceId "rootDisk is required") $mergedDevice.rootDisk }}
{{- if not (hasPrefix "/dev" $rootDiskDevice) }}{{- fail (printf "Device '%s' (rootDisk '%s'): '%s'" $deviceId $rootDiskDevice "rootDisk does not begin with /dev") }}{{- end }}
{{- $rootDiskRootfsPartitionNumber := printf "%s" $imageObj.rootfsPartitionNumber }}
# rootDiskRootfsPartitionNumber is {{$rootDiskRootfsPartitionNumber}}
{{- $rootDiskRootfsPartitionDevice := "unknown" }}

{{- if hasPrefix "/dev/disk/" $rootDiskDevice }}
# YES! {{$rootDiskDevice}} begins with /dev/disk
{{- $rootDiskRootfsPartitionDevice = printf "%s-part%s" $rootDiskDevice $rootDiskRootfsPartitionNumber }}
{{- else }}
# NO! {{$rootDiskDevice}} does not begin with /dev/disk
{{- if regexMatch "[0-9]$" $rootDiskDevice }}
# YES! {{$rootDiskDevice}} ends with a digit - REGEX MATCH
{{- $rootDiskRootfsPartitionDevice = printf "%sp%s" $rootDiskDevice $rootDiskRootfsPartitionNumber }}
{{- else }}
# NO! {{$rootDiskDevice}} does not end with a digit - REGEX NOT MATCH
{{- $rootDiskRootfsPartitionDevice = printf "%s%s" $rootDiskDevice $rootDiskRootfsPartitionNumber }}
{{- end }}
{{- end }}
# Thus at the end of the day, $rootDiskRootfsPartitionDevice is {{$rootDiskRootfsPartitionDevice}}
---
apiVersion: "tinkerbell.org/v1alpha1"
kind: Hardware
metadata:
name: "{{ $deviceId }}-hardware"
labels:
"app.kubernetes.io/instance": "{{ $.Release.Name }}"
"app.kubernetes.io/part-of": "tinkerbell-showcase"
spec:
disks:
- device: "{{ $rootDiskDevice }}"
metadata:
facility:
facility_code: sandbox
instance:
hostname: "{{ $deviceId }}"
id: "{{ $dev.mac }}"
operating_system:
distro: "ubuntu" # @TODO
os_slug: "ubuntu_20_04"
version: "20.04"
interfaces:
- dhcp:
arch: "{{ $dev.arch }}"
hostname: "{{ $deviceId }}"
ip:
address: "{{ $dev.ipv4.address }}"
netmask: "{{ $dev.ipv4.netmask }}"
gateway: "{{ $dev.ipv4.gateway }}"
lease_time: 86400
mac: "{{ $dev.mac }}"
name_servers:
{{- range $mergedDevice.ipv4.dns }}
- {{ . | quote }}
{{- end }}
uefi: {{ $dev.uefi }}
netboot:
allowPXE: true
allowWorkflow: true
ipxe: # @TODO
contents: |
echo Showcase starting for {{$deviceId}} with hook {{$mergedDevice.hookRef}} and image {{$mergedDevice.imageRef}}...
set download-url {{ $.Values.tinkerbell.hookURL }}
set kernel-params tink_worker_image=quay.io/tinkerbell/tink-worker:latest facility= syslog_host={{ $.Values.tinkerbell.syslogHost }} grpc_authority={{ $.Values.tinkerbell.grpcAuthority }} tinkerbell_tls=false worker_id={{$dev.mac}} hw_addr={{$dev.mac}} modules=loop,squashfs,sd-mod,usb-storage initrd={{ $hookObj.initrd }} {{$hookObj.kernelCommandLine}} {{ $dev.extraKernelCommandLine }}
echo Kernel image: ${download-url}/{{ $hookObj.kernel }}
echo Kernel initrd: ${download-url}/{{ $hookObj.initrd }}
echo Kernel cmdline: ${kernel-params}
kernel ${download-url}/{{ $hookObj.kernel }} ${kernel-params}
initrd ${download-url}/{{ $hookObj.initrd }}
boot
---
apiVersion: "tinkerbell.org/v1alpha1"
kind: Template
metadata:
name: "{{ $deviceId }}-template"
labels:
"app.kubernetes.io/instance": "{{ $.Release.Name }}"
"app.kubernetes.io/part-of": "tinkerbell-showcase"
spec:
data: |
version: "0.1"
name: "{{ $deviceId }}-template"
global_timeout: 1800
tasks:
- name: "os-installation-{{ $deviceId }}"
worker: "{{ $dev.mac }}"
volumes:
- /dev:/dev
- /dev/console:/dev/console
- /lib/firmware:/lib/firmware:ro
actions:
- name: "stream-ubuntu-image-{{ $deviceId }}"
#image: quay.io/tinkerbell-actions/image2disk:v1.0.0
image: {{ $.Values.actions.repository }}/image2disk:{{ $.Values.actions.version }}
timeout: 600
environment:
DEST_DISK: "{{ $rootDiskDevice }}"
IMG_URL: "{{ $.Values.tinkerbell.imagesURL }}/{{ $imageObj.image }}"
COMPRESSED: true
{{- if $imageObj.doGrowPart }}
- name: "grow-partition-{{ $deviceId }}"
image: {{ $.Values.actions.repository }}/cexec:{{ $.Values.actions.version }}
timeout: 90
environment:
BLOCK_DEVICE: {{ $rootDiskRootfsPartitionDevice }}
FS_TYPE: ext4
CHROOT: y
DEFAULT_INTERPRETER: "/bin/sh -c"
CMD_LINE: "growpart {{ $rootDiskDevice }} {{$rootDiskRootfsPartitionNumber}} && resize2fs {{$rootDiskRootfsPartitionDevice}}"
{{- end }}
- name: "fix-resolv-{{ $deviceId }}"
image: {{ $.Values.actions.repository }}/cexec:{{ $.Values.actions.version }}
timeout: 90
environment:
BLOCK_DEVICE: {{$rootDiskRootfsPartitionDevice}}
FS_TYPE: ext4
CHROOT: y
DEFAULT_INTERPRETER: "/bin/sh -c"
CMD_LINE: "echo 'list /etc/resolv.conf: '; ls -la /etc/resolv.conf; echo 'cat /etc/resolv.conf'; cat /etc/resolv.conf; echo 'moving...'; mv -v /etc/resolv.conf /etc/resolv.conf.orig.tink; echo 'nameserver {{ index $mergedDevice.ipv4.dns 0 }} ' > /etc/resolv.conf; echo 'new resolf.conf:' ; cat /etc/resolv.conf"
{{- if $imageObj.doUserAndSshSetup }}
- name: "install-openssl-{{ $deviceId }}"
image: {{ $.Values.actions.repository }}/cexec:{{ $.Values.actions.version }}
timeout: 90
environment:
BLOCK_DEVICE: {{$rootDiskRootfsPartitionDevice}}
FS_TYPE: ext4
CHROOT: y
DEFAULT_INTERPRETER: "/bin/sh -c"
CMD_LINE: "apt -y update && apt -y install openssl"
- name: "create-user-{{ $deviceId }}"
image: {{ $.Values.actions.repository }}/cexec:{{ $.Values.actions.version }}
timeout: 90
environment:
BLOCK_DEVICE: {{$rootDiskRootfsPartitionDevice}}
FS_TYPE: ext4
CHROOT: y
DEFAULT_INTERPRETER: "/bin/sh -c"
CMD_LINE: "useradd -p $(openssl passwd -1 tink) -s /bin/bash -d /home/tink/ -m -G sudo tink"
- name: "enable-ssh-{{ $deviceId }}"
image: {{ $.Values.actions.repository }}/cexec:{{ $.Values.actions.version }}
timeout: 90
environment:
BLOCK_DEVICE: {{$rootDiskRootfsPartitionDevice}}
FS_TYPE: ext4
CHROOT: y
DEFAULT_INTERPRETER: "/bin/sh -c"
CMD_LINE: "ssh-keygen -A; systemctl enable ssh.service; echo 'PasswordAuthentication yes' > /etc/ssh/sshd_config.d/60-cloudimg-settings.conf"
- name: "disable-apparmor-{{ $deviceId }}"
image: {{ $.Values.actions.repository }}/cexec:{{ $.Values.actions.version }}
timeout: 90
environment:
BLOCK_DEVICE: {{$rootDiskRootfsPartitionDevice}}
FS_TYPE: ext4
CHROOT: y
DEFAULT_INTERPRETER: "/bin/sh -c"
CMD_LINE: "systemctl disable apparmor; systemctl disable snapd"
- name: "write-netplan-{{ $deviceId }}"
image: {{ $.Values.actions.repository }}/writefile:{{ $.Values.actions.version }}
timeout: 90
environment:
DEST_DISK: {{$rootDiskRootfsPartitionDevice}}
FS_TYPE: ext4
DEST_PATH: /etc/netplan/config.yaml
CONTENTS: |
network:
version: 2
renderer: networkd
ethernets:
id0:
match:
name: en*
dhcp4: true
UID: 0
GID: 0
MODE: 0644
DIRMODE: 0755
{{- end }}
- name: "revert-fix-resolv-{{ $deviceId }}"
image: {{ $.Values.actions.repository }}/cexec:{{ $.Values.actions.version }}
timeout: 90
environment:
BLOCK_DEVICE: {{$rootDiskRootfsPartitionDevice}}
FS_TYPE: ext4
CHROOT: y
DEFAULT_INTERPRETER: "/bin/sh -c"
CMD_LINE: "rm -v /etc/resolv.conf; mv -v /etc/resolv.conf.orig.tink /etc/resolv.conf"
{{- if eq $hookObj.bootMode "kexec" }}
- name: "kexec-{{ $deviceId }}"
image: ghcr.io/jacobweinstock/waitdaemon:latest
timeout: 90
pid: host
environment:
BLOCK_DEVICE: {{$rootDiskRootfsPartitionDevice}}
FS_TYPE: ext4
IMAGE: {{ $.Values.actions.repository }}/kexec:{{ $.Values.actions.version }}
WAIT_SECONDS: 1
volumes:
- /var/run/docker.sock:/var/run/docker.sock
{{- end }}
{{- if eq $hookObj.bootMode "reboot" }}
- name: "reboot-{{ $deviceId }}"
image: ghcr.io/jacobweinstock/waitdaemon:latest
timeout: 90
pid: host
command: ["reboot"]
environment:
IMAGE: alpine
WAIT_SECONDS: 1
volumes:
- /var/run/docker.sock:/var/run/docker.sock
{{- end }}
---
apiVersion: "tinkerbell.org/v1alpha1"
kind: Workflow
metadata:
name: "{{ $deviceId }}-workflow"
labels:
"app.kubernetes.io/instance": "{{ $.Release.Name }}"
"app.kubernetes.io/part-of": "tinkerbell-showcase"
spec:
templateRef: "{{ $deviceId }}-template"
hardwareRef: "{{ $deviceId }}-hardware"
hardwareMap:
device_1: "{{ $dev.mac }}"
{{- end }}
{{- end }}
94 changes: 94 additions & 0 deletions tinkerbell/showcase/templates/hooks.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
{{- range $hookId, $hook := .Values.provision.hook }}
{{- $common := $.Values.provision.common -}}
{{- $hook = merge $hook $common }}
---
# --> HookId: {{$hookId}}
# downloadId: {{ $hook.hookDownloadId }}
# downloadFile: {{ $hook.downloadFile }}
# hookDownloadBaseUrl: {{ $hook.hookDownloadBaseUrl }}
{{- $hash := trunc 8 (sha1sum (printf "%s-%s-%s" $hook.hookDownloadId $hook.hookDownloadBaseUrl $hook.downloadFile)) }}
# Loop over $.Values.hardware.devices (a dictionary) and for each value, check if it's hookRef matches this $hookId - if so, enable this hook.
{{- $enabledByDeviceUsage := false }}
{{- range $device := $.Values.hardware.devices }}
# Testing {{$device.hookRef}} against {{$hookId}}...
{{- if and $device.enabled (eq $device.hookRef $hookId) -}}
# YES!
{{- $enabledByDeviceUsage = true -}}
{{- else -}}
# NO!
{{- end -}}
{{- end }}

{{- if or $enabledByDeviceUsage $hook.enabled }}
# Enabled hook: {{$hookId}} - force enabled? {{$hook.enabled}} -- enabled by device usage? {{$enabledByDeviceUsage}}
apiVersion: v1
kind: ConfigMap
metadata:
name: "download-hook-{{$hookId}}-{{$hash}}"
namespace: "{{ $.Release.Namespace }}"
labels:
"app.kubernetes.io/instance": "{{ $.Release.Name }}"
"app.kubernetes.io/part-of": "tinkerbell-showcase"
data:
entrypoint.sh: |
#!/usr/bin/env bash
# This script is designed to download the Hook artifacts.
set -euxo pipefail
if [[ -f "/output/download_hook_{{$hookId}}_{{$hash}}.done" ]]; then
echo "Hook already downloaded ({{$hookId}} - hash: {{$hash}}), skipping."
exit 0
fi
mkdir -p "/output/download_hook_{{$hookId}}"
cd "/output/download_hook_{{$hookId}}"
declare down_url="{{ $hook.hookDownloadBaseURL }}{{$hook.downloadFile}}"
declare down_file="/output/download_hook_{{$hookId}}/{{$hook.downloadFile}}"
wget -O "${down_file}.tmp" "${down_url}"
mv -v "${down_file}.tmp" "${down_file}"
mkdir -p "/output/download_hook_{{$hookId}}/extract"
cd "/output/download_hook_{{$hookId}}/extract"
tar -xvzf "${down_file}"
mv -v "/output/download_hook_{{$hookId}}/extract/"* /output/
rm -rf "/output/download_hook_{{$hookId}}"
ls -lah "/output/{{$hook.kernel}}"
ls -lah "/output/{{$hook.initrd}}"
touch "/output/download_hook_{{$hookId}}_{{$hash}}.done"
---
apiVersion: batch/v1
kind: Job
metadata:
name: "download-hook-{{$hookId}}-{{$hash}}"
namespace: {{ $.Release.Namespace }}
labels:
"app.kubernetes.io/instance": "{{ $.Release.Name }}"
"app.kubernetes.io/part-of": "tinkerbell-showcase"
spec:
backoffLimit: 50
template:
metadata:
labels:
app: download-hook
spec:
containers:
- name: download-hook-{{$hookId}}
image: bash:5
command: [ "/script/entrypoint.sh" ]
volumeMounts:
- mountPath: /output
name: hook-artifacts
- mountPath: /script
name: configmap-volume
restartPolicy: OnFailure
volumes:
- name: hook-artifacts
hostPath:
path: {{ $.Values.tinkerbell.hostDirectory | quote }}
type: DirectoryOrCreate
- name: configmap-volume
configMap:
defaultMode: 0700
name: "download-hook-{{$hookId}}-{{$hash}}"
{{- else }}
# Disabled Hook: {{$hookId}}
{{- end }}
{{- end }}
Loading

0 comments on commit b1caf88

Please sign in to comment.