Init project
This commit is contained in:
		
							
								
								
									
										25
									
								
								.devcontainer/devcontainer.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								.devcontainer/devcontainer.json
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,25 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "Kubebuilder DevContainer",
 | 
			
		||||
  "image": "golang:1.24",
 | 
			
		||||
  "features": {
 | 
			
		||||
    "ghcr.io/devcontainers/features/docker-in-docker:2": {},
 | 
			
		||||
    "ghcr.io/devcontainers/features/git:1": {}
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  "runArgs": ["--network=host"],
 | 
			
		||||
 | 
			
		||||
  "customizations": {
 | 
			
		||||
    "vscode": {
 | 
			
		||||
      "settings": {
 | 
			
		||||
        "terminal.integrated.shell.linux": "/bin/bash"
 | 
			
		||||
      },
 | 
			
		||||
      "extensions": [
 | 
			
		||||
        "ms-kubernetes-tools.vscode-kubernetes-tools",
 | 
			
		||||
        "ms-azuretools.vscode-docker"
 | 
			
		||||
      ]
 | 
			
		||||
    }
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  "onCreateCommand": "bash .devcontainer/post-install.sh"
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										23
									
								
								.devcontainer/post-install.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.devcontainer/post-install.sh
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,23 @@
 | 
			
		||||
#!/bin/bash
 | 
			
		||||
set -x
 | 
			
		||||
 | 
			
		||||
curl -Lo ./kind https://kind.sigs.k8s.io/dl/latest/kind-linux-amd64
 | 
			
		||||
chmod +x ./kind
 | 
			
		||||
mv ./kind /usr/local/bin/kind
 | 
			
		||||
 | 
			
		||||
curl -L -o kubebuilder https://go.kubebuilder.io/dl/latest/linux/amd64
 | 
			
		||||
chmod +x kubebuilder
 | 
			
		||||
mv kubebuilder /usr/local/bin/
 | 
			
		||||
 | 
			
		||||
KUBECTL_VERSION=$(curl -L -s https://dl.k8s.io/release/stable.txt)
 | 
			
		||||
curl -LO "https://dl.k8s.io/release/$KUBECTL_VERSION/bin/linux/amd64/kubectl"
 | 
			
		||||
chmod +x kubectl
 | 
			
		||||
mv kubectl /usr/local/bin/kubectl
 | 
			
		||||
 | 
			
		||||
docker network create -d=bridge --subnet=172.19.0.0/24 kind
 | 
			
		||||
 | 
			
		||||
kind version
 | 
			
		||||
kubebuilder version
 | 
			
		||||
docker --version
 | 
			
		||||
go version
 | 
			
		||||
kubectl version --client
 | 
			
		||||
							
								
								
									
										3
									
								
								.dockerignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.dockerignore
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
			
		||||
# More info: https://docs.docker.com/engine/reference/builder/#dockerignore-file
 | 
			
		||||
# Ignore build and test binaries.
 | 
			
		||||
bin/
 | 
			
		||||
							
								
								
									
										23
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,23 @@
 | 
			
		||||
name: Lint
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  push:
 | 
			
		||||
  pull_request:
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  lint:
 | 
			
		||||
    name: Run on Ubuntu
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Clone the code
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
 | 
			
		||||
      - name: Setup Go
 | 
			
		||||
        uses: actions/setup-go@v5
 | 
			
		||||
        with:
 | 
			
		||||
          go-version-file: go.mod
 | 
			
		||||
 | 
			
		||||
      - name: Run linter
 | 
			
		||||
        uses: golangci/golangci-lint-action@v8
 | 
			
		||||
        with:
 | 
			
		||||
          version: v2.1.0
 | 
			
		||||
							
								
								
									
										32
									
								
								.github/workflows/test-e2e.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								.github/workflows/test-e2e.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,32 @@
 | 
			
		||||
name: E2E Tests
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  push:
 | 
			
		||||
  pull_request:
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  test-e2e:
 | 
			
		||||
    name: Run on Ubuntu
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Clone the code
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
 | 
			
		||||
      - name: Setup Go
 | 
			
		||||
        uses: actions/setup-go@v5
 | 
			
		||||
        with:
 | 
			
		||||
          go-version-file: go.mod
 | 
			
		||||
 | 
			
		||||
      - name: Install the latest version of kind
 | 
			
		||||
        run: |
 | 
			
		||||
          curl -Lo ./kind https://kind.sigs.k8s.io/dl/latest/kind-linux-amd64
 | 
			
		||||
          chmod +x ./kind
 | 
			
		||||
          sudo mv ./kind /usr/local/bin/kind
 | 
			
		||||
 | 
			
		||||
      - name: Verify kind installation
 | 
			
		||||
        run: kind version
 | 
			
		||||
 | 
			
		||||
      - name: Running Test e2e
 | 
			
		||||
        run: |
 | 
			
		||||
          go mod tidy
 | 
			
		||||
          make test-e2e
 | 
			
		||||
							
								
								
									
										23
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,23 @@
 | 
			
		||||
name: Tests
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  push:
 | 
			
		||||
  pull_request:
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  test:
 | 
			
		||||
    name: Run on Ubuntu
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Clone the code
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
 | 
			
		||||
      - name: Setup Go
 | 
			
		||||
        uses: actions/setup-go@v5
 | 
			
		||||
        with:
 | 
			
		||||
          go-version-file: go.mod
 | 
			
		||||
 | 
			
		||||
      - name: Running Tests
 | 
			
		||||
        run: |
 | 
			
		||||
          go mod tidy
 | 
			
		||||
          make test
 | 
			
		||||
							
								
								
									
										27
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
# Binaries for programs and plugins
 | 
			
		||||
*.exe
 | 
			
		||||
*.exe~
 | 
			
		||||
*.dll
 | 
			
		||||
*.so
 | 
			
		||||
*.dylib
 | 
			
		||||
bin/*
 | 
			
		||||
Dockerfile.cross
 | 
			
		||||
 | 
			
		||||
# Test binary, built with `go test -c`
 | 
			
		||||
*.test
 | 
			
		||||
 | 
			
		||||
# Output of the go coverage tool, specifically when used with LiteIDE
 | 
			
		||||
*.out
 | 
			
		||||
 | 
			
		||||
# Go workspace file
 | 
			
		||||
go.work
 | 
			
		||||
 | 
			
		||||
# Kubernetes Generated files - skip generated files, except for vendored files
 | 
			
		||||
!vendor/**/zz_generated.*
 | 
			
		||||
 | 
			
		||||
# editor and IDE paraphernalia
 | 
			
		||||
.idea
 | 
			
		||||
.vscode
 | 
			
		||||
*.swp
 | 
			
		||||
*.swo
 | 
			
		||||
*~
 | 
			
		||||
							
								
								
									
										52
									
								
								.golangci.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								.golangci.yml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,52 @@
 | 
			
		||||
version: "2"
 | 
			
		||||
run:
 | 
			
		||||
  allow-parallel-runners: true
 | 
			
		||||
linters:
 | 
			
		||||
  default: none
 | 
			
		||||
  enable:
 | 
			
		||||
    - copyloopvar
 | 
			
		||||
    - dupl
 | 
			
		||||
    - errcheck
 | 
			
		||||
    - ginkgolinter
 | 
			
		||||
    - goconst
 | 
			
		||||
    - gocyclo
 | 
			
		||||
    - govet
 | 
			
		||||
    - ineffassign
 | 
			
		||||
    - lll
 | 
			
		||||
    - misspell
 | 
			
		||||
    - nakedret
 | 
			
		||||
    - prealloc
 | 
			
		||||
    - revive
 | 
			
		||||
    - staticcheck
 | 
			
		||||
    - unconvert
 | 
			
		||||
    - unparam
 | 
			
		||||
    - unused
 | 
			
		||||
  settings:
 | 
			
		||||
    revive:
 | 
			
		||||
      rules:
 | 
			
		||||
        - name: comment-spacings
 | 
			
		||||
        - name: import-shadowing
 | 
			
		||||
  exclusions:
 | 
			
		||||
    generated: lax
 | 
			
		||||
    rules:
 | 
			
		||||
      - linters:
 | 
			
		||||
          - lll
 | 
			
		||||
        path: api/*
 | 
			
		||||
      - linters:
 | 
			
		||||
          - dupl
 | 
			
		||||
          - lll
 | 
			
		||||
        path: internal/*
 | 
			
		||||
    paths:
 | 
			
		||||
      - third_party$
 | 
			
		||||
      - builtin$
 | 
			
		||||
      - examples$
 | 
			
		||||
formatters:
 | 
			
		||||
  enable:
 | 
			
		||||
    - gofmt
 | 
			
		||||
    - goimports
 | 
			
		||||
  exclusions:
 | 
			
		||||
    generated: lax
 | 
			
		||||
    paths:
 | 
			
		||||
      - third_party$
 | 
			
		||||
      - builtin$
 | 
			
		||||
      - examples$
 | 
			
		||||
							
								
								
									
										33
									
								
								Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								Dockerfile
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,33 @@
 | 
			
		||||
# Build the manager binary
 | 
			
		||||
FROM golang:1.24 AS builder
 | 
			
		||||
ARG TARGETOS
 | 
			
		||||
ARG TARGETARCH
 | 
			
		||||
 | 
			
		||||
WORKDIR /workspace
 | 
			
		||||
# Copy the Go Modules manifests
 | 
			
		||||
COPY go.mod go.mod
 | 
			
		||||
COPY go.sum go.sum
 | 
			
		||||
# cache deps before building and copying source so that we don't need to re-download as much
 | 
			
		||||
# and so that source changes don't invalidate our downloaded layer
 | 
			
		||||
RUN go mod download
 | 
			
		||||
 | 
			
		||||
# Copy the go source
 | 
			
		||||
COPY cmd/main.go cmd/main.go
 | 
			
		||||
COPY api/ api/
 | 
			
		||||
COPY internal/ internal/
 | 
			
		||||
 | 
			
		||||
# Build
 | 
			
		||||
# the GOARCH has not a default value to allow the binary be built according to the host where the command
 | 
			
		||||
# was called. For example, if we call make docker-build in a local env which has the Apple Silicon M1 SO
 | 
			
		||||
# the docker BUILDPLATFORM arg will be linux/arm64 when for Apple x86 it will be linux/amd64. Therefore,
 | 
			
		||||
# by leaving it empty we can ensure that the container and binary shipped on it will have the same platform.
 | 
			
		||||
RUN CGO_ENABLED=0 GOOS=${TARGETOS:-linux} GOARCH=${TARGETARCH} go build -a -o manager cmd/main.go
 | 
			
		||||
 | 
			
		||||
# Use distroless as minimal base image to package the manager binary
 | 
			
		||||
# Refer to https://github.com/GoogleContainerTools/distroless for more details
 | 
			
		||||
FROM gcr.io/distroless/static:nonroot
 | 
			
		||||
WORKDIR /
 | 
			
		||||
COPY --from=builder /workspace/manager .
 | 
			
		||||
USER 65532:65532
 | 
			
		||||
 | 
			
		||||
ENTRYPOINT ["/manager"]
 | 
			
		||||
							
								
								
									
										232
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										232
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,232 @@
 | 
			
		||||
# Image URL to use all building/pushing image targets
 | 
			
		||||
IMG ?= controller:latest
 | 
			
		||||
 | 
			
		||||
# Get the currently used golang install path (in GOPATH/bin, unless GOBIN is set)
 | 
			
		||||
ifeq (,$(shell go env GOBIN))
 | 
			
		||||
GOBIN=$(shell go env GOPATH)/bin
 | 
			
		||||
else
 | 
			
		||||
GOBIN=$(shell go env GOBIN)
 | 
			
		||||
endif
 | 
			
		||||
 | 
			
		||||
# CONTAINER_TOOL defines the container tool to be used for building images.
 | 
			
		||||
# Be aware that the target commands are only tested with Docker which is
 | 
			
		||||
# scaffolded by default. However, you might want to replace it to use other
 | 
			
		||||
# tools. (i.e. podman)
 | 
			
		||||
CONTAINER_TOOL ?= docker
 | 
			
		||||
 | 
			
		||||
# Setting SHELL to bash allows bash commands to be executed by recipes.
 | 
			
		||||
# Options are set to exit when a recipe line exits non-zero or a piped command fails.
 | 
			
		||||
SHELL = /usr/bin/env bash -o pipefail
 | 
			
		||||
.SHELLFLAGS = -ec
 | 
			
		||||
 | 
			
		||||
.PHONY: all
 | 
			
		||||
all: build
 | 
			
		||||
 | 
			
		||||
##@ General
 | 
			
		||||
 | 
			
		||||
# The help target prints out all targets with their descriptions organized
 | 
			
		||||
# beneath their categories. The categories are represented by '##@' and the
 | 
			
		||||
# target descriptions by '##'. The awk command is responsible for reading the
 | 
			
		||||
# entire set of makefiles included in this invocation, looking for lines of the
 | 
			
		||||
# file as xyz: ## something, and then pretty-format the target and help. Then,
 | 
			
		||||
# if there's a line with ##@ something, that gets pretty-printed as a category.
 | 
			
		||||
# More info on the usage of ANSI control characters for terminal formatting:
 | 
			
		||||
# https://en.wikipedia.org/wiki/ANSI_escape_code#SGR_parameters
 | 
			
		||||
# More info on the awk command:
 | 
			
		||||
# http://linuxcommand.org/lc3_adv_awk.php
 | 
			
		||||
 | 
			
		||||
.PHONY: help
 | 
			
		||||
help: ## Display this help.
 | 
			
		||||
	@awk 'BEGIN {FS = ":.*##"; printf "\nUsage:\n  make \033[36m<target>\033[0m\n"} /^[a-zA-Z_0-9-]+:.*?##/ { printf "  \033[36m%-15s\033[0m %s\n", $$1, $$2 } /^##@/ { printf "\n\033[1m%s\033[0m\n", substr($$0, 5) } ' $(MAKEFILE_LIST)
 | 
			
		||||
 | 
			
		||||
##@ Development
 | 
			
		||||
 | 
			
		||||
.PHONY: manifests
 | 
			
		||||
manifests: controller-gen ## Generate WebhookConfiguration, ClusterRole and CustomResourceDefinition objects.
 | 
			
		||||
	$(CONTROLLER_GEN) rbac:roleName=manager-role crd webhook paths="./..." output:crd:artifacts:config=config/crd/bases
 | 
			
		||||
 | 
			
		||||
.PHONY: generate
 | 
			
		||||
generate: controller-gen ## Generate code containing DeepCopy, DeepCopyInto, and DeepCopyObject method implementations.
 | 
			
		||||
	$(CONTROLLER_GEN) object:headerFile="hack/boilerplate.go.txt" paths="./..."
 | 
			
		||||
 | 
			
		||||
.PHONY: fmt
 | 
			
		||||
fmt: ## Run go fmt against code.
 | 
			
		||||
	go fmt ./...
 | 
			
		||||
 | 
			
		||||
.PHONY: vet
 | 
			
		||||
vet: ## Run go vet against code.
 | 
			
		||||
	go vet ./...
 | 
			
		||||
 | 
			
		||||
.PHONY: test
 | 
			
		||||
test: manifests generate fmt vet setup-envtest ## Run tests.
 | 
			
		||||
	KUBEBUILDER_ASSETS="$(shell $(ENVTEST) use $(ENVTEST_K8S_VERSION) --bin-dir $(LOCALBIN) -p path)" go test $$(go list ./... | grep -v /e2e) -coverprofile cover.out
 | 
			
		||||
 | 
			
		||||
# TODO(user): To use a different vendor for e2e tests, modify the setup under 'tests/e2e'.
 | 
			
		||||
# The default setup assumes Kind is pre-installed and builds/loads the Manager Docker image locally.
 | 
			
		||||
# CertManager is installed by default; skip with:
 | 
			
		||||
# - CERT_MANAGER_INSTALL_SKIP=true
 | 
			
		||||
KIND_CLUSTER ?= yaho-test-e2e
 | 
			
		||||
 | 
			
		||||
.PHONY: setup-test-e2e
 | 
			
		||||
setup-test-e2e: ## Set up a Kind cluster for e2e tests if it does not exist
 | 
			
		||||
	@command -v $(KIND) >/dev/null 2>&1 || { \
 | 
			
		||||
		echo "Kind is not installed. Please install Kind manually."; \
 | 
			
		||||
		exit 1; \
 | 
			
		||||
	}
 | 
			
		||||
	$(KIND) create cluster --name $(KIND_CLUSTER)
 | 
			
		||||
 | 
			
		||||
.PHONY: test-e2e
 | 
			
		||||
test-e2e: setup-test-e2e manifests generate fmt vet ## Run the e2e tests. Expected an isolated environment using Kind.
 | 
			
		||||
	KIND_CLUSTER=$(KIND_CLUSTER) go test ./test/e2e/ -v -ginkgo.v
 | 
			
		||||
	$(MAKE) cleanup-test-e2e
 | 
			
		||||
 | 
			
		||||
.PHONY: cleanup-test-e2e
 | 
			
		||||
cleanup-test-e2e: ## Tear down the Kind cluster used for e2e tests
 | 
			
		||||
	@$(KIND) delete cluster --name $(KIND_CLUSTER)
 | 
			
		||||
 | 
			
		||||
.PHONY: lint
 | 
			
		||||
lint: golangci-lint ## Run golangci-lint linter
 | 
			
		||||
	$(GOLANGCI_LINT) run
 | 
			
		||||
 | 
			
		||||
.PHONY: lint-fix
 | 
			
		||||
lint-fix: golangci-lint ## Run golangci-lint linter and perform fixes
 | 
			
		||||
	$(GOLANGCI_LINT) run --fix
 | 
			
		||||
 | 
			
		||||
.PHONY: lint-config
 | 
			
		||||
lint-config: golangci-lint ## Verify golangci-lint linter configuration
 | 
			
		||||
	$(GOLANGCI_LINT) config verify
 | 
			
		||||
 | 
			
		||||
##@ Build
 | 
			
		||||
 | 
			
		||||
.PHONY: build
 | 
			
		||||
build: manifests generate fmt vet ## Build manager binary.
 | 
			
		||||
	go build -o bin/manager cmd/main.go
 | 
			
		||||
 | 
			
		||||
.PHONY: run
 | 
			
		||||
run: manifests generate fmt vet ## Run a controller from your host.
 | 
			
		||||
	go run ./cmd/main.go
 | 
			
		||||
 | 
			
		||||
# If you wish to build the manager image targeting other platforms you can use the --platform flag.
 | 
			
		||||
# (i.e. docker build --platform linux/arm64). However, you must enable docker buildKit for it.
 | 
			
		||||
# More info: https://docs.docker.com/develop/develop-images/build_enhancements/
 | 
			
		||||
.PHONY: docker-build
 | 
			
		||||
docker-build: ## Build docker image with the manager.
 | 
			
		||||
	$(CONTAINER_TOOL) build -t ${IMG} .
 | 
			
		||||
 | 
			
		||||
.PHONY: docker-push
 | 
			
		||||
docker-push: ## Push docker image with the manager.
 | 
			
		||||
	$(CONTAINER_TOOL) push ${IMG}
 | 
			
		||||
 | 
			
		||||
# PLATFORMS defines the target platforms for the manager image be built to provide support to multiple
 | 
			
		||||
# architectures. (i.e. make docker-buildx IMG=myregistry/mypoperator:0.0.1). To use this option you need to:
 | 
			
		||||
# - be able to use docker buildx. More info: https://docs.docker.com/build/buildx/
 | 
			
		||||
# - have enabled BuildKit. More info: https://docs.docker.com/develop/develop-images/build_enhancements/
 | 
			
		||||
# - be able to push the image to your registry (i.e. if you do not set a valid value via IMG=<myregistry/image:<tag>> then the export will fail)
 | 
			
		||||
# To adequately provide solutions that are compatible with multiple platforms, you should consider using this option.
 | 
			
		||||
PLATFORMS ?= linux/arm64,linux/amd64,linux/s390x,linux/ppc64le
 | 
			
		||||
.PHONY: docker-buildx
 | 
			
		||||
docker-buildx: ## Build and push docker image for the manager for cross-platform support
 | 
			
		||||
	# copy existing Dockerfile and insert --platform=${BUILDPLATFORM} into Dockerfile.cross, and preserve the original Dockerfile
 | 
			
		||||
	sed -e '1 s/\(^FROM\)/FROM --platform=\$$\{BUILDPLATFORM\}/; t' -e ' 1,// s//FROM --platform=\$$\{BUILDPLATFORM\}/' Dockerfile > Dockerfile.cross
 | 
			
		||||
	- $(CONTAINER_TOOL) buildx create --name yaho-builder
 | 
			
		||||
	$(CONTAINER_TOOL) buildx use yaho-builder
 | 
			
		||||
	- $(CONTAINER_TOOL) buildx build --push --platform=$(PLATFORMS) --tag ${IMG} -f Dockerfile.cross .
 | 
			
		||||
	- $(CONTAINER_TOOL) buildx rm yaho-builder
 | 
			
		||||
	rm Dockerfile.cross
 | 
			
		||||
 | 
			
		||||
.PHONY: build-installer
 | 
			
		||||
build-installer: manifests generate kustomize ## Generate a consolidated YAML with CRDs and deployment.
 | 
			
		||||
	mkdir -p dist
 | 
			
		||||
	cd config/manager && $(KUSTOMIZE) edit set image controller=${IMG}
 | 
			
		||||
	$(KUSTOMIZE) build config/default > dist/install.yaml
 | 
			
		||||
 | 
			
		||||
##@ Deployment
 | 
			
		||||
 | 
			
		||||
ifndef ignore-not-found
 | 
			
		||||
  ignore-not-found = false
 | 
			
		||||
endif
 | 
			
		||||
 | 
			
		||||
.PHONY: install
 | 
			
		||||
install: manifests kustomize ## Install CRDs into the K8s cluster specified in ~/.kube/config.
 | 
			
		||||
	$(KUSTOMIZE) build config/crd | $(KUBECTL) apply -f -
 | 
			
		||||
 | 
			
		||||
.PHONY: uninstall
 | 
			
		||||
uninstall: manifests kustomize ## Uninstall CRDs from the K8s cluster specified in ~/.kube/config. Call with ignore-not-found=true to ignore resource not found errors during deletion.
 | 
			
		||||
	$(KUSTOMIZE) build config/crd | $(KUBECTL) delete --ignore-not-found=$(ignore-not-found) -f -
 | 
			
		||||
 | 
			
		||||
.PHONY: deploy
 | 
			
		||||
deploy: manifests kustomize ## Deploy controller to the K8s cluster specified in ~/.kube/config.
 | 
			
		||||
	cd config/manager && $(KUSTOMIZE) edit set image controller=${IMG}
 | 
			
		||||
	$(KUSTOMIZE) build config/default | $(KUBECTL) apply -f -
 | 
			
		||||
 | 
			
		||||
.PHONY: undeploy
 | 
			
		||||
undeploy: kustomize ## Undeploy controller from the K8s cluster specified in ~/.kube/config. Call with ignore-not-found=true to ignore resource not found errors during deletion.
 | 
			
		||||
	$(KUSTOMIZE) build config/default | $(KUBECTL) delete --ignore-not-found=$(ignore-not-found) -f -
 | 
			
		||||
 | 
			
		||||
##@ Dependencies
 | 
			
		||||
 | 
			
		||||
## Location to install dependencies to
 | 
			
		||||
LOCALBIN ?= $(shell pwd)/bin
 | 
			
		||||
$(LOCALBIN):
 | 
			
		||||
	mkdir -p $(LOCALBIN)
 | 
			
		||||
 | 
			
		||||
## Tool Binaries
 | 
			
		||||
KUBECTL ?= kubectl
 | 
			
		||||
KIND ?= kind
 | 
			
		||||
KUSTOMIZE ?= $(LOCALBIN)/kustomize
 | 
			
		||||
CONTROLLER_GEN ?= $(LOCALBIN)/controller-gen
 | 
			
		||||
ENVTEST ?= $(LOCALBIN)/setup-envtest
 | 
			
		||||
GOLANGCI_LINT = $(LOCALBIN)/golangci-lint
 | 
			
		||||
 | 
			
		||||
## Tool Versions
 | 
			
		||||
KUSTOMIZE_VERSION ?= v5.6.0
 | 
			
		||||
CONTROLLER_TOOLS_VERSION ?= v0.18.0
 | 
			
		||||
#ENVTEST_VERSION is the version of controller-runtime release branch to fetch the envtest setup script (i.e. release-0.20)
 | 
			
		||||
ENVTEST_VERSION ?= $(shell go list -m -f "{{ .Version }}" sigs.k8s.io/controller-runtime | awk -F'[v.]' '{printf "release-%d.%d", $$2, $$3}')
 | 
			
		||||
#ENVTEST_K8S_VERSION is the version of Kubernetes to use for setting up ENVTEST binaries (i.e. 1.31)
 | 
			
		||||
ENVTEST_K8S_VERSION ?= $(shell go list -m -f "{{ .Version }}" k8s.io/api | awk -F'[v.]' '{printf "1.%d", $$3}')
 | 
			
		||||
GOLANGCI_LINT_VERSION ?= v2.1.0
 | 
			
		||||
 | 
			
		||||
.PHONY: kustomize
 | 
			
		||||
kustomize: $(KUSTOMIZE) ## Download kustomize locally if necessary.
 | 
			
		||||
$(KUSTOMIZE): $(LOCALBIN)
 | 
			
		||||
	$(call go-install-tool,$(KUSTOMIZE),sigs.k8s.io/kustomize/kustomize/v5,$(KUSTOMIZE_VERSION))
 | 
			
		||||
 | 
			
		||||
.PHONY: controller-gen
 | 
			
		||||
controller-gen: $(CONTROLLER_GEN) ## Download controller-gen locally if necessary.
 | 
			
		||||
$(CONTROLLER_GEN): $(LOCALBIN)
 | 
			
		||||
	$(call go-install-tool,$(CONTROLLER_GEN),sigs.k8s.io/controller-tools/cmd/controller-gen,$(CONTROLLER_TOOLS_VERSION))
 | 
			
		||||
 | 
			
		||||
.PHONY: setup-envtest
 | 
			
		||||
setup-envtest: envtest ## Download the binaries required for ENVTEST in the local bin directory.
 | 
			
		||||
	@echo "Setting up envtest binaries for Kubernetes version $(ENVTEST_K8S_VERSION)..."
 | 
			
		||||
	@$(ENVTEST) use $(ENVTEST_K8S_VERSION) --bin-dir $(LOCALBIN) -p path || { \
 | 
			
		||||
		echo "Error: Failed to set up envtest binaries for version $(ENVTEST_K8S_VERSION)."; \
 | 
			
		||||
		exit 1; \
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
.PHONY: envtest
 | 
			
		||||
envtest: $(ENVTEST) ## Download setup-envtest locally if necessary.
 | 
			
		||||
$(ENVTEST): $(LOCALBIN)
 | 
			
		||||
	$(call go-install-tool,$(ENVTEST),sigs.k8s.io/controller-runtime/tools/setup-envtest,$(ENVTEST_VERSION))
 | 
			
		||||
 | 
			
		||||
.PHONY: golangci-lint
 | 
			
		||||
golangci-lint: $(GOLANGCI_LINT) ## Download golangci-lint locally if necessary.
 | 
			
		||||
$(GOLANGCI_LINT): $(LOCALBIN)
 | 
			
		||||
	$(call go-install-tool,$(GOLANGCI_LINT),github.com/golangci/golangci-lint/v2/cmd/golangci-lint,$(GOLANGCI_LINT_VERSION))
 | 
			
		||||
 | 
			
		||||
# go-install-tool will 'go install' any package with custom target and name of binary, if it doesn't exist
 | 
			
		||||
# $1 - target path with name of binary
 | 
			
		||||
# $2 - package url which can be installed
 | 
			
		||||
# $3 - specific version of package
 | 
			
		||||
define go-install-tool
 | 
			
		||||
@[ -f "$(1)-$(3)" ] || { \
 | 
			
		||||
set -e; \
 | 
			
		||||
package=$(2)@$(3) ;\
 | 
			
		||||
echo "Downloading $${package}" ;\
 | 
			
		||||
rm -f $(1) || true ;\
 | 
			
		||||
GOBIN=$(LOCALBIN) go install $${package} ;\
 | 
			
		||||
mv $(1) $(1)-$(3) ;\
 | 
			
		||||
} ;\
 | 
			
		||||
ln -sf $(1)-$(3) $(1)
 | 
			
		||||
endef
 | 
			
		||||
							
								
								
									
										39
									
								
								PROJECT
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								PROJECT
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
			
		||||
# Code generated by tool. DO NOT EDIT.
 | 
			
		||||
# This file is used to track the info used to scaffold your project
 | 
			
		||||
# and allow the plugins properly work.
 | 
			
		||||
# More info: https://book.kubebuilder.io/reference/project-config.html
 | 
			
		||||
cliVersion: 4.6.0
 | 
			
		||||
domain: badhouseplants.net
 | 
			
		||||
layout:
 | 
			
		||||
- go.kubebuilder.io/v4
 | 
			
		||||
projectName: yaho
 | 
			
		||||
repo: github.com/allanger/yaho
 | 
			
		||||
resources:
 | 
			
		||||
- api:
 | 
			
		||||
    crdVersion: v1
 | 
			
		||||
    namespaced: true
 | 
			
		||||
  controller: true
 | 
			
		||||
  domain: badhouseplants.net
 | 
			
		||||
  group: yaho
 | 
			
		||||
  kind: HelmRelease
 | 
			
		||||
  path: github.com/allanger/yaho/api/v1alpha1
 | 
			
		||||
  version: v1alpha1
 | 
			
		||||
- api:
 | 
			
		||||
    crdVersion: v1
 | 
			
		||||
    namespaced: true
 | 
			
		||||
  controller: true
 | 
			
		||||
  domain: badhouseplants.net
 | 
			
		||||
  group: yaho
 | 
			
		||||
  kind: HelmDiff
 | 
			
		||||
  path: github.com/allanger/yaho/api/v1alpha1
 | 
			
		||||
  version: v1alpha1
 | 
			
		||||
- api:
 | 
			
		||||
    crdVersion: v1
 | 
			
		||||
    namespaced: true
 | 
			
		||||
  controller: true
 | 
			
		||||
  domain: badhouseplants.net
 | 
			
		||||
  group: yaho
 | 
			
		||||
  kind: HelmValues
 | 
			
		||||
  path: github.com/allanger/yaho/api/v1alpha1
 | 
			
		||||
  version: v1alpha1
 | 
			
		||||
version: "3"
 | 
			
		||||
							
								
								
									
										36
									
								
								api/v1alpha1/groupversion_info.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								api/v1alpha1/groupversion_info.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,36 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
// Package v1alpha1 contains API Schema definitions for the yaho v1alpha1 API group.
 | 
			
		||||
// +kubebuilder:object:generate=true
 | 
			
		||||
// +groupName=yaho.badhouseplants.net
 | 
			
		||||
package v1alpha1
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"k8s.io/apimachinery/pkg/runtime/schema"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/scheme"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var (
 | 
			
		||||
	// GroupVersion is group version used to register these objects.
 | 
			
		||||
	GroupVersion = schema.GroupVersion{Group: "yaho.badhouseplants.net", Version: "v1alpha1"}
 | 
			
		||||
 | 
			
		||||
	// SchemeBuilder is used to add go types to the GroupVersionKind scheme.
 | 
			
		||||
	SchemeBuilder = &scheme.Builder{GroupVersion: GroupVersion}
 | 
			
		||||
 | 
			
		||||
	// AddToScheme adds the types in this group-version to the given scheme.
 | 
			
		||||
	AddToScheme = SchemeBuilder.AddToScheme
 | 
			
		||||
)
 | 
			
		||||
							
								
								
									
										64
									
								
								api/v1alpha1/helmdiff_types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								api/v1alpha1/helmdiff_types.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,64 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package v1alpha1
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// EDIT THIS FILE!  THIS IS SCAFFOLDING FOR YOU TO OWN!
 | 
			
		||||
// NOTE: json tags are required.  Any new fields you add must have json tags for the fields to be serialized.
 | 
			
		||||
 | 
			
		||||
// HelmDiffSpec defines the desired state of HelmDiff.
 | 
			
		||||
type HelmDiffSpec struct {
 | 
			
		||||
	// INSERT ADDITIONAL SPEC FIELDS - desired state of cluster
 | 
			
		||||
	// Important: Run "make" to regenerate code after modifying this file
 | 
			
		||||
 | 
			
		||||
	// Foo is an example field of HelmDiff. Edit helmdiff_types.go to remove/update
 | 
			
		||||
	Foo string `json:"foo,omitempty"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// HelmDiffStatus defines the observed state of HelmDiff.
 | 
			
		||||
type HelmDiffStatus struct {
 | 
			
		||||
	// INSERT ADDITIONAL STATUS FIELD - define observed state of cluster
 | 
			
		||||
	// Important: Run "make" to regenerate code after modifying this file
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// +kubebuilder:object:root=true
 | 
			
		||||
// +kubebuilder:subresource:status
 | 
			
		||||
 | 
			
		||||
// HelmDiff is the Schema for the helmdiffs API.
 | 
			
		||||
type HelmDiff struct {
 | 
			
		||||
	metav1.TypeMeta   `json:",inline"`
 | 
			
		||||
	metav1.ObjectMeta `json:"metadata,omitempty"`
 | 
			
		||||
 | 
			
		||||
	Spec   HelmDiffSpec   `json:"spec,omitempty"`
 | 
			
		||||
	Status HelmDiffStatus `json:"status,omitempty"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// +kubebuilder:object:root=true
 | 
			
		||||
 | 
			
		||||
// HelmDiffList contains a list of HelmDiff.
 | 
			
		||||
type HelmDiffList struct {
 | 
			
		||||
	metav1.TypeMeta `json:",inline"`
 | 
			
		||||
	metav1.ListMeta `json:"metadata,omitempty"`
 | 
			
		||||
	Items           []HelmDiff `json:"items"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	SchemeBuilder.Register(&HelmDiff{}, &HelmDiffList{})
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										58
									
								
								api/v1alpha1/helmrelease_types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								api/v1alpha1/helmrelease_types.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,58 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package v1alpha1
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// HelmReleaseSpec defines the desired state of HelmRelease.
 | 
			
		||||
type HelmReleaseSpec struct {
 | 
			
		||||
	Repository string `json:"repository"`
 | 
			
		||||
	Chart string `json:"chart"`
 | 
			
		||||
	Version string `json:"version"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// HelmReleaseStatus defines the observed state of HelmRelease.
 | 
			
		||||
type HelmReleaseStatus struct {
 | 
			
		||||
	Values []string `json:"values,omitempty"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// +kubebuilder:object:root=true
 | 
			
		||||
// +kubebuilder:subresource:status
 | 
			
		||||
 | 
			
		||||
// HelmRelease is the Schema for the helmreleases API.
 | 
			
		||||
type HelmRelease struct {
 | 
			
		||||
	metav1.TypeMeta   `json:",inline"`
 | 
			
		||||
	metav1.ObjectMeta `json:"metadata,omitempty"`
 | 
			
		||||
 | 
			
		||||
	Spec   HelmReleaseSpec   `json:"spec,omitempty"`
 | 
			
		||||
	Status HelmReleaseStatus `json:"status,omitempty"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// +kubebuilder:object:root=true
 | 
			
		||||
 | 
			
		||||
// HelmReleaseList contains a list of HelmRelease.
 | 
			
		||||
type HelmReleaseList struct {
 | 
			
		||||
	metav1.TypeMeta `json:",inline"`
 | 
			
		||||
	metav1.ListMeta `json:"metadata,omitempty"`
 | 
			
		||||
	Items           []HelmRelease `json:"items"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	SchemeBuilder.Register(&HelmRelease{}, &HelmReleaseList{})
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										66
									
								
								api/v1alpha1/helmvalues_types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								api/v1alpha1/helmvalues_types.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,66 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package v1alpha1
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// EDIT THIS FILE!  THIS IS SCAFFOLDING FOR YOU TO OWN!
 | 
			
		||||
// NOTE: json tags are required.  Any new fields you add must have json tags for the fields to be serialized.
 | 
			
		||||
 | 
			
		||||
// HelmValuesSpec defines the desired state of HelmValues.
 | 
			
		||||
type HelmValuesSpec struct {
 | 
			
		||||
	// INSERT ADDITIONAL SPEC FIELDS - desired state of cluster
 | 
			
		||||
	// Important: Run "make" to regenerate code after modifying this file
 | 
			
		||||
 | 
			
		||||
	// Foo is an example field of HelmValues. Edit helmvalues_types.go to remove/update
 | 
			
		||||
	Foo string `json:"foo,omitempty"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// HelmValuesStatus defines the observed state of HelmValues.
 | 
			
		||||
type HelmValuesStatus struct {
 | 
			
		||||
	// INSERT ADDITIONAL STATUS FIELD - define observed state of cluster
 | 
			
		||||
	// Important: Run "make" to regenerate code after modifying this file
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// +kubebuilder:object:root=true
 | 
			
		||||
// +kubebuilder:subresource:status
 | 
			
		||||
 | 
			
		||||
// HelmValues is the Schema for the helmvalues API.
 | 
			
		||||
type HelmValues struct {
 | 
			
		||||
	metav1.TypeMeta   `json:",inline"`
 | 
			
		||||
	metav1.ObjectMeta `json:"metadata,omitempty"`
 | 
			
		||||
 | 
			
		||||
	Spec   HelmValuesSpec   `json:"spec,omitempty"`
 | 
			
		||||
	Data   HelmValuesData   `json:"data,omitempty"`
 | 
			
		||||
	Status HelmValuesStatus `json:"status,omitempty"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type HelmValuesData struct {}
 | 
			
		||||
// +kubebuilder:object:root=true
 | 
			
		||||
 | 
			
		||||
// HelmValuesList contains a list of HelmValues.
 | 
			
		||||
type HelmValuesList struct {
 | 
			
		||||
	metav1.TypeMeta `json:",inline"`
 | 
			
		||||
	metav1.ListMeta `json:"metadata,omitempty"`
 | 
			
		||||
	Items           []HelmValues `json:"items"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	SchemeBuilder.Register(&HelmValues{}, &HelmValuesList{})
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										313
									
								
								api/v1alpha1/zz_generated.deepcopy.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										313
									
								
								api/v1alpha1/zz_generated.deepcopy.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,313 @@
 | 
			
		||||
//go:build !ignore_autogenerated
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
// Code generated by controller-gen. DO NOT EDIT.
 | 
			
		||||
 | 
			
		||||
package v1alpha1
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	runtime "k8s.io/apimachinery/pkg/runtime"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmDiff) DeepCopyInto(out *HelmDiff) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
	out.TypeMeta = in.TypeMeta
 | 
			
		||||
	in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
 | 
			
		||||
	out.Spec = in.Spec
 | 
			
		||||
	out.Status = in.Status
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmDiff.
 | 
			
		||||
func (in *HelmDiff) DeepCopy() *HelmDiff {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmDiff)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
 | 
			
		||||
func (in *HelmDiff) DeepCopyObject() runtime.Object {
 | 
			
		||||
	if c := in.DeepCopy(); c != nil {
 | 
			
		||||
		return c
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmDiffList) DeepCopyInto(out *HelmDiffList) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
	out.TypeMeta = in.TypeMeta
 | 
			
		||||
	in.ListMeta.DeepCopyInto(&out.ListMeta)
 | 
			
		||||
	if in.Items != nil {
 | 
			
		||||
		in, out := &in.Items, &out.Items
 | 
			
		||||
		*out = make([]HelmDiff, len(*in))
 | 
			
		||||
		for i := range *in {
 | 
			
		||||
			(*in)[i].DeepCopyInto(&(*out)[i])
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmDiffList.
 | 
			
		||||
func (in *HelmDiffList) DeepCopy() *HelmDiffList {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmDiffList)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
 | 
			
		||||
func (in *HelmDiffList) DeepCopyObject() runtime.Object {
 | 
			
		||||
	if c := in.DeepCopy(); c != nil {
 | 
			
		||||
		return c
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmDiffSpec) DeepCopyInto(out *HelmDiffSpec) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmDiffSpec.
 | 
			
		||||
func (in *HelmDiffSpec) DeepCopy() *HelmDiffSpec {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmDiffSpec)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmDiffStatus) DeepCopyInto(out *HelmDiffStatus) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmDiffStatus.
 | 
			
		||||
func (in *HelmDiffStatus) DeepCopy() *HelmDiffStatus {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmDiffStatus)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmRelease) DeepCopyInto(out *HelmRelease) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
	out.TypeMeta = in.TypeMeta
 | 
			
		||||
	in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
 | 
			
		||||
	out.Spec = in.Spec
 | 
			
		||||
	in.Status.DeepCopyInto(&out.Status)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmRelease.
 | 
			
		||||
func (in *HelmRelease) DeepCopy() *HelmRelease {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmRelease)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
 | 
			
		||||
func (in *HelmRelease) DeepCopyObject() runtime.Object {
 | 
			
		||||
	if c := in.DeepCopy(); c != nil {
 | 
			
		||||
		return c
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmReleaseList) DeepCopyInto(out *HelmReleaseList) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
	out.TypeMeta = in.TypeMeta
 | 
			
		||||
	in.ListMeta.DeepCopyInto(&out.ListMeta)
 | 
			
		||||
	if in.Items != nil {
 | 
			
		||||
		in, out := &in.Items, &out.Items
 | 
			
		||||
		*out = make([]HelmRelease, len(*in))
 | 
			
		||||
		for i := range *in {
 | 
			
		||||
			(*in)[i].DeepCopyInto(&(*out)[i])
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmReleaseList.
 | 
			
		||||
func (in *HelmReleaseList) DeepCopy() *HelmReleaseList {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmReleaseList)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
 | 
			
		||||
func (in *HelmReleaseList) DeepCopyObject() runtime.Object {
 | 
			
		||||
	if c := in.DeepCopy(); c != nil {
 | 
			
		||||
		return c
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmReleaseSpec) DeepCopyInto(out *HelmReleaseSpec) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmReleaseSpec.
 | 
			
		||||
func (in *HelmReleaseSpec) DeepCopy() *HelmReleaseSpec {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmReleaseSpec)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmReleaseStatus) DeepCopyInto(out *HelmReleaseStatus) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
	if in.Values != nil {
 | 
			
		||||
		in, out := &in.Values, &out.Values
 | 
			
		||||
		*out = make([]string, len(*in))
 | 
			
		||||
		copy(*out, *in)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmReleaseStatus.
 | 
			
		||||
func (in *HelmReleaseStatus) DeepCopy() *HelmReleaseStatus {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmReleaseStatus)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmValues) DeepCopyInto(out *HelmValues) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
	out.TypeMeta = in.TypeMeta
 | 
			
		||||
	in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
 | 
			
		||||
	out.Spec = in.Spec
 | 
			
		||||
	out.Data = in.Data
 | 
			
		||||
	out.Status = in.Status
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmValues.
 | 
			
		||||
func (in *HelmValues) DeepCopy() *HelmValues {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmValues)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
 | 
			
		||||
func (in *HelmValues) DeepCopyObject() runtime.Object {
 | 
			
		||||
	if c := in.DeepCopy(); c != nil {
 | 
			
		||||
		return c
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmValuesData) DeepCopyInto(out *HelmValuesData) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmValuesData.
 | 
			
		||||
func (in *HelmValuesData) DeepCopy() *HelmValuesData {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmValuesData)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmValuesList) DeepCopyInto(out *HelmValuesList) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
	out.TypeMeta = in.TypeMeta
 | 
			
		||||
	in.ListMeta.DeepCopyInto(&out.ListMeta)
 | 
			
		||||
	if in.Items != nil {
 | 
			
		||||
		in, out := &in.Items, &out.Items
 | 
			
		||||
		*out = make([]HelmValues, len(*in))
 | 
			
		||||
		for i := range *in {
 | 
			
		||||
			(*in)[i].DeepCopyInto(&(*out)[i])
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmValuesList.
 | 
			
		||||
func (in *HelmValuesList) DeepCopy() *HelmValuesList {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmValuesList)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
 | 
			
		||||
func (in *HelmValuesList) DeepCopyObject() runtime.Object {
 | 
			
		||||
	if c := in.DeepCopy(); c != nil {
 | 
			
		||||
		return c
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmValuesSpec) DeepCopyInto(out *HelmValuesSpec) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmValuesSpec.
 | 
			
		||||
func (in *HelmValuesSpec) DeepCopy() *HelmValuesSpec {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmValuesSpec)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | 
			
		||||
func (in *HelmValuesStatus) DeepCopyInto(out *HelmValuesStatus) {
 | 
			
		||||
	*out = *in
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HelmValuesStatus.
 | 
			
		||||
func (in *HelmValuesStatus) DeepCopy() *HelmValuesStatus {
 | 
			
		||||
	if in == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	out := new(HelmValuesStatus)
 | 
			
		||||
	in.DeepCopyInto(out)
 | 
			
		||||
	return out
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										258
									
								
								cmd/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										258
									
								
								cmd/main.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,258 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package main
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"crypto/tls"
 | 
			
		||||
	"flag"
 | 
			
		||||
	"os"
 | 
			
		||||
	"path/filepath"
 | 
			
		||||
 | 
			
		||||
	// Import all Kubernetes client auth plugins (e.g. Azure, GCP, OIDC, etc.)
 | 
			
		||||
	// to ensure that exec-entrypoint and run can make use of them.
 | 
			
		||||
	_ "k8s.io/client-go/plugin/pkg/client/auth"
 | 
			
		||||
 | 
			
		||||
	"k8s.io/apimachinery/pkg/runtime"
 | 
			
		||||
	utilruntime "k8s.io/apimachinery/pkg/util/runtime"
 | 
			
		||||
	clientgoscheme "k8s.io/client-go/kubernetes/scheme"
 | 
			
		||||
	ctrl "sigs.k8s.io/controller-runtime"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/certwatcher"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/healthz"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/log/zap"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/metrics/filters"
 | 
			
		||||
	metricsserver "sigs.k8s.io/controller-runtime/pkg/metrics/server"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/webhook"
 | 
			
		||||
 | 
			
		||||
	yahov1alpha1 "github.com/allanger/yaho/api/v1alpha1"
 | 
			
		||||
	"github.com/allanger/yaho/internal/controller"
 | 
			
		||||
	// +kubebuilder:scaffold:imports
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var (
 | 
			
		||||
	scheme   = runtime.NewScheme()
 | 
			
		||||
	setupLog = ctrl.Log.WithName("setup")
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	utilruntime.Must(clientgoscheme.AddToScheme(scheme))
 | 
			
		||||
 | 
			
		||||
	utilruntime.Must(yahov1alpha1.AddToScheme(scheme))
 | 
			
		||||
	// +kubebuilder:scaffold:scheme
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// nolint:gocyclo
 | 
			
		||||
func main() {
 | 
			
		||||
	var metricsAddr string
 | 
			
		||||
	var metricsCertPath, metricsCertName, metricsCertKey string
 | 
			
		||||
	var webhookCertPath, webhookCertName, webhookCertKey string
 | 
			
		||||
	var enableLeaderElection bool
 | 
			
		||||
	var probeAddr string
 | 
			
		||||
	var secureMetrics bool
 | 
			
		||||
	var enableHTTP2 bool
 | 
			
		||||
	var tlsOpts []func(*tls.Config)
 | 
			
		||||
	flag.StringVar(&metricsAddr, "metrics-bind-address", "0", "The address the metrics endpoint binds to. "+
 | 
			
		||||
		"Use :8443 for HTTPS or :8080 for HTTP, or leave as 0 to disable the metrics service.")
 | 
			
		||||
	flag.StringVar(&probeAddr, "health-probe-bind-address", ":8081", "The address the probe endpoint binds to.")
 | 
			
		||||
	flag.BoolVar(&enableLeaderElection, "leader-elect", false,
 | 
			
		||||
		"Enable leader election for controller manager. "+
 | 
			
		||||
			"Enabling this will ensure there is only one active controller manager.")
 | 
			
		||||
	flag.BoolVar(&secureMetrics, "metrics-secure", true,
 | 
			
		||||
		"If set, the metrics endpoint is served securely via HTTPS. Use --metrics-secure=false to use HTTP instead.")
 | 
			
		||||
	flag.StringVar(&webhookCertPath, "webhook-cert-path", "", "The directory that contains the webhook certificate.")
 | 
			
		||||
	flag.StringVar(&webhookCertName, "webhook-cert-name", "tls.crt", "The name of the webhook certificate file.")
 | 
			
		||||
	flag.StringVar(&webhookCertKey, "webhook-cert-key", "tls.key", "The name of the webhook key file.")
 | 
			
		||||
	flag.StringVar(&metricsCertPath, "metrics-cert-path", "",
 | 
			
		||||
		"The directory that contains the metrics server certificate.")
 | 
			
		||||
	flag.StringVar(&metricsCertName, "metrics-cert-name", "tls.crt", "The name of the metrics server certificate file.")
 | 
			
		||||
	flag.StringVar(&metricsCertKey, "metrics-cert-key", "tls.key", "The name of the metrics server key file.")
 | 
			
		||||
	flag.BoolVar(&enableHTTP2, "enable-http2", false,
 | 
			
		||||
		"If set, HTTP/2 will be enabled for the metrics and webhook servers")
 | 
			
		||||
	opts := zap.Options{
 | 
			
		||||
		Development: true,
 | 
			
		||||
	}
 | 
			
		||||
	opts.BindFlags(flag.CommandLine)
 | 
			
		||||
	flag.Parse()
 | 
			
		||||
 | 
			
		||||
	ctrl.SetLogger(zap.New(zap.UseFlagOptions(&opts)))
 | 
			
		||||
 | 
			
		||||
	// if the enable-http2 flag is false (the default), http/2 should be disabled
 | 
			
		||||
	// due to its vulnerabilities. More specifically, disabling http/2 will
 | 
			
		||||
	// prevent from being vulnerable to the HTTP/2 Stream Cancellation and
 | 
			
		||||
	// Rapid Reset CVEs. For more information see:
 | 
			
		||||
	// - https://github.com/advisories/GHSA-qppj-fm5r-hxr3
 | 
			
		||||
	// - https://github.com/advisories/GHSA-4374-p667-p6c8
 | 
			
		||||
	disableHTTP2 := func(c *tls.Config) {
 | 
			
		||||
		setupLog.Info("disabling http/2")
 | 
			
		||||
		c.NextProtos = []string{"http/1.1"}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if !enableHTTP2 {
 | 
			
		||||
		tlsOpts = append(tlsOpts, disableHTTP2)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create watchers for metrics and webhooks certificates
 | 
			
		||||
	var metricsCertWatcher, webhookCertWatcher *certwatcher.CertWatcher
 | 
			
		||||
 | 
			
		||||
	// Initial webhook TLS options
 | 
			
		||||
	webhookTLSOpts := tlsOpts
 | 
			
		||||
 | 
			
		||||
	if len(webhookCertPath) > 0 {
 | 
			
		||||
		setupLog.Info("Initializing webhook certificate watcher using provided certificates",
 | 
			
		||||
			"webhook-cert-path", webhookCertPath, "webhook-cert-name", webhookCertName, "webhook-cert-key", webhookCertKey)
 | 
			
		||||
 | 
			
		||||
		var err error
 | 
			
		||||
		webhookCertWatcher, err = certwatcher.New(
 | 
			
		||||
			filepath.Join(webhookCertPath, webhookCertName),
 | 
			
		||||
			filepath.Join(webhookCertPath, webhookCertKey),
 | 
			
		||||
		)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			setupLog.Error(err, "Failed to initialize webhook certificate watcher")
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		webhookTLSOpts = append(webhookTLSOpts, func(config *tls.Config) {
 | 
			
		||||
			config.GetCertificate = webhookCertWatcher.GetCertificate
 | 
			
		||||
		})
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	webhookServer := webhook.NewServer(webhook.Options{
 | 
			
		||||
		TLSOpts: webhookTLSOpts,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Metrics endpoint is enabled in 'config/default/kustomization.yaml'. The Metrics options configure the server.
 | 
			
		||||
	// More info:
 | 
			
		||||
	// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.21.0/pkg/metrics/server
 | 
			
		||||
	// - https://book.kubebuilder.io/reference/metrics.html
 | 
			
		||||
	metricsServerOptions := metricsserver.Options{
 | 
			
		||||
		BindAddress:   metricsAddr,
 | 
			
		||||
		SecureServing: secureMetrics,
 | 
			
		||||
		TLSOpts:       tlsOpts,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if secureMetrics {
 | 
			
		||||
		// FilterProvider is used to protect the metrics endpoint with authn/authz.
 | 
			
		||||
		// These configurations ensure that only authorized users and service accounts
 | 
			
		||||
		// can access the metrics endpoint. The RBAC are configured in 'config/rbac/kustomization.yaml'. More info:
 | 
			
		||||
		// https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.21.0/pkg/metrics/filters#WithAuthenticationAndAuthorization
 | 
			
		||||
		metricsServerOptions.FilterProvider = filters.WithAuthenticationAndAuthorization
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// If the certificate is not specified, controller-runtime will automatically
 | 
			
		||||
	// generate self-signed certificates for the metrics server. While convenient for development and testing,
 | 
			
		||||
	// this setup is not recommended for production.
 | 
			
		||||
	//
 | 
			
		||||
	// TODO(user): If you enable certManager, uncomment the following lines:
 | 
			
		||||
	// - [METRICS-WITH-CERTS] at config/default/kustomization.yaml to generate and use certificates
 | 
			
		||||
	// managed by cert-manager for the metrics server.
 | 
			
		||||
	// - [PROMETHEUS-WITH-CERTS] at config/prometheus/kustomization.yaml for TLS certification.
 | 
			
		||||
	if len(metricsCertPath) > 0 {
 | 
			
		||||
		setupLog.Info("Initializing metrics certificate watcher using provided certificates",
 | 
			
		||||
			"metrics-cert-path", metricsCertPath, "metrics-cert-name", metricsCertName, "metrics-cert-key", metricsCertKey)
 | 
			
		||||
 | 
			
		||||
		var err error
 | 
			
		||||
		metricsCertWatcher, err = certwatcher.New(
 | 
			
		||||
			filepath.Join(metricsCertPath, metricsCertName),
 | 
			
		||||
			filepath.Join(metricsCertPath, metricsCertKey),
 | 
			
		||||
		)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			setupLog.Error(err, "to initialize metrics certificate watcher", "error", err)
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		metricsServerOptions.TLSOpts = append(metricsServerOptions.TLSOpts, func(config *tls.Config) {
 | 
			
		||||
			config.GetCertificate = metricsCertWatcher.GetCertificate
 | 
			
		||||
		})
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	mgr, err := ctrl.NewManager(ctrl.GetConfigOrDie(), ctrl.Options{
 | 
			
		||||
		Scheme:                 scheme,
 | 
			
		||||
		Metrics:                metricsServerOptions,
 | 
			
		||||
		WebhookServer:          webhookServer,
 | 
			
		||||
		HealthProbeBindAddress: probeAddr,
 | 
			
		||||
		LeaderElection:         enableLeaderElection,
 | 
			
		||||
		LeaderElectionID:       "56aa4792.badhouseplants.net",
 | 
			
		||||
		// LeaderElectionReleaseOnCancel defines if the leader should step down voluntarily
 | 
			
		||||
		// when the Manager ends. This requires the binary to immediately end when the
 | 
			
		||||
		// Manager is stopped, otherwise, this setting is unsafe. Setting this significantly
 | 
			
		||||
		// speeds up voluntary leader transitions as the new leader don't have to wait
 | 
			
		||||
		// LeaseDuration time first.
 | 
			
		||||
		//
 | 
			
		||||
		// In the default scaffold provided, the program ends immediately after
 | 
			
		||||
		// the manager stops, so would be fine to enable this option. However,
 | 
			
		||||
		// if you are doing or is intended to do any operation such as perform cleanups
 | 
			
		||||
		// after the manager stops then its usage might be unsafe.
 | 
			
		||||
		// LeaderElectionReleaseOnCancel: true,
 | 
			
		||||
	})
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		setupLog.Error(err, "unable to start manager")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if err := (&controller.HelmReleaseReconciler{
 | 
			
		||||
		Client: mgr.GetClient(),
 | 
			
		||||
		Scheme: mgr.GetScheme(),
 | 
			
		||||
	}).SetupWithManager(mgr); err != nil {
 | 
			
		||||
		setupLog.Error(err, "unable to create controller", "controller", "HelmRelease")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
	if err := (&controller.HelmDiffReconciler{
 | 
			
		||||
		Client: mgr.GetClient(),
 | 
			
		||||
		Scheme: mgr.GetScheme(),
 | 
			
		||||
	}).SetupWithManager(mgr); err != nil {
 | 
			
		||||
		setupLog.Error(err, "unable to create controller", "controller", "HelmDiff")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
	if err := (&controller.HelmValuesReconciler{
 | 
			
		||||
		Client: mgr.GetClient(),
 | 
			
		||||
		Scheme: mgr.GetScheme(),
 | 
			
		||||
	}).SetupWithManager(mgr); err != nil {
 | 
			
		||||
		setupLog.Error(err, "unable to create controller", "controller", "HelmValues")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
	// +kubebuilder:scaffold:builder
 | 
			
		||||
 | 
			
		||||
	if metricsCertWatcher != nil {
 | 
			
		||||
		setupLog.Info("Adding metrics certificate watcher to manager")
 | 
			
		||||
		if err := mgr.Add(metricsCertWatcher); err != nil {
 | 
			
		||||
			setupLog.Error(err, "unable to add metrics certificate watcher to manager")
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if webhookCertWatcher != nil {
 | 
			
		||||
		setupLog.Info("Adding webhook certificate watcher to manager")
 | 
			
		||||
		if err := mgr.Add(webhookCertWatcher); err != nil {
 | 
			
		||||
			setupLog.Error(err, "unable to add webhook certificate watcher to manager")
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if err := mgr.AddHealthzCheck("healthz", healthz.Ping); err != nil {
 | 
			
		||||
		setupLog.Error(err, "unable to set up health check")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
	if err := mgr.AddReadyzCheck("readyz", healthz.Ping); err != nil {
 | 
			
		||||
		setupLog.Error(err, "unable to set up ready check")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	setupLog.Info("starting manager")
 | 
			
		||||
	if err := mgr.Start(ctrl.SetupSignalHandler()); err != nil {
 | 
			
		||||
		setupLog.Error(err, "problem running manager")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										54
									
								
								config/crd/bases/yaho.badhouseplants.net_helmdiffs.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								config/crd/bases/yaho.badhouseplants.net_helmdiffs.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,54 @@
 | 
			
		||||
---
 | 
			
		||||
apiVersion: apiextensions.k8s.io/v1
 | 
			
		||||
kind: CustomResourceDefinition
 | 
			
		||||
metadata:
 | 
			
		||||
  annotations:
 | 
			
		||||
    controller-gen.kubebuilder.io/version: v0.18.0
 | 
			
		||||
  name: helmdiffs.yaho.badhouseplants.net
 | 
			
		||||
spec:
 | 
			
		||||
  group: yaho.badhouseplants.net
 | 
			
		||||
  names:
 | 
			
		||||
    kind: HelmDiff
 | 
			
		||||
    listKind: HelmDiffList
 | 
			
		||||
    plural: helmdiffs
 | 
			
		||||
    singular: helmdiff
 | 
			
		||||
  scope: Namespaced
 | 
			
		||||
  versions:
 | 
			
		||||
  - name: v1alpha1
 | 
			
		||||
    schema:
 | 
			
		||||
      openAPIV3Schema:
 | 
			
		||||
        description: HelmDiff is the Schema for the helmdiffs API.
 | 
			
		||||
        properties:
 | 
			
		||||
          apiVersion:
 | 
			
		||||
            description: |-
 | 
			
		||||
              APIVersion defines the versioned schema of this representation of an object.
 | 
			
		||||
              Servers should convert recognized schemas to the latest internal value, and
 | 
			
		||||
              may reject unrecognized values.
 | 
			
		||||
              More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
 | 
			
		||||
            type: string
 | 
			
		||||
          kind:
 | 
			
		||||
            description: |-
 | 
			
		||||
              Kind is a string value representing the REST resource this object represents.
 | 
			
		||||
              Servers may infer this from the endpoint the client submits requests to.
 | 
			
		||||
              Cannot be updated.
 | 
			
		||||
              In CamelCase.
 | 
			
		||||
              More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
 | 
			
		||||
            type: string
 | 
			
		||||
          metadata:
 | 
			
		||||
            type: object
 | 
			
		||||
          spec:
 | 
			
		||||
            description: HelmDiffSpec defines the desired state of HelmDiff.
 | 
			
		||||
            properties:
 | 
			
		||||
              foo:
 | 
			
		||||
                description: Foo is an example field of HelmDiff. Edit helmdiff_types.go
 | 
			
		||||
                  to remove/update
 | 
			
		||||
                type: string
 | 
			
		||||
            type: object
 | 
			
		||||
          status:
 | 
			
		||||
            description: HelmDiffStatus defines the observed state of HelmDiff.
 | 
			
		||||
            type: object
 | 
			
		||||
        type: object
 | 
			
		||||
    served: true
 | 
			
		||||
    storage: true
 | 
			
		||||
    subresources:
 | 
			
		||||
      status: {}
 | 
			
		||||
							
								
								
									
										65
									
								
								config/crd/bases/yaho.badhouseplants.net_helmreleases.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								config/crd/bases/yaho.badhouseplants.net_helmreleases.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,65 @@
 | 
			
		||||
---
 | 
			
		||||
apiVersion: apiextensions.k8s.io/v1
 | 
			
		||||
kind: CustomResourceDefinition
 | 
			
		||||
metadata:
 | 
			
		||||
  annotations:
 | 
			
		||||
    controller-gen.kubebuilder.io/version: v0.18.0
 | 
			
		||||
  name: helmreleases.yaho.badhouseplants.net
 | 
			
		||||
spec:
 | 
			
		||||
  group: yaho.badhouseplants.net
 | 
			
		||||
  names:
 | 
			
		||||
    kind: HelmRelease
 | 
			
		||||
    listKind: HelmReleaseList
 | 
			
		||||
    plural: helmreleases
 | 
			
		||||
    singular: helmrelease
 | 
			
		||||
  scope: Namespaced
 | 
			
		||||
  versions:
 | 
			
		||||
  - name: v1alpha1
 | 
			
		||||
    schema:
 | 
			
		||||
      openAPIV3Schema:
 | 
			
		||||
        description: HelmRelease is the Schema for the helmreleases API.
 | 
			
		||||
        properties:
 | 
			
		||||
          apiVersion:
 | 
			
		||||
            description: |-
 | 
			
		||||
              APIVersion defines the versioned schema of this representation of an object.
 | 
			
		||||
              Servers should convert recognized schemas to the latest internal value, and
 | 
			
		||||
              may reject unrecognized values.
 | 
			
		||||
              More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
 | 
			
		||||
            type: string
 | 
			
		||||
          kind:
 | 
			
		||||
            description: |-
 | 
			
		||||
              Kind is a string value representing the REST resource this object represents.
 | 
			
		||||
              Servers may infer this from the endpoint the client submits requests to.
 | 
			
		||||
              Cannot be updated.
 | 
			
		||||
              In CamelCase.
 | 
			
		||||
              More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
 | 
			
		||||
            type: string
 | 
			
		||||
          metadata:
 | 
			
		||||
            type: object
 | 
			
		||||
          spec:
 | 
			
		||||
            description: HelmReleaseSpec defines the desired state of HelmRelease.
 | 
			
		||||
            properties:
 | 
			
		||||
              chart:
 | 
			
		||||
                type: string
 | 
			
		||||
              repository:
 | 
			
		||||
                type: string
 | 
			
		||||
              version:
 | 
			
		||||
                type: string
 | 
			
		||||
            required:
 | 
			
		||||
            - chart
 | 
			
		||||
            - repository
 | 
			
		||||
            - version
 | 
			
		||||
            type: object
 | 
			
		||||
          status:
 | 
			
		||||
            description: HelmReleaseStatus defines the observed state of HelmRelease.
 | 
			
		||||
            properties:
 | 
			
		||||
              values:
 | 
			
		||||
                items:
 | 
			
		||||
                  type: string
 | 
			
		||||
                type: array
 | 
			
		||||
            type: object
 | 
			
		||||
        type: object
 | 
			
		||||
    served: true
 | 
			
		||||
    storage: true
 | 
			
		||||
    subresources:
 | 
			
		||||
      status: {}
 | 
			
		||||
							
								
								
									
										56
									
								
								config/crd/bases/yaho.badhouseplants.net_helmvalues.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								config/crd/bases/yaho.badhouseplants.net_helmvalues.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,56 @@
 | 
			
		||||
---
 | 
			
		||||
apiVersion: apiextensions.k8s.io/v1
 | 
			
		||||
kind: CustomResourceDefinition
 | 
			
		||||
metadata:
 | 
			
		||||
  annotations:
 | 
			
		||||
    controller-gen.kubebuilder.io/version: v0.18.0
 | 
			
		||||
  name: helmvalues.yaho.badhouseplants.net
 | 
			
		||||
spec:
 | 
			
		||||
  group: yaho.badhouseplants.net
 | 
			
		||||
  names:
 | 
			
		||||
    kind: HelmValues
 | 
			
		||||
    listKind: HelmValuesList
 | 
			
		||||
    plural: helmvalues
 | 
			
		||||
    singular: helmvalues
 | 
			
		||||
  scope: Namespaced
 | 
			
		||||
  versions:
 | 
			
		||||
  - name: v1alpha1
 | 
			
		||||
    schema:
 | 
			
		||||
      openAPIV3Schema:
 | 
			
		||||
        description: HelmValues is the Schema for the helmvalues API.
 | 
			
		||||
        properties:
 | 
			
		||||
          apiVersion:
 | 
			
		||||
            description: |-
 | 
			
		||||
              APIVersion defines the versioned schema of this representation of an object.
 | 
			
		||||
              Servers should convert recognized schemas to the latest internal value, and
 | 
			
		||||
              may reject unrecognized values.
 | 
			
		||||
              More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
 | 
			
		||||
            type: string
 | 
			
		||||
          data:
 | 
			
		||||
            type: object
 | 
			
		||||
          kind:
 | 
			
		||||
            description: |-
 | 
			
		||||
              Kind is a string value representing the REST resource this object represents.
 | 
			
		||||
              Servers may infer this from the endpoint the client submits requests to.
 | 
			
		||||
              Cannot be updated.
 | 
			
		||||
              In CamelCase.
 | 
			
		||||
              More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
 | 
			
		||||
            type: string
 | 
			
		||||
          metadata:
 | 
			
		||||
            type: object
 | 
			
		||||
          spec:
 | 
			
		||||
            description: HelmValuesSpec defines the desired state of HelmValues.
 | 
			
		||||
            properties:
 | 
			
		||||
              foo:
 | 
			
		||||
                description: Foo is an example field of HelmValues. Edit helmvalues_types.go
 | 
			
		||||
                  to remove/update
 | 
			
		||||
                type: string
 | 
			
		||||
            type: object
 | 
			
		||||
          status:
 | 
			
		||||
            description: HelmValuesStatus defines the observed state of HelmValues.
 | 
			
		||||
            type: object
 | 
			
		||||
        type: object
 | 
			
		||||
    served: true
 | 
			
		||||
    storage: true
 | 
			
		||||
    subresources:
 | 
			
		||||
      status: {}
 | 
			
		||||
							
								
								
									
										18
									
								
								config/crd/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								config/crd/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
# This kustomization.yaml is not intended to be run by itself,
 | 
			
		||||
# since it depends on service name and namespace that are out of this kustomize package.
 | 
			
		||||
# It should be run by config/default
 | 
			
		||||
resources:
 | 
			
		||||
- bases/yaho.badhouseplants.net_helmreleases.yaml
 | 
			
		||||
- bases/yaho.badhouseplants.net_helmdiffs.yaml
 | 
			
		||||
- bases/yaho.badhouseplants.net_helmvalues.yaml
 | 
			
		||||
# +kubebuilder:scaffold:crdkustomizeresource
 | 
			
		||||
 | 
			
		||||
patches:
 | 
			
		||||
# [WEBHOOK] To enable webhook, uncomment all the sections with [WEBHOOK] prefix.
 | 
			
		||||
# patches here are for enabling the conversion webhook for each CRD
 | 
			
		||||
# +kubebuilder:scaffold:crdkustomizewebhookpatch
 | 
			
		||||
 | 
			
		||||
# [WEBHOOK] To enable webhook, uncomment the following section
 | 
			
		||||
# the following config is for teaching kustomize how to do kustomization for CRDs.
 | 
			
		||||
#configurations:
 | 
			
		||||
#- kustomizeconfig.yaml
 | 
			
		||||
							
								
								
									
										19
									
								
								config/crd/kustomizeconfig.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								config/crd/kustomizeconfig.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
# This file is for teaching kustomize how to substitute name and namespace reference in CRD
 | 
			
		||||
nameReference:
 | 
			
		||||
- kind: Service
 | 
			
		||||
  version: v1
 | 
			
		||||
  fieldSpecs:
 | 
			
		||||
  - kind: CustomResourceDefinition
 | 
			
		||||
    version: v1
 | 
			
		||||
    group: apiextensions.k8s.io
 | 
			
		||||
    path: spec/conversion/webhook/clientConfig/service/name
 | 
			
		||||
 | 
			
		||||
namespace:
 | 
			
		||||
- kind: CustomResourceDefinition
 | 
			
		||||
  version: v1
 | 
			
		||||
  group: apiextensions.k8s.io
 | 
			
		||||
  path: spec/conversion/webhook/clientConfig/service/namespace
 | 
			
		||||
  create: false
 | 
			
		||||
 | 
			
		||||
varReference:
 | 
			
		||||
- path: metadata/annotations
 | 
			
		||||
							
								
								
									
										30
									
								
								config/default/cert_metrics_manager_patch.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								config/default/cert_metrics_manager_patch.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,30 @@
 | 
			
		||||
# This patch adds the args, volumes, and ports to allow the manager to use the metrics-server certs.
 | 
			
		||||
 | 
			
		||||
# Add the volumeMount for the metrics-server certs
 | 
			
		||||
- op: add
 | 
			
		||||
  path: /spec/template/spec/containers/0/volumeMounts/-
 | 
			
		||||
  value:
 | 
			
		||||
    mountPath: /tmp/k8s-metrics-server/metrics-certs
 | 
			
		||||
    name: metrics-certs
 | 
			
		||||
    readOnly: true
 | 
			
		||||
 | 
			
		||||
# Add the --metrics-cert-path argument for the metrics server
 | 
			
		||||
- op: add
 | 
			
		||||
  path: /spec/template/spec/containers/0/args/-
 | 
			
		||||
  value: --metrics-cert-path=/tmp/k8s-metrics-server/metrics-certs
 | 
			
		||||
 | 
			
		||||
# Add the metrics-server certs volume configuration
 | 
			
		||||
- op: add
 | 
			
		||||
  path: /spec/template/spec/volumes/-
 | 
			
		||||
  value:
 | 
			
		||||
    name: metrics-certs
 | 
			
		||||
    secret:
 | 
			
		||||
      secretName: metrics-server-cert
 | 
			
		||||
      optional: false
 | 
			
		||||
      items:
 | 
			
		||||
        - key: ca.crt
 | 
			
		||||
          path: ca.crt
 | 
			
		||||
        - key: tls.crt
 | 
			
		||||
          path: tls.crt
 | 
			
		||||
        - key: tls.key
 | 
			
		||||
          path: tls.key
 | 
			
		||||
							
								
								
									
										234
									
								
								config/default/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										234
									
								
								config/default/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,234 @@
 | 
			
		||||
# Adds namespace to all resources.
 | 
			
		||||
namespace: yaho-system
 | 
			
		||||
 | 
			
		||||
# Value of this field is prepended to the
 | 
			
		||||
# names of all resources, e.g. a deployment named
 | 
			
		||||
# "wordpress" becomes "alices-wordpress".
 | 
			
		||||
# Note that it should also match with the prefix (text before '-') of the namespace
 | 
			
		||||
# field above.
 | 
			
		||||
namePrefix: yaho-
 | 
			
		||||
 | 
			
		||||
# Labels to add to all resources and selectors.
 | 
			
		||||
#labels:
 | 
			
		||||
#- includeSelectors: true
 | 
			
		||||
#  pairs:
 | 
			
		||||
#    someName: someValue
 | 
			
		||||
 | 
			
		||||
resources:
 | 
			
		||||
- ../crd
 | 
			
		||||
- ../rbac
 | 
			
		||||
- ../manager
 | 
			
		||||
# [WEBHOOK] To enable webhook, uncomment all the sections with [WEBHOOK] prefix including the one in
 | 
			
		||||
# crd/kustomization.yaml
 | 
			
		||||
#- ../webhook
 | 
			
		||||
# [CERTMANAGER] To enable cert-manager, uncomment all sections with 'CERTMANAGER'. 'WEBHOOK' components are required.
 | 
			
		||||
#- ../certmanager
 | 
			
		||||
# [PROMETHEUS] To enable prometheus monitor, uncomment all sections with 'PROMETHEUS'.
 | 
			
		||||
#- ../prometheus
 | 
			
		||||
# [METRICS] Expose the controller manager metrics service.
 | 
			
		||||
- metrics_service.yaml
 | 
			
		||||
# [NETWORK POLICY] Protect the /metrics endpoint and Webhook Server with NetworkPolicy.
 | 
			
		||||
# Only Pod(s) running a namespace labeled with 'metrics: enabled' will be able to gather the metrics.
 | 
			
		||||
# Only CR(s) which requires webhooks and are applied on namespaces labeled with 'webhooks: enabled' will
 | 
			
		||||
# be able to communicate with the Webhook Server.
 | 
			
		||||
#- ../network-policy
 | 
			
		||||
 | 
			
		||||
# Uncomment the patches line if you enable Metrics
 | 
			
		||||
patches:
 | 
			
		||||
# [METRICS] The following patch will enable the metrics endpoint using HTTPS and the port :8443.
 | 
			
		||||
# More info: https://book.kubebuilder.io/reference/metrics
 | 
			
		||||
- path: manager_metrics_patch.yaml
 | 
			
		||||
  target:
 | 
			
		||||
    kind: Deployment
 | 
			
		||||
 | 
			
		||||
# Uncomment the patches line if you enable Metrics and CertManager
 | 
			
		||||
# [METRICS-WITH-CERTS] To enable metrics protected with certManager, uncomment the following line.
 | 
			
		||||
# This patch will protect the metrics with certManager self-signed certs.
 | 
			
		||||
#- path: cert_metrics_manager_patch.yaml
 | 
			
		||||
#  target:
 | 
			
		||||
#    kind: Deployment
 | 
			
		||||
 | 
			
		||||
# [WEBHOOK] To enable webhook, uncomment all the sections with [WEBHOOK] prefix including the one in
 | 
			
		||||
# crd/kustomization.yaml
 | 
			
		||||
#- path: manager_webhook_patch.yaml
 | 
			
		||||
#  target:
 | 
			
		||||
#    kind: Deployment
 | 
			
		||||
 | 
			
		||||
# [CERTMANAGER] To enable cert-manager, uncomment all sections with 'CERTMANAGER' prefix.
 | 
			
		||||
# Uncomment the following replacements to add the cert-manager CA injection annotations
 | 
			
		||||
#replacements:
 | 
			
		||||
# - source: # Uncomment the following block to enable certificates for metrics
 | 
			
		||||
#     kind: Service
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: controller-manager-metrics-service
 | 
			
		||||
#     fieldPath: metadata.name
 | 
			
		||||
#   targets:
 | 
			
		||||
#     - select:
 | 
			
		||||
#         kind: Certificate
 | 
			
		||||
#         group: cert-manager.io
 | 
			
		||||
#         version: v1
 | 
			
		||||
#         name: metrics-certs
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - spec.dnsNames.0
 | 
			
		||||
#         - spec.dnsNames.1
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '.'
 | 
			
		||||
#         index: 0
 | 
			
		||||
#         create: true
 | 
			
		||||
#     - select: # Uncomment the following to set the Service name for TLS config in Prometheus ServiceMonitor
 | 
			
		||||
#         kind: ServiceMonitor
 | 
			
		||||
#         group: monitoring.coreos.com
 | 
			
		||||
#         version: v1
 | 
			
		||||
#         name: controller-manager-metrics-monitor
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - spec.endpoints.0.tlsConfig.serverName
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '.'
 | 
			
		||||
#         index: 0
 | 
			
		||||
#         create: true
 | 
			
		||||
#
 | 
			
		||||
# - source:
 | 
			
		||||
#     kind: Service
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: controller-manager-metrics-service
 | 
			
		||||
#     fieldPath: metadata.namespace
 | 
			
		||||
#   targets:
 | 
			
		||||
#     - select:
 | 
			
		||||
#         kind: Certificate
 | 
			
		||||
#         group: cert-manager.io
 | 
			
		||||
#         version: v1
 | 
			
		||||
#         name: metrics-certs
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - spec.dnsNames.0
 | 
			
		||||
#         - spec.dnsNames.1
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '.'
 | 
			
		||||
#         index: 1
 | 
			
		||||
#         create: true
 | 
			
		||||
#     - select: # Uncomment the following to set the Service namespace for TLS in Prometheus ServiceMonitor
 | 
			
		||||
#         kind: ServiceMonitor
 | 
			
		||||
#         group: monitoring.coreos.com
 | 
			
		||||
#         version: v1
 | 
			
		||||
#         name: controller-manager-metrics-monitor
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - spec.endpoints.0.tlsConfig.serverName
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '.'
 | 
			
		||||
#         index: 1
 | 
			
		||||
#         create: true
 | 
			
		||||
#
 | 
			
		||||
# - source: # Uncomment the following block if you have any webhook
 | 
			
		||||
#     kind: Service
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: webhook-service
 | 
			
		||||
#     fieldPath: .metadata.name # Name of the service
 | 
			
		||||
#   targets:
 | 
			
		||||
#     - select:
 | 
			
		||||
#         kind: Certificate
 | 
			
		||||
#         group: cert-manager.io
 | 
			
		||||
#         version: v1
 | 
			
		||||
#         name: serving-cert
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - .spec.dnsNames.0
 | 
			
		||||
#         - .spec.dnsNames.1
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '.'
 | 
			
		||||
#         index: 0
 | 
			
		||||
#         create: true
 | 
			
		||||
# - source:
 | 
			
		||||
#     kind: Service
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: webhook-service
 | 
			
		||||
#     fieldPath: .metadata.namespace # Namespace of the service
 | 
			
		||||
#   targets:
 | 
			
		||||
#     - select:
 | 
			
		||||
#         kind: Certificate
 | 
			
		||||
#         group: cert-manager.io
 | 
			
		||||
#         version: v1
 | 
			
		||||
#         name: serving-cert
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - .spec.dnsNames.0
 | 
			
		||||
#         - .spec.dnsNames.1
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '.'
 | 
			
		||||
#         index: 1
 | 
			
		||||
#         create: true
 | 
			
		||||
#
 | 
			
		||||
# - source: # Uncomment the following block if you have a ValidatingWebhook (--programmatic-validation)
 | 
			
		||||
#     kind: Certificate
 | 
			
		||||
#     group: cert-manager.io
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: serving-cert # This name should match the one in certificate.yaml
 | 
			
		||||
#     fieldPath: .metadata.namespace # Namespace of the certificate CR
 | 
			
		||||
#   targets:
 | 
			
		||||
#     - select:
 | 
			
		||||
#         kind: ValidatingWebhookConfiguration
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - .metadata.annotations.[cert-manager.io/inject-ca-from]
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '/'
 | 
			
		||||
#         index: 0
 | 
			
		||||
#         create: true
 | 
			
		||||
# - source:
 | 
			
		||||
#     kind: Certificate
 | 
			
		||||
#     group: cert-manager.io
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: serving-cert
 | 
			
		||||
#     fieldPath: .metadata.name
 | 
			
		||||
#   targets:
 | 
			
		||||
#     - select:
 | 
			
		||||
#         kind: ValidatingWebhookConfiguration
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - .metadata.annotations.[cert-manager.io/inject-ca-from]
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '/'
 | 
			
		||||
#         index: 1
 | 
			
		||||
#         create: true
 | 
			
		||||
#
 | 
			
		||||
# - source: # Uncomment the following block if you have a DefaultingWebhook (--defaulting )
 | 
			
		||||
#     kind: Certificate
 | 
			
		||||
#     group: cert-manager.io
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: serving-cert
 | 
			
		||||
#     fieldPath: .metadata.namespace # Namespace of the certificate CR
 | 
			
		||||
#   targets:
 | 
			
		||||
#     - select:
 | 
			
		||||
#         kind: MutatingWebhookConfiguration
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - .metadata.annotations.[cert-manager.io/inject-ca-from]
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '/'
 | 
			
		||||
#         index: 0
 | 
			
		||||
#         create: true
 | 
			
		||||
# - source:
 | 
			
		||||
#     kind: Certificate
 | 
			
		||||
#     group: cert-manager.io
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: serving-cert
 | 
			
		||||
#     fieldPath: .metadata.name
 | 
			
		||||
#   targets:
 | 
			
		||||
#     - select:
 | 
			
		||||
#         kind: MutatingWebhookConfiguration
 | 
			
		||||
#       fieldPaths:
 | 
			
		||||
#         - .metadata.annotations.[cert-manager.io/inject-ca-from]
 | 
			
		||||
#       options:
 | 
			
		||||
#         delimiter: '/'
 | 
			
		||||
#         index: 1
 | 
			
		||||
#         create: true
 | 
			
		||||
#
 | 
			
		||||
# - source: # Uncomment the following block if you have a ConversionWebhook (--conversion)
 | 
			
		||||
#     kind: Certificate
 | 
			
		||||
#     group: cert-manager.io
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: serving-cert
 | 
			
		||||
#     fieldPath: .metadata.namespace # Namespace of the certificate CR
 | 
			
		||||
#   targets: # Do not remove or uncomment the following scaffold marker; required to generate code for target CRD.
 | 
			
		||||
# +kubebuilder:scaffold:crdkustomizecainjectionns
 | 
			
		||||
# - source:
 | 
			
		||||
#     kind: Certificate
 | 
			
		||||
#     group: cert-manager.io
 | 
			
		||||
#     version: v1
 | 
			
		||||
#     name: serving-cert
 | 
			
		||||
#     fieldPath: .metadata.name
 | 
			
		||||
#   targets: # Do not remove or uncomment the following scaffold marker; required to generate code for target CRD.
 | 
			
		||||
# +kubebuilder:scaffold:crdkustomizecainjectionname
 | 
			
		||||
							
								
								
									
										4
									
								
								config/default/manager_metrics_patch.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								config/default/manager_metrics_patch.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
			
		||||
# This patch adds the args to allow exposing the metrics endpoint using HTTPS
 | 
			
		||||
- op: add
 | 
			
		||||
  path: /spec/template/spec/containers/0/args/0
 | 
			
		||||
  value: --metrics-bind-address=:8443
 | 
			
		||||
							
								
								
									
										18
									
								
								config/default/metrics_service.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								config/default/metrics_service.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
apiVersion: v1
 | 
			
		||||
kind: Service
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    control-plane: controller-manager
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: controller-manager-metrics-service
 | 
			
		||||
  namespace: system
 | 
			
		||||
spec:
 | 
			
		||||
  ports:
 | 
			
		||||
  - name: https
 | 
			
		||||
    port: 8443
 | 
			
		||||
    protocol: TCP
 | 
			
		||||
    targetPort: 8443
 | 
			
		||||
  selector:
 | 
			
		||||
    control-plane: controller-manager
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
							
								
								
									
										8
									
								
								config/manager/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								config/manager/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
			
		||||
resources:
 | 
			
		||||
- manager.yaml
 | 
			
		||||
apiVersion: kustomize.config.k8s.io/v1beta1
 | 
			
		||||
kind: Kustomization
 | 
			
		||||
images:
 | 
			
		||||
- name: controller
 | 
			
		||||
  newName: controller
 | 
			
		||||
  newTag: latest
 | 
			
		||||
							
								
								
									
										99
									
								
								config/manager/manager.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								config/manager/manager.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,99 @@
 | 
			
		||||
apiVersion: v1
 | 
			
		||||
kind: Namespace
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    control-plane: controller-manager
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: system
 | 
			
		||||
---
 | 
			
		||||
apiVersion: apps/v1
 | 
			
		||||
kind: Deployment
 | 
			
		||||
metadata:
 | 
			
		||||
  name: controller-manager
 | 
			
		||||
  namespace: system
 | 
			
		||||
  labels:
 | 
			
		||||
    control-plane: controller-manager
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
spec:
 | 
			
		||||
  selector:
 | 
			
		||||
    matchLabels:
 | 
			
		||||
      control-plane: controller-manager
 | 
			
		||||
      app.kubernetes.io/name: yaho
 | 
			
		||||
  replicas: 1
 | 
			
		||||
  template:
 | 
			
		||||
    metadata:
 | 
			
		||||
      annotations:
 | 
			
		||||
        kubectl.kubernetes.io/default-container: manager
 | 
			
		||||
      labels:
 | 
			
		||||
        control-plane: controller-manager
 | 
			
		||||
        app.kubernetes.io/name: yaho
 | 
			
		||||
    spec:
 | 
			
		||||
      # TODO(user): Uncomment the following code to configure the nodeAffinity expression
 | 
			
		||||
      # according to the platforms which are supported by your solution.
 | 
			
		||||
      # It is considered best practice to support multiple architectures. You can
 | 
			
		||||
      # build your manager image using the makefile target docker-buildx.
 | 
			
		||||
      # affinity:
 | 
			
		||||
      #   nodeAffinity:
 | 
			
		||||
      #     requiredDuringSchedulingIgnoredDuringExecution:
 | 
			
		||||
      #       nodeSelectorTerms:
 | 
			
		||||
      #         - matchExpressions:
 | 
			
		||||
      #           - key: kubernetes.io/arch
 | 
			
		||||
      #             operator: In
 | 
			
		||||
      #             values:
 | 
			
		||||
      #               - amd64
 | 
			
		||||
      #               - arm64
 | 
			
		||||
      #               - ppc64le
 | 
			
		||||
      #               - s390x
 | 
			
		||||
      #           - key: kubernetes.io/os
 | 
			
		||||
      #             operator: In
 | 
			
		||||
      #             values:
 | 
			
		||||
      #               - linux
 | 
			
		||||
      securityContext:
 | 
			
		||||
        # Projects are configured by default to adhere to the "restricted" Pod Security Standards.
 | 
			
		||||
        # This ensures that deployments meet the highest security requirements for Kubernetes.
 | 
			
		||||
        # For more details, see: https://kubernetes.io/docs/concepts/security/pod-security-standards/#restricted
 | 
			
		||||
        runAsNonRoot: true
 | 
			
		||||
        seccompProfile:
 | 
			
		||||
          type: RuntimeDefault
 | 
			
		||||
      containers:
 | 
			
		||||
      - command:
 | 
			
		||||
        - /manager
 | 
			
		||||
        args:
 | 
			
		||||
          - --leader-elect
 | 
			
		||||
          - --health-probe-bind-address=:8081
 | 
			
		||||
        image: controller:latest
 | 
			
		||||
        imagePullPolicy: Never
 | 
			
		||||
        name: manager
 | 
			
		||||
        ports: []
 | 
			
		||||
        securityContext:
 | 
			
		||||
          allowPrivilegeEscalation: false
 | 
			
		||||
          capabilities:
 | 
			
		||||
            drop:
 | 
			
		||||
            - "ALL"
 | 
			
		||||
        livenessProbe:
 | 
			
		||||
          httpGet:
 | 
			
		||||
            path: /healthz
 | 
			
		||||
            port: 8081
 | 
			
		||||
          initialDelaySeconds: 15
 | 
			
		||||
          periodSeconds: 20
 | 
			
		||||
        readinessProbe:
 | 
			
		||||
          httpGet:
 | 
			
		||||
            path: /readyz
 | 
			
		||||
            port: 8081
 | 
			
		||||
          initialDelaySeconds: 5
 | 
			
		||||
          periodSeconds: 10
 | 
			
		||||
        # TODO(user): Configure the resources accordingly based on the project requirements.
 | 
			
		||||
        # More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/
 | 
			
		||||
        resources:
 | 
			
		||||
          limits:
 | 
			
		||||
            cpu: 500m
 | 
			
		||||
            memory: 128Mi
 | 
			
		||||
          requests:
 | 
			
		||||
            cpu: 10m
 | 
			
		||||
            memory: 64Mi
 | 
			
		||||
        volumeMounts: []
 | 
			
		||||
      volumes: []
 | 
			
		||||
      serviceAccountName: controller-manager
 | 
			
		||||
      terminationGracePeriodSeconds: 10
 | 
			
		||||
							
								
								
									
										27
									
								
								config/network-policy/allow-metrics-traffic.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								config/network-policy/allow-metrics-traffic.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
# This NetworkPolicy allows ingress traffic
 | 
			
		||||
# with Pods running on namespaces labeled with 'metrics: enabled'. Only Pods on those
 | 
			
		||||
# namespaces are able to gather data from the metrics endpoint.
 | 
			
		||||
apiVersion: networking.k8s.io/v1
 | 
			
		||||
kind: NetworkPolicy
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: allow-metrics-traffic
 | 
			
		||||
  namespace: system
 | 
			
		||||
spec:
 | 
			
		||||
  podSelector:
 | 
			
		||||
    matchLabels:
 | 
			
		||||
      control-plane: controller-manager
 | 
			
		||||
      app.kubernetes.io/name: yaho
 | 
			
		||||
  policyTypes:
 | 
			
		||||
    - Ingress
 | 
			
		||||
  ingress:
 | 
			
		||||
    # This allows ingress traffic from any namespace with the label metrics: enabled
 | 
			
		||||
    - from:
 | 
			
		||||
      - namespaceSelector:
 | 
			
		||||
          matchLabels:
 | 
			
		||||
            metrics: enabled  # Only from namespaces with this label
 | 
			
		||||
      ports:
 | 
			
		||||
        - port: 8443
 | 
			
		||||
          protocol: TCP
 | 
			
		||||
							
								
								
									
										2
									
								
								config/network-policy/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								config/network-policy/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,2 @@
 | 
			
		||||
resources:
 | 
			
		||||
- allow-metrics-traffic.yaml
 | 
			
		||||
							
								
								
									
										11
									
								
								config/prometheus/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								config/prometheus/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,11 @@
 | 
			
		||||
resources:
 | 
			
		||||
- monitor.yaml
 | 
			
		||||
 | 
			
		||||
# [PROMETHEUS-WITH-CERTS] The following patch configures the ServiceMonitor in ../prometheus
 | 
			
		||||
# to securely reference certificates created and managed by cert-manager.
 | 
			
		||||
# Additionally, ensure that you uncomment the [METRICS WITH CERTMANAGER] patch under config/default/kustomization.yaml
 | 
			
		||||
# to mount the "metrics-server-cert" secret in the Manager Deployment.
 | 
			
		||||
#patches:
 | 
			
		||||
#  - path: monitor_tls_patch.yaml
 | 
			
		||||
#    target:
 | 
			
		||||
#      kind: ServiceMonitor
 | 
			
		||||
							
								
								
									
										27
									
								
								config/prometheus/monitor.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								config/prometheus/monitor.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
# Prometheus Monitor Service (Metrics)
 | 
			
		||||
apiVersion: monitoring.coreos.com/v1
 | 
			
		||||
kind: ServiceMonitor
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    control-plane: controller-manager
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: controller-manager-metrics-monitor
 | 
			
		||||
  namespace: system
 | 
			
		||||
spec:
 | 
			
		||||
  endpoints:
 | 
			
		||||
    - path: /metrics
 | 
			
		||||
      port: https # Ensure this is the name of the port that exposes HTTPS metrics
 | 
			
		||||
      scheme: https
 | 
			
		||||
      bearerTokenFile: /var/run/secrets/kubernetes.io/serviceaccount/token
 | 
			
		||||
      tlsConfig:
 | 
			
		||||
        # TODO(user): The option insecureSkipVerify: true is not recommended for production since it disables
 | 
			
		||||
        # certificate verification, exposing the system to potential man-in-the-middle attacks.
 | 
			
		||||
        # For production environments, it is recommended to use cert-manager for automatic TLS certificate management.
 | 
			
		||||
        # To apply this configuration, enable cert-manager and use the patch located at config/prometheus/servicemonitor_tls_patch.yaml,
 | 
			
		||||
        # which securely references the certificate from the 'metrics-server-cert' secret.
 | 
			
		||||
        insecureSkipVerify: true
 | 
			
		||||
  selector:
 | 
			
		||||
    matchLabels:
 | 
			
		||||
      control-plane: controller-manager
 | 
			
		||||
      app.kubernetes.io/name: yaho
 | 
			
		||||
							
								
								
									
										19
									
								
								config/prometheus/monitor_tls_patch.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								config/prometheus/monitor_tls_patch.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
# Patch for Prometheus ServiceMonitor to enable secure TLS configuration
 | 
			
		||||
# using certificates managed by cert-manager
 | 
			
		||||
- op: replace
 | 
			
		||||
  path: /spec/endpoints/0/tlsConfig
 | 
			
		||||
  value:
 | 
			
		||||
    # SERVICE_NAME and SERVICE_NAMESPACE will be substituted by kustomize
 | 
			
		||||
    serverName: SERVICE_NAME.SERVICE_NAMESPACE.svc
 | 
			
		||||
    insecureSkipVerify: false
 | 
			
		||||
    ca:
 | 
			
		||||
      secret:
 | 
			
		||||
        name: metrics-server-cert
 | 
			
		||||
        key: ca.crt
 | 
			
		||||
    cert:
 | 
			
		||||
      secret:
 | 
			
		||||
        name: metrics-server-cert
 | 
			
		||||
        key: tls.crt
 | 
			
		||||
    keySecret:
 | 
			
		||||
      name: metrics-server-cert
 | 
			
		||||
      key: tls.key
 | 
			
		||||
							
								
								
									
										27
									
								
								config/rbac/helmdiff_admin_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								config/rbac/helmdiff_admin_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants full permissions ('*') over yaho.badhouseplants.net.
 | 
			
		||||
# This role is intended for users authorized to modify roles and bindings within the cluster,
 | 
			
		||||
# enabling them to delegate specific permissions to other users or groups as needed.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmdiff-admin-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs
 | 
			
		||||
  verbs:
 | 
			
		||||
  - '*'
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										33
									
								
								config/rbac/helmdiff_editor_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								config/rbac/helmdiff_editor_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,33 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants permissions to create, update, and delete resources within the yaho.badhouseplants.net.
 | 
			
		||||
# This role is intended for users who need to manage these resources
 | 
			
		||||
# but should not control RBAC or manage permissions for others.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmdiff-editor-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs
 | 
			
		||||
  verbs:
 | 
			
		||||
  - create
 | 
			
		||||
  - delete
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - patch
 | 
			
		||||
  - update
 | 
			
		||||
  - watch
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										29
									
								
								config/rbac/helmdiff_viewer_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								config/rbac/helmdiff_viewer_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,29 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants read-only access to yaho.badhouseplants.net resources.
 | 
			
		||||
# This role is intended for users who need visibility into these resources
 | 
			
		||||
# without permissions to modify them. It is ideal for monitoring purposes and limited-access viewing.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmdiff-viewer-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - watch
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										27
									
								
								config/rbac/helmrelease_admin_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								config/rbac/helmrelease_admin_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants full permissions ('*') over yaho.badhouseplants.net.
 | 
			
		||||
# This role is intended for users authorized to modify roles and bindings within the cluster,
 | 
			
		||||
# enabling them to delegate specific permissions to other users or groups as needed.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmrelease-admin-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmreleases
 | 
			
		||||
  verbs:
 | 
			
		||||
  - '*'
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmreleases/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										33
									
								
								config/rbac/helmrelease_editor_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								config/rbac/helmrelease_editor_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,33 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants permissions to create, update, and delete resources within the yaho.badhouseplants.net.
 | 
			
		||||
# This role is intended for users who need to manage these resources
 | 
			
		||||
# but should not control RBAC or manage permissions for others.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmrelease-editor-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmreleases
 | 
			
		||||
  verbs:
 | 
			
		||||
  - create
 | 
			
		||||
  - delete
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - patch
 | 
			
		||||
  - update
 | 
			
		||||
  - watch
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmreleases/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										29
									
								
								config/rbac/helmrelease_viewer_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								config/rbac/helmrelease_viewer_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,29 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants read-only access to yaho.badhouseplants.net resources.
 | 
			
		||||
# This role is intended for users who need visibility into these resources
 | 
			
		||||
# without permissions to modify them. It is ideal for monitoring purposes and limited-access viewing.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmrelease-viewer-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmreleases
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - watch
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmreleases/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										27
									
								
								config/rbac/helmvalues_admin_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								config/rbac/helmvalues_admin_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants full permissions ('*') over yaho.badhouseplants.net.
 | 
			
		||||
# This role is intended for users authorized to modify roles and bindings within the cluster,
 | 
			
		||||
# enabling them to delegate specific permissions to other users or groups as needed.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmvalues-admin-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmvalues
 | 
			
		||||
  verbs:
 | 
			
		||||
  - '*'
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmvalues/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										33
									
								
								config/rbac/helmvalues_editor_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								config/rbac/helmvalues_editor_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,33 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants permissions to create, update, and delete resources within the yaho.badhouseplants.net.
 | 
			
		||||
# This role is intended for users who need to manage these resources
 | 
			
		||||
# but should not control RBAC or manage permissions for others.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmvalues-editor-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmvalues
 | 
			
		||||
  verbs:
 | 
			
		||||
  - create
 | 
			
		||||
  - delete
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - patch
 | 
			
		||||
  - update
 | 
			
		||||
  - watch
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmvalues/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										29
									
								
								config/rbac/helmvalues_viewer_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								config/rbac/helmvalues_viewer_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,29 @@
 | 
			
		||||
# This rule is not used by the project yaho itself.
 | 
			
		||||
# It is provided to allow the cluster admin to help manage permissions for users.
 | 
			
		||||
#
 | 
			
		||||
# Grants read-only access to yaho.badhouseplants.net resources.
 | 
			
		||||
# This role is intended for users who need visibility into these resources
 | 
			
		||||
# without permissions to modify them. It is ideal for monitoring purposes and limited-access viewing.
 | 
			
		||||
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmvalues-viewer-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmvalues
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - watch
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmvalues/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										34
									
								
								config/rbac/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								config/rbac/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,34 @@
 | 
			
		||||
resources:
 | 
			
		||||
# All RBAC will be applied under this service account in
 | 
			
		||||
# the deployment namespace. You may comment out this resource
 | 
			
		||||
# if your manager will use a service account that exists at
 | 
			
		||||
# runtime. Be sure to update RoleBinding and ClusterRoleBinding
 | 
			
		||||
# subjects if changing service account names.
 | 
			
		||||
- service_account.yaml
 | 
			
		||||
- role.yaml
 | 
			
		||||
- role_binding.yaml
 | 
			
		||||
- leader_election_role.yaml
 | 
			
		||||
- leader_election_role_binding.yaml
 | 
			
		||||
# The following RBAC configurations are used to protect
 | 
			
		||||
# the metrics endpoint with authn/authz. These configurations
 | 
			
		||||
# ensure that only authorized users and service accounts
 | 
			
		||||
# can access the metrics endpoint. Comment the following
 | 
			
		||||
# permissions if you want to disable this protection.
 | 
			
		||||
# More info: https://book.kubebuilder.io/reference/metrics.html
 | 
			
		||||
- metrics_auth_role.yaml
 | 
			
		||||
- metrics_auth_role_binding.yaml
 | 
			
		||||
- metrics_reader_role.yaml
 | 
			
		||||
# For each CRD, "Admin", "Editor" and "Viewer" roles are scaffolded by
 | 
			
		||||
# default, aiding admins in cluster management. Those roles are
 | 
			
		||||
# not used by the yaho itself. You can comment the following lines
 | 
			
		||||
# if you do not want those helpers be installed with your Project.
 | 
			
		||||
- helmvalues_admin_role.yaml
 | 
			
		||||
- helmvalues_editor_role.yaml
 | 
			
		||||
- helmvalues_viewer_role.yaml
 | 
			
		||||
- helmdiff_admin_role.yaml
 | 
			
		||||
- helmdiff_editor_role.yaml
 | 
			
		||||
- helmdiff_viewer_role.yaml
 | 
			
		||||
- helmrelease_admin_role.yaml
 | 
			
		||||
- helmrelease_editor_role.yaml
 | 
			
		||||
- helmrelease_viewer_role.yaml
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										40
									
								
								config/rbac/leader_election_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								config/rbac/leader_election_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,40 @@
 | 
			
		||||
# permissions to do leader election.
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: Role
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: leader-election-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - ""
 | 
			
		||||
  resources:
 | 
			
		||||
  - configmaps
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - watch
 | 
			
		||||
  - create
 | 
			
		||||
  - update
 | 
			
		||||
  - patch
 | 
			
		||||
  - delete
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - coordination.k8s.io
 | 
			
		||||
  resources:
 | 
			
		||||
  - leases
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - watch
 | 
			
		||||
  - create
 | 
			
		||||
  - update
 | 
			
		||||
  - patch
 | 
			
		||||
  - delete
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - ""
 | 
			
		||||
  resources:
 | 
			
		||||
  - events
 | 
			
		||||
  verbs:
 | 
			
		||||
  - create
 | 
			
		||||
  - patch
 | 
			
		||||
							
								
								
									
										15
									
								
								config/rbac/leader_election_role_binding.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								config/rbac/leader_election_role_binding.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: RoleBinding
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: leader-election-rolebinding
 | 
			
		||||
roleRef:
 | 
			
		||||
  apiGroup: rbac.authorization.k8s.io
 | 
			
		||||
  kind: Role
 | 
			
		||||
  name: leader-election-role
 | 
			
		||||
subjects:
 | 
			
		||||
- kind: ServiceAccount
 | 
			
		||||
  name: controller-manager
 | 
			
		||||
  namespace: system
 | 
			
		||||
							
								
								
									
										17
									
								
								config/rbac/metrics_auth_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								config/rbac/metrics_auth_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,17 @@
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  name: metrics-auth-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - authentication.k8s.io
 | 
			
		||||
  resources:
 | 
			
		||||
  - tokenreviews
 | 
			
		||||
  verbs:
 | 
			
		||||
  - create
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - authorization.k8s.io
 | 
			
		||||
  resources:
 | 
			
		||||
  - subjectaccessreviews
 | 
			
		||||
  verbs:
 | 
			
		||||
  - create
 | 
			
		||||
							
								
								
									
										12
									
								
								config/rbac/metrics_auth_role_binding.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								config/rbac/metrics_auth_role_binding.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRoleBinding
 | 
			
		||||
metadata:
 | 
			
		||||
  name: metrics-auth-rolebinding
 | 
			
		||||
roleRef:
 | 
			
		||||
  apiGroup: rbac.authorization.k8s.io
 | 
			
		||||
  kind: ClusterRole
 | 
			
		||||
  name: metrics-auth-role
 | 
			
		||||
subjects:
 | 
			
		||||
- kind: ServiceAccount
 | 
			
		||||
  name: controller-manager
 | 
			
		||||
  namespace: system
 | 
			
		||||
							
								
								
									
										9
									
								
								config/rbac/metrics_reader_role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								config/rbac/metrics_reader_role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,9 @@
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  name: metrics-reader
 | 
			
		||||
rules:
 | 
			
		||||
- nonResourceURLs:
 | 
			
		||||
  - "/metrics"
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
							
								
								
									
										38
									
								
								config/rbac/role.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								config/rbac/role.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,38 @@
 | 
			
		||||
---
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRole
 | 
			
		||||
metadata:
 | 
			
		||||
  name: manager-role
 | 
			
		||||
rules:
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs
 | 
			
		||||
  - helmreleases
 | 
			
		||||
  - helmvalues
 | 
			
		||||
  verbs:
 | 
			
		||||
  - create
 | 
			
		||||
  - delete
 | 
			
		||||
  - get
 | 
			
		||||
  - list
 | 
			
		||||
  - patch
 | 
			
		||||
  - update
 | 
			
		||||
  - watch
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs/finalizers
 | 
			
		||||
  - helmreleases/finalizers
 | 
			
		||||
  - helmvalues/finalizers
 | 
			
		||||
  verbs:
 | 
			
		||||
  - update
 | 
			
		||||
- apiGroups:
 | 
			
		||||
  - yaho.badhouseplants.net
 | 
			
		||||
  resources:
 | 
			
		||||
  - helmdiffs/status
 | 
			
		||||
  - helmreleases/status
 | 
			
		||||
  - helmvalues/status
 | 
			
		||||
  verbs:
 | 
			
		||||
  - get
 | 
			
		||||
  - patch
 | 
			
		||||
  - update
 | 
			
		||||
							
								
								
									
										15
									
								
								config/rbac/role_binding.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								config/rbac/role_binding.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
apiVersion: rbac.authorization.k8s.io/v1
 | 
			
		||||
kind: ClusterRoleBinding
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: manager-rolebinding
 | 
			
		||||
roleRef:
 | 
			
		||||
  apiGroup: rbac.authorization.k8s.io
 | 
			
		||||
  kind: ClusterRole
 | 
			
		||||
  name: manager-role
 | 
			
		||||
subjects:
 | 
			
		||||
- kind: ServiceAccount
 | 
			
		||||
  name: controller-manager
 | 
			
		||||
  namespace: system
 | 
			
		||||
							
								
								
									
										8
									
								
								config/rbac/service_account.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								config/rbac/service_account.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
			
		||||
apiVersion: v1
 | 
			
		||||
kind: ServiceAccount
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: controller-manager
 | 
			
		||||
  namespace: system
 | 
			
		||||
							
								
								
									
										6
									
								
								config/samples/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								config/samples/kustomization.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,6 @@
 | 
			
		||||
## Append samples of your project ##
 | 
			
		||||
resources:
 | 
			
		||||
- yaho_v1alpha1_helmrelease.yaml
 | 
			
		||||
- yaho_v1alpha1_helmdiff.yaml
 | 
			
		||||
- yaho_v1alpha1_helmvalues.yaml
 | 
			
		||||
# +kubebuilder:scaffold:manifestskustomizesamples
 | 
			
		||||
							
								
								
									
										9
									
								
								config/samples/yaho_v1alpha1_helmdiff.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								config/samples/yaho_v1alpha1_helmdiff.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,9 @@
 | 
			
		||||
apiVersion: yaho.badhouseplants.net/v1alpha1
 | 
			
		||||
kind: HelmDiff
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmdiff-sample
 | 
			
		||||
spec:
 | 
			
		||||
  # TODO(user): Add fields here
 | 
			
		||||
							
								
								
									
										12
									
								
								config/samples/yaho_v1alpha1_helmrelease.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								config/samples/yaho_v1alpha1_helmrelease.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
			
		||||
apiVersion: yaho.badhouseplants.net/v1alpha1
 | 
			
		||||
kind: HelmRelease
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmrelease-sample
 | 
			
		||||
spec:
 | 
			
		||||
  repository: https://coredns.github.io/helm
 | 
			
		||||
  chart: coredns
 | 
			
		||||
  version: 1.43.0
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										9
									
								
								config/samples/yaho_v1alpha1_helmvalues.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								config/samples/yaho_v1alpha1_helmvalues.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,9 @@
 | 
			
		||||
apiVersion: yaho.badhouseplants.net/v1alpha1
 | 
			
		||||
kind: HelmValues
 | 
			
		||||
metadata:
 | 
			
		||||
  labels:
 | 
			
		||||
    app.kubernetes.io/name: yaho
 | 
			
		||||
    app.kubernetes.io/managed-by: kustomize
 | 
			
		||||
  name: helmvalues-sample
 | 
			
		||||
spec:
 | 
			
		||||
  # TODO(user): Add fields here
 | 
			
		||||
							
								
								
									
										187
									
								
								go.mod
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										187
									
								
								go.mod
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,187 @@
 | 
			
		||||
module github.com/allanger/yaho
 | 
			
		||||
 | 
			
		||||
go 1.24.0
 | 
			
		||||
 | 
			
		||||
require (
 | 
			
		||||
	github.com/go-git/go-git/v5 v5.16.2
 | 
			
		||||
	github.com/onsi/ginkgo/v2 v2.22.0
 | 
			
		||||
	github.com/onsi/gomega v1.36.1
 | 
			
		||||
	github.com/stretchr/testify v1.10.0
 | 
			
		||||
	helm.sh/helm v2.17.0+incompatible
 | 
			
		||||
	helm.sh/helm/v3 v3.18.4
 | 
			
		||||
	honnef.co/go/tools v0.6.1
 | 
			
		||||
	k8s.io/apimachinery v0.33.2
 | 
			
		||||
	k8s.io/client-go v0.33.2
 | 
			
		||||
	sigs.k8s.io/controller-runtime v0.21.0
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
require (
 | 
			
		||||
	cel.dev/expr v0.19.1 // indirect
 | 
			
		||||
	dario.cat/mergo v1.0.1 // indirect
 | 
			
		||||
	github.com/Azure/go-ansiterm v0.0.0-20250102033503-faa5f7b0171c // indirect
 | 
			
		||||
	github.com/BurntSushi/toml v1.5.0 // indirect
 | 
			
		||||
	github.com/MakeNowJust/heredoc v1.0.0 // indirect
 | 
			
		||||
	github.com/Masterminds/goutils v1.1.1 // indirect
 | 
			
		||||
	github.com/Masterminds/semver v1.5.0 // indirect
 | 
			
		||||
	github.com/Masterminds/semver/v3 v3.3.0 // indirect
 | 
			
		||||
	github.com/Masterminds/sprig v2.22.0+incompatible // indirect
 | 
			
		||||
	github.com/Masterminds/sprig/v3 v3.3.0 // indirect
 | 
			
		||||
	github.com/Masterminds/squirrel v1.5.4 // indirect
 | 
			
		||||
	github.com/Microsoft/go-winio v0.6.2 // indirect
 | 
			
		||||
	github.com/ProtonMail/go-crypto v1.1.6 // indirect
 | 
			
		||||
	github.com/antlr4-go/antlr/v4 v4.13.0 // indirect
 | 
			
		||||
	github.com/asaskevich/govalidator v0.0.0-20230301143203-a9d515a09cc2 // indirect
 | 
			
		||||
	github.com/beorn7/perks v1.0.1 // indirect
 | 
			
		||||
	github.com/blang/semver/v4 v4.0.0 // indirect
 | 
			
		||||
	github.com/cenkalti/backoff/v4 v4.3.0 // indirect
 | 
			
		||||
	github.com/cespare/xxhash/v2 v2.3.0 // indirect
 | 
			
		||||
	github.com/chai2010/gettext-go v1.0.2 // indirect
 | 
			
		||||
	github.com/cloudflare/circl v1.6.1 // indirect
 | 
			
		||||
	github.com/containerd/containerd v1.7.27 // indirect
 | 
			
		||||
	github.com/containerd/errdefs v0.3.0 // indirect
 | 
			
		||||
	github.com/containerd/log v0.1.0 // indirect
 | 
			
		||||
	github.com/containerd/platforms v0.2.1 // indirect
 | 
			
		||||
	github.com/cyphar/filepath-securejoin v0.4.1 // indirect
 | 
			
		||||
	github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect
 | 
			
		||||
	github.com/emicklei/go-restful/v3 v3.11.0 // indirect
 | 
			
		||||
	github.com/emirpasic/gods v1.18.1 // indirect
 | 
			
		||||
	github.com/evanphx/json-patch v5.9.11+incompatible // indirect
 | 
			
		||||
	github.com/evanphx/json-patch/v5 v5.9.11 // indirect
 | 
			
		||||
	github.com/exponent-io/jsonpath v0.0.0-20210407135951-1de76d718b3f // indirect
 | 
			
		||||
	github.com/fatih/color v1.13.0 // indirect
 | 
			
		||||
	github.com/felixge/httpsnoop v1.0.4 // indirect
 | 
			
		||||
	github.com/fsnotify/fsnotify v1.7.0 // indirect
 | 
			
		||||
	github.com/fxamacker/cbor/v2 v2.7.0 // indirect
 | 
			
		||||
	github.com/ghodss/yaml v1.0.0 // indirect
 | 
			
		||||
	github.com/go-errors/errors v1.4.2 // indirect
 | 
			
		||||
	github.com/go-git/gcfg v1.5.1-0.20230307220236-3a3c6141e376 // indirect
 | 
			
		||||
	github.com/go-git/go-billy/v5 v5.6.2 // indirect
 | 
			
		||||
	github.com/go-gorp/gorp/v3 v3.1.0 // indirect
 | 
			
		||||
	github.com/go-logr/logr v1.4.2 // indirect
 | 
			
		||||
	github.com/go-logr/stdr v1.2.2 // indirect
 | 
			
		||||
	github.com/go-logr/zapr v1.3.0 // indirect
 | 
			
		||||
	github.com/go-openapi/jsonpointer v0.21.0 // indirect
 | 
			
		||||
	github.com/go-openapi/jsonreference v0.20.2 // indirect
 | 
			
		||||
	github.com/go-openapi/swag v0.23.0 // indirect
 | 
			
		||||
	github.com/go-task/slim-sprig/v3 v3.0.0 // indirect
 | 
			
		||||
	github.com/gobwas/glob v0.2.3 // indirect
 | 
			
		||||
	github.com/gogo/protobuf v1.3.2 // indirect
 | 
			
		||||
	github.com/golang/groupcache v0.0.0-20241129210726-2c02b8208cf8 // indirect
 | 
			
		||||
	github.com/golang/protobuf v1.5.4 // indirect
 | 
			
		||||
	github.com/google/btree v1.1.3 // indirect
 | 
			
		||||
	github.com/google/cel-go v0.23.2 // indirect
 | 
			
		||||
	github.com/google/gnostic-models v0.6.9 // indirect
 | 
			
		||||
	github.com/google/go-cmp v0.7.0 // indirect
 | 
			
		||||
	github.com/google/pprof v0.0.0-20241029153458-d1b30febd7db // indirect
 | 
			
		||||
	github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect
 | 
			
		||||
	github.com/google/uuid v1.6.0 // indirect
 | 
			
		||||
	github.com/gorilla/websocket v1.5.4-0.20250319132907-e064f32e3674 // indirect
 | 
			
		||||
	github.com/gosuri/uitable v0.0.4 // indirect
 | 
			
		||||
	github.com/gregjones/httpcache v0.0.0-20190611155906-901d90724c79 // indirect
 | 
			
		||||
	github.com/grpc-ecosystem/grpc-gateway/v2 v2.24.0 // indirect
 | 
			
		||||
	github.com/hashicorp/errwrap v1.1.0 // indirect
 | 
			
		||||
	github.com/hashicorp/go-multierror v1.1.1 // indirect
 | 
			
		||||
	github.com/huandu/xstrings v1.5.0 // indirect
 | 
			
		||||
	github.com/imdario/mergo v0.3.13 // indirect
 | 
			
		||||
	github.com/inconshreveable/mousetrap v1.1.0 // indirect
 | 
			
		||||
	github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 // indirect
 | 
			
		||||
	github.com/jmoiron/sqlx v1.4.0 // indirect
 | 
			
		||||
	github.com/josharian/intern v1.0.0 // indirect
 | 
			
		||||
	github.com/json-iterator/go v1.1.12 // indirect
 | 
			
		||||
	github.com/kevinburke/ssh_config v1.2.0 // indirect
 | 
			
		||||
	github.com/klauspost/compress v1.18.0 // indirect
 | 
			
		||||
	github.com/lann/builder v0.0.0-20180802200727-47ae307949d0 // indirect
 | 
			
		||||
	github.com/lann/ps v0.0.0-20150810152359-62de8c46ede0 // indirect
 | 
			
		||||
	github.com/lib/pq v1.10.9 // indirect
 | 
			
		||||
	github.com/liggitt/tabwriter v0.0.0-20181228230101-89fcab3d43de // indirect
 | 
			
		||||
	github.com/mailru/easyjson v0.7.7 // indirect
 | 
			
		||||
	github.com/mattn/go-colorable v0.1.13 // indirect
 | 
			
		||||
	github.com/mattn/go-isatty v0.0.17 // indirect
 | 
			
		||||
	github.com/mattn/go-runewidth v0.0.9 // indirect
 | 
			
		||||
	github.com/mitchellh/copystructure v1.2.0 // indirect
 | 
			
		||||
	github.com/mitchellh/go-wordwrap v1.0.1 // indirect
 | 
			
		||||
	github.com/mitchellh/reflectwalk v1.0.2 // indirect
 | 
			
		||||
	github.com/moby/spdystream v0.5.0 // indirect
 | 
			
		||||
	github.com/moby/term v0.5.2 // indirect
 | 
			
		||||
	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
 | 
			
		||||
	github.com/modern-go/reflect2 v1.0.2 // indirect
 | 
			
		||||
	github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00 // indirect
 | 
			
		||||
	github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect
 | 
			
		||||
	github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f // indirect
 | 
			
		||||
	github.com/opencontainers/go-digest v1.0.0 // indirect
 | 
			
		||||
	github.com/opencontainers/image-spec v1.1.1 // indirect
 | 
			
		||||
	github.com/peterbourgon/diskv v2.0.1+incompatible // indirect
 | 
			
		||||
	github.com/pjbgf/sha1cd v0.3.2 // indirect
 | 
			
		||||
	github.com/pkg/errors v0.9.1 // indirect
 | 
			
		||||
	github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect
 | 
			
		||||
	github.com/prometheus/client_golang v1.22.0 // indirect
 | 
			
		||||
	github.com/prometheus/client_model v0.6.1 // indirect
 | 
			
		||||
	github.com/prometheus/common v0.62.0 // indirect
 | 
			
		||||
	github.com/prometheus/procfs v0.15.1 // indirect
 | 
			
		||||
	github.com/rubenv/sql-migrate v1.8.0 // indirect
 | 
			
		||||
	github.com/russross/blackfriday/v2 v2.1.0 // indirect
 | 
			
		||||
	github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 // indirect
 | 
			
		||||
	github.com/shopspring/decimal v1.4.0 // indirect
 | 
			
		||||
	github.com/sirupsen/logrus v1.9.3 // indirect
 | 
			
		||||
	github.com/skeema/knownhosts v1.3.1 // indirect
 | 
			
		||||
	github.com/spf13/cast v1.7.0 // indirect
 | 
			
		||||
	github.com/spf13/cobra v1.9.1 // indirect
 | 
			
		||||
	github.com/spf13/pflag v1.0.6 // indirect
 | 
			
		||||
	github.com/stoewer/go-strcase v1.3.0 // indirect
 | 
			
		||||
	github.com/x448/float16 v0.8.4 // indirect
 | 
			
		||||
	github.com/xanzy/ssh-agent v0.3.3 // indirect
 | 
			
		||||
	github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb // indirect
 | 
			
		||||
	github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect
 | 
			
		||||
	github.com/xeipuuv/gojsonschema v1.2.0 // indirect
 | 
			
		||||
	github.com/xlab/treeprint v1.2.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/auto/sdk v1.1.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.58.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel v1.33.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.33.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.33.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/metric v1.33.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/sdk v1.33.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/trace v1.33.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/proto/otlp v1.4.0 // indirect
 | 
			
		||||
	go.uber.org/multierr v1.11.0 // indirect
 | 
			
		||||
	go.uber.org/zap v1.27.0 // indirect
 | 
			
		||||
	golang.org/x/crypto v0.39.0 // indirect
 | 
			
		||||
	golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56 // indirect
 | 
			
		||||
	golang.org/x/mod v0.25.0 // indirect
 | 
			
		||||
	golang.org/x/net v0.40.0 // indirect
 | 
			
		||||
	golang.org/x/oauth2 v0.28.0 // indirect
 | 
			
		||||
	golang.org/x/sync v0.15.0 // indirect
 | 
			
		||||
	golang.org/x/sys v0.33.0 // indirect
 | 
			
		||||
	golang.org/x/term v0.32.0 // indirect
 | 
			
		||||
	golang.org/x/text v0.26.0 // indirect
 | 
			
		||||
	golang.org/x/time v0.9.0 // indirect
 | 
			
		||||
	golang.org/x/tools v0.33.0 // indirect
 | 
			
		||||
	gomodules.xyz/jsonpatch/v2 v2.4.0 // indirect
 | 
			
		||||
	google.golang.org/genproto/googleapis/api v0.0.0-20241209162323-e6fa225c2576 // indirect
 | 
			
		||||
	google.golang.org/genproto/googleapis/rpc v0.0.0-20241209162323-e6fa225c2576 // indirect
 | 
			
		||||
	google.golang.org/grpc v1.68.1 // indirect
 | 
			
		||||
	google.golang.org/protobuf v1.36.5 // indirect
 | 
			
		||||
	gopkg.in/evanphx/json-patch.v4 v4.12.0 // indirect
 | 
			
		||||
	gopkg.in/inf.v0 v0.9.1 // indirect
 | 
			
		||||
	gopkg.in/warnings.v0 v0.1.2 // indirect
 | 
			
		||||
	gopkg.in/yaml.v2 v2.4.0 // indirect
 | 
			
		||||
	gopkg.in/yaml.v3 v3.0.1 // indirect
 | 
			
		||||
	k8s.io/api v0.33.2 // indirect
 | 
			
		||||
	k8s.io/apiextensions-apiserver v0.33.2 // indirect
 | 
			
		||||
	k8s.io/apiserver v0.33.2 // indirect
 | 
			
		||||
	k8s.io/cli-runtime v0.33.2 // indirect
 | 
			
		||||
	k8s.io/component-base v0.33.2 // indirect
 | 
			
		||||
	k8s.io/helm v2.17.0+incompatible // indirect
 | 
			
		||||
	k8s.io/klog/v2 v2.130.1 // indirect
 | 
			
		||||
	k8s.io/kube-openapi v0.0.0-20250318190949-c8a335a9a2ff // indirect
 | 
			
		||||
	k8s.io/kubectl v0.33.2 // indirect
 | 
			
		||||
	k8s.io/utils v0.0.0-20241104100929-3ea5e8cea738 // indirect
 | 
			
		||||
	oras.land/oras-go/v2 v2.6.0 // indirect
 | 
			
		||||
	sigs.k8s.io/apiserver-network-proxy/konnectivity-client v0.31.2 // indirect
 | 
			
		||||
	sigs.k8s.io/json v0.0.0-20241010143419-9aa6b5e7a4b3 // indirect
 | 
			
		||||
	sigs.k8s.io/kustomize/api v0.19.0 // indirect
 | 
			
		||||
	sigs.k8s.io/kustomize/kyaml v0.19.0 // indirect
 | 
			
		||||
	sigs.k8s.io/randfill v1.0.0 // indirect
 | 
			
		||||
	sigs.k8s.io/structured-merge-diff/v4 v4.6.0 // indirect
 | 
			
		||||
	sigs.k8s.io/yaml v1.4.0 // indirect
 | 
			
		||||
)
 | 
			
		||||
							
								
								
									
										546
									
								
								go.sum
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										546
									
								
								go.sum
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,546 @@
 | 
			
		||||
cel.dev/expr v0.19.1 h1:NciYrtDRIR0lNCnH1LFJegdjspNx9fI59O7TWcua/W4=
 | 
			
		||||
cel.dev/expr v0.19.1/go.mod h1:MrpN08Q+lEBs+bGYdLxxHkZoUSsCp0nSKTs0nTymJgw=
 | 
			
		||||
dario.cat/mergo v1.0.1 h1:Ra4+bf83h2ztPIQYNP99R6m+Y7KfnARDfID+a+vLl4s=
 | 
			
		||||
dario.cat/mergo v1.0.1/go.mod h1:uNxQE+84aUszobStD9th8a29P2fMDhsBdgRYvZOxGmk=
 | 
			
		||||
filippo.io/edwards25519 v1.1.0 h1:FNf4tywRC1HmFuKW5xopWpigGjJKiJSV0Cqo0cJWDaA=
 | 
			
		||||
filippo.io/edwards25519 v1.1.0/go.mod h1:BxyFTGdWcka3PhytdK4V28tE5sGfRvvvRV7EaN4VDT4=
 | 
			
		||||
github.com/AdaLogics/go-fuzz-headers v0.0.0-20230811130428-ced1acdcaa24 h1:bvDV9vkmnHYOMsOr4WLk+Vo07yKIzd94sVoIqshQ4bU=
 | 
			
		||||
github.com/AdaLogics/go-fuzz-headers v0.0.0-20230811130428-ced1acdcaa24/go.mod h1:8o94RPi1/7XTJvwPpRSzSUedZrtlirdB3r9Z20bi2f8=
 | 
			
		||||
github.com/Azure/go-ansiterm v0.0.0-20250102033503-faa5f7b0171c h1:udKWzYgxTojEKWjV8V+WSxDXJ4NFATAsZjh8iIbsQIg=
 | 
			
		||||
github.com/Azure/go-ansiterm v0.0.0-20250102033503-faa5f7b0171c/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
 | 
			
		||||
github.com/BurntSushi/toml v1.5.0 h1:W5quZX/G/csjUnuI8SUYlsHs9M38FC7znL0lIO+DvMg=
 | 
			
		||||
github.com/BurntSushi/toml v1.5.0/go.mod h1:ukJfTF/6rtPPRCnwkur4qwRxa8vTRFBF0uk2lLoLwho=
 | 
			
		||||
github.com/DATA-DOG/go-sqlmock v1.5.2 h1:OcvFkGmslmlZibjAjaHm3L//6LiuBgolP7OputlJIzU=
 | 
			
		||||
github.com/DATA-DOG/go-sqlmock v1.5.2/go.mod h1:88MAG/4G7SMwSE3CeA0ZKzrT5CiOU3OJ+JlNzwDqpNU=
 | 
			
		||||
github.com/MakeNowJust/heredoc v1.0.0 h1:cXCdzVdstXyiTqTvfqk9SDHpKNjxuom+DOlyEeQ4pzQ=
 | 
			
		||||
github.com/MakeNowJust/heredoc v1.0.0/go.mod h1:mG5amYoWBHf8vpLOuehzbGGw0EHxpZZ6lCpQ4fNJ8LE=
 | 
			
		||||
github.com/Masterminds/goutils v1.1.1 h1:5nUrii3FMTL5diU80unEVvNevw1nH4+ZV4DSLVJLSYI=
 | 
			
		||||
github.com/Masterminds/goutils v1.1.1/go.mod h1:8cTjp+g8YejhMuvIA5y2vz3BpJxksy863GQaJW2MFNU=
 | 
			
		||||
github.com/Masterminds/semver v1.5.0 h1:H65muMkzWKEuNDnfl9d70GUjFniHKHRbFPGBuZ3QEww=
 | 
			
		||||
github.com/Masterminds/semver v1.5.0/go.mod h1:MB6lktGJrhw8PrUyiEoblNEGEQ+RzHPF078ddwwvV3Y=
 | 
			
		||||
github.com/Masterminds/semver/v3 v3.3.0 h1:B8LGeaivUe71a5qox1ICM/JLl0NqZSW5CHyL+hmvYS0=
 | 
			
		||||
github.com/Masterminds/semver/v3 v3.3.0/go.mod h1:4V+yj/TJE1HU9XfppCwVMZq3I84lprf4nC11bSS5beM=
 | 
			
		||||
github.com/Masterminds/sprig v2.22.0+incompatible h1:z4yfnGrZ7netVz+0EDJ0Wi+5VZCSYp4Z0m2dk6cEM60=
 | 
			
		||||
github.com/Masterminds/sprig v2.22.0+incompatible/go.mod h1:y6hNFY5UBTIWBxnzTeuNhlNS5hqE0NB0E6fgfo2Br3o=
 | 
			
		||||
github.com/Masterminds/sprig/v3 v3.3.0 h1:mQh0Yrg1XPo6vjYXgtf5OtijNAKJRNcTdOOGZe3tPhs=
 | 
			
		||||
github.com/Masterminds/sprig/v3 v3.3.0/go.mod h1:Zy1iXRYNqNLUolqCpL4uhk6SHUMAOSCzdgBfDb35Lz0=
 | 
			
		||||
github.com/Masterminds/squirrel v1.5.4 h1:uUcX/aBc8O7Fg9kaISIUsHXdKuqehiXAMQTYX8afzqM=
 | 
			
		||||
github.com/Masterminds/squirrel v1.5.4/go.mod h1:NNaOrjSoIDfDA40n7sr2tPNZRfjzjA400rg+riTZj10=
 | 
			
		||||
github.com/Microsoft/go-winio v0.5.2/go.mod h1:WpS1mjBmmwHBEWmogvA2mj8546UReBk4v8QkMxJ6pZY=
 | 
			
		||||
github.com/Microsoft/go-winio v0.6.2 h1:F2VQgta7ecxGYO8k3ZZz3RS8fVIXVxONVUPlNERoyfY=
 | 
			
		||||
github.com/Microsoft/go-winio v0.6.2/go.mod h1:yd8OoFMLzJbo9gZq8j5qaps8bJ9aShtEA8Ipt1oGCvU=
 | 
			
		||||
github.com/ProtonMail/go-crypto v1.1.6 h1:ZcV+Ropw6Qn0AX9brlQLAUXfqLBc7Bl+f/DmNxpLfdw=
 | 
			
		||||
github.com/ProtonMail/go-crypto v1.1.6/go.mod h1:rA3QumHc/FZ8pAHreoekgiAbzpNsfQAosU5td4SnOrE=
 | 
			
		||||
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be h1:9AeTilPcZAjCFIImctFaOjnTIavg87rW78vTPkQqLI8=
 | 
			
		||||
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be/go.mod h1:ySMOLuWl6zY27l47sB3qLNK6tF2fkHG55UZxx8oIVo4=
 | 
			
		||||
github.com/antlr4-go/antlr/v4 v4.13.0 h1:lxCg3LAv+EUK6t1i0y1V6/SLeUi0eKEKdhQAlS8TVTI=
 | 
			
		||||
github.com/antlr4-go/antlr/v4 v4.13.0/go.mod h1:pfChB/xh/Unjila75QW7+VU4TSnWnnk9UTnmpPaOR2g=
 | 
			
		||||
github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5 h1:0CwZNZbxp69SHPdPJAN/hZIm0C4OItdklCFmMRWYpio=
 | 
			
		||||
github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5/go.mod h1:wHh0iHkYZB8zMSxRWpUBQtwG5a7fFgvEO+odwuTv2gs=
 | 
			
		||||
github.com/asaskevich/govalidator v0.0.0-20230301143203-a9d515a09cc2 h1:DklsrG3dyBCFEj5IhUbnKptjxatkF07cF2ak3yi77so=
 | 
			
		||||
github.com/asaskevich/govalidator v0.0.0-20230301143203-a9d515a09cc2/go.mod h1:WaHUgvxTVq04UNunO+XhnAqY/wQc+bxr74GqbsZ/Jqw=
 | 
			
		||||
github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM=
 | 
			
		||||
github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
 | 
			
		||||
github.com/blang/semver/v4 v4.0.0 h1:1PFHFE6yCCTv8C1TeyNNarDzntLi7wMI5i/pzqYIsAM=
 | 
			
		||||
github.com/blang/semver/v4 v4.0.0/go.mod h1:IbckMUScFkM3pff0VJDNKRiT6TG/YpiHIM2yvyW5YoQ=
 | 
			
		||||
github.com/bshuster-repo/logrus-logstash-hook v1.0.0 h1:e+C0SB5R1pu//O4MQ3f9cFuPGoOVeF2fE4Og9otCc70=
 | 
			
		||||
github.com/bshuster-repo/logrus-logstash-hook v1.0.0/go.mod h1:zsTqEiSzDgAa/8GZR7E1qaXrhYNDKBYy5/dWPTIflbk=
 | 
			
		||||
github.com/cenkalti/backoff/v4 v4.3.0 h1:MyRJ/UdXutAwSAT+s3wNd7MfTIcy71VQueUuFK343L8=
 | 
			
		||||
github.com/cenkalti/backoff/v4 v4.3.0/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE=
 | 
			
		||||
github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs=
 | 
			
		||||
github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
 | 
			
		||||
github.com/chai2010/gettext-go v1.0.2 h1:1Lwwip6Q2QGsAdl/ZKPCwTe9fe0CjlUbqj5bFNSjIRk=
 | 
			
		||||
github.com/chai2010/gettext-go v1.0.2/go.mod h1:y+wnP2cHYaVj19NZhYKAwEMH2CI1gNHeQQ+5AjwawxA=
 | 
			
		||||
github.com/cloudflare/circl v1.6.1 h1:zqIqSPIndyBh1bjLVVDHMPpVKqp8Su/V+6MeDzzQBQ0=
 | 
			
		||||
github.com/cloudflare/circl v1.6.1/go.mod h1:uddAzsPgqdMAYatqJ0lsjX1oECcQLIlRpzZh3pJrofs=
 | 
			
		||||
github.com/containerd/containerd v1.7.27 h1:yFyEyojddO3MIGVER2xJLWoCIn+Up4GaHFquP7hsFII=
 | 
			
		||||
github.com/containerd/containerd v1.7.27/go.mod h1:xZmPnl75Vc+BLGt4MIfu6bp+fy03gdHAn9bz+FreFR0=
 | 
			
		||||
github.com/containerd/errdefs v0.3.0 h1:FSZgGOeK4yuT/+DnF07/Olde/q4KBoMsaamhXxIMDp4=
 | 
			
		||||
github.com/containerd/errdefs v0.3.0/go.mod h1:+YBYIdtsnF4Iw6nWZhJcqGSg/dwvV7tyJ/kCkyJ2k+M=
 | 
			
		||||
github.com/containerd/log v0.1.0 h1:TCJt7ioM2cr/tfR8GPbGf9/VRAX8D2B4PjzCpfX540I=
 | 
			
		||||
github.com/containerd/log v0.1.0/go.mod h1:VRRf09a7mHDIRezVKTRCrOq78v577GXq3bSa3EhrzVo=
 | 
			
		||||
github.com/containerd/platforms v0.2.1 h1:zvwtM3rz2YHPQsF2CHYM8+KtB5dvhISiXh5ZpSBQv6A=
 | 
			
		||||
github.com/containerd/platforms v0.2.1/go.mod h1:XHCb+2/hzowdiut9rkudds9bE5yJ7npe7dG/wG+uFPw=
 | 
			
		||||
github.com/coreos/go-systemd/v22 v22.5.0 h1:RrqgGjYQKalulkV8NGVIfkXQf6YYmOyiJKk8iXXhfZs=
 | 
			
		||||
github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc=
 | 
			
		||||
github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6NIQQ7OS05n1F4g=
 | 
			
		||||
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
 | 
			
		||||
github.com/creack/pty v1.1.18 h1:n56/Zwd5o6whRC5PMGretI4IdRLlmBXYNjScPaBgsbY=
 | 
			
		||||
github.com/creack/pty v1.1.18/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4=
 | 
			
		||||
github.com/cyphar/filepath-securejoin v0.4.1 h1:JyxxyPEaktOD+GAnqIqTf9A8tHyAG22rowi7HkoSU1s=
 | 
			
		||||
github.com/cyphar/filepath-securejoin v0.4.1/go.mod h1:Sdj7gXlvMcPZsbhwhQ33GguGLDGQL7h7bg04C/+u9jI=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78=
 | 
			
		||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc=
 | 
			
		||||
github.com/distribution/distribution/v3 v3.0.0 h1:q4R8wemdRQDClzoNNStftB2ZAfqOiN6UX90KJc4HjyM=
 | 
			
		||||
github.com/distribution/distribution/v3 v3.0.0/go.mod h1:tRNuFoZsUdyRVegq8xGNeds4KLjwLCRin/tTo6i1DhU=
 | 
			
		||||
github.com/distribution/reference v0.6.0 h1:0IXCQ5g4/QMHHkarYzh5l+u8T3t73zM5QvfrDyIgxBk=
 | 
			
		||||
github.com/distribution/reference v0.6.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E=
 | 
			
		||||
github.com/docker/docker-credential-helpers v0.8.2 h1:bX3YxiGzFP5sOXWc3bTPEXdEaZSeVMrFgOr3T+zrFAo=
 | 
			
		||||
github.com/docker/docker-credential-helpers v0.8.2/go.mod h1:P3ci7E3lwkZg6XiHdRKft1KckHiO9a2rNtyFbZ/ry9M=
 | 
			
		||||
github.com/docker/go-events v0.0.0-20190806004212-e31b211e4f1c h1:+pKlWGMw7gf6bQ+oDZB4KHQFypsfjYlq/C4rfL7D3g8=
 | 
			
		||||
github.com/docker/go-events v0.0.0-20190806004212-e31b211e4f1c/go.mod h1:Uw6UezgYA44ePAFQYUehOuCzmy5zmg/+nl2ZfMWGkpA=
 | 
			
		||||
github.com/docker/go-metrics v0.0.1 h1:AgB/0SvBxihN0X8OR4SjsblXkbMvalQ8cjmtKQ2rQV8=
 | 
			
		||||
github.com/docker/go-metrics v0.0.1/go.mod h1:cG1hvH2utMXtqgqqYE9plW6lDxS3/5ayHzueweSI3Vw=
 | 
			
		||||
github.com/elazarl/goproxy v1.7.2 h1:Y2o6urb7Eule09PjlhQRGNsqRfPmYI3KKQLFpCAV3+o=
 | 
			
		||||
github.com/elazarl/goproxy v1.7.2/go.mod h1:82vkLNir0ALaW14Rc399OTTjyNREgmdL2cVoIbS6XaE=
 | 
			
		||||
github.com/emicklei/go-restful/v3 v3.11.0 h1:rAQeMHw1c7zTmncogyy8VvRZwtkmkZ4FxERmMY4rD+g=
 | 
			
		||||
github.com/emicklei/go-restful/v3 v3.11.0/go.mod h1:6n3XBCmQQb25CM2LCACGz8ukIrRry+4bhvbpWn3mrbc=
 | 
			
		||||
github.com/emirpasic/gods v1.18.1 h1:FXtiHYKDGKCW2KzwZKx0iC0PQmdlorYgdFG9jPXJ1Bc=
 | 
			
		||||
github.com/emirpasic/gods v1.18.1/go.mod h1:8tpGGwCnJ5H4r6BWwaV6OrWmMoPhUl5jm/FMNAnJvWQ=
 | 
			
		||||
github.com/evanphx/json-patch v5.9.11+incompatible h1:ixHHqfcGvxhWkniF1tWxBHA0yb4Z+d1UQi45df52xW8=
 | 
			
		||||
github.com/evanphx/json-patch v5.9.11+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
 | 
			
		||||
github.com/evanphx/json-patch/v5 v5.9.11 h1:/8HVnzMq13/3x9TPvjG08wUGqBTmZBsCWzjTM0wiaDU=
 | 
			
		||||
github.com/evanphx/json-patch/v5 v5.9.11/go.mod h1:3j+LviiESTElxA4p3EMKAB9HXj3/XEtnUf6OZxqIQTM=
 | 
			
		||||
github.com/exponent-io/jsonpath v0.0.0-20210407135951-1de76d718b3f h1:Wl78ApPPB2Wvf/TIe2xdyJxTlb6obmF18d8QdkxNDu4=
 | 
			
		||||
github.com/exponent-io/jsonpath v0.0.0-20210407135951-1de76d718b3f/go.mod h1:OSYXu++VVOHnXeitef/D8n/6y4QV8uLHSFXX4NeXMGc=
 | 
			
		||||
github.com/fatih/color v1.13.0 h1:8LOYc1KYPPmyKMuN8QV2DNRWNbLo6LZ0iLs8+mlH53w=
 | 
			
		||||
github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk=
 | 
			
		||||
github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg=
 | 
			
		||||
github.com/felixge/httpsnoop v1.0.4/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
 | 
			
		||||
github.com/foxcpp/go-mockdns v1.1.0 h1:jI0rD8M0wuYAxL7r/ynTrCQQq0BVqfB99Vgk7DlmewI=
 | 
			
		||||
github.com/foxcpp/go-mockdns v1.1.0/go.mod h1:IhLeSFGed3mJIAXPH2aiRQB+kqz7oqu8ld2qVbOu7Wk=
 | 
			
		||||
github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
 | 
			
		||||
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
 | 
			
		||||
github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA=
 | 
			
		||||
github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM=
 | 
			
		||||
github.com/fxamacker/cbor/v2 v2.7.0 h1:iM5WgngdRBanHcxugY4JySA0nk1wZorNOpTgCMedv5E=
 | 
			
		||||
github.com/fxamacker/cbor/v2 v2.7.0/go.mod h1:pxXPTn3joSm21Gbwsv0w9OSA2y1HFR9qXEeXQVeNoDQ=
 | 
			
		||||
github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk=
 | 
			
		||||
github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
 | 
			
		||||
github.com/gliderlabs/ssh v0.3.8 h1:a4YXD1V7xMF9g5nTkdfnja3Sxy1PVDCj1Zg4Wb8vY6c=
 | 
			
		||||
github.com/gliderlabs/ssh v0.3.8/go.mod h1:xYoytBv1sV0aL3CavoDuJIQNURXkkfPA/wxQ1pL1fAU=
 | 
			
		||||
github.com/go-errors/errors v1.4.2 h1:J6MZopCL4uSllY1OfXM374weqZFFItUbrImctkmUxIA=
 | 
			
		||||
github.com/go-errors/errors v1.4.2/go.mod h1:sIVyrIiJhuEF+Pj9Ebtd6P/rEYROXFi3BopGUQ5a5Og=
 | 
			
		||||
github.com/go-git/gcfg v1.5.1-0.20230307220236-3a3c6141e376 h1:+zs/tPmkDkHx3U66DAb0lQFJrpS6731Oaa12ikc+DiI=
 | 
			
		||||
github.com/go-git/gcfg v1.5.1-0.20230307220236-3a3c6141e376/go.mod h1:an3vInlBmSxCcxctByoQdvwPiA7DTK7jaaFDBTtu0ic=
 | 
			
		||||
github.com/go-git/go-billy/v5 v5.6.2 h1:6Q86EsPXMa7c3YZ3aLAQsMA0VlWmy43r6FHqa/UNbRM=
 | 
			
		||||
github.com/go-git/go-billy/v5 v5.6.2/go.mod h1:rcFC2rAsp/erv7CMz9GczHcuD0D32fWzH+MJAU+jaUU=
 | 
			
		||||
github.com/go-git/go-git-fixtures/v4 v4.3.2-0.20231010084843-55a94097c399 h1:eMje31YglSBqCdIqdhKBW8lokaMrL3uTkpGYlE2OOT4=
 | 
			
		||||
github.com/go-git/go-git-fixtures/v4 v4.3.2-0.20231010084843-55a94097c399/go.mod h1:1OCfN199q1Jm3HZlxleg+Dw/mwps2Wbk9frAWm+4FII=
 | 
			
		||||
github.com/go-git/go-git/v5 v5.16.2 h1:fT6ZIOjE5iEnkzKyxTHK1W4HGAsPhqEqiSAssSO77hM=
 | 
			
		||||
github.com/go-git/go-git/v5 v5.16.2/go.mod h1:4Ge4alE/5gPs30F2H1esi2gPd69R0C39lolkucHBOp8=
 | 
			
		||||
github.com/go-gorp/gorp/v3 v3.1.0 h1:ItKF/Vbuj31dmV4jxA1qblpSwkl9g1typ24xoe70IGs=
 | 
			
		||||
github.com/go-gorp/gorp/v3 v3.1.0/go.mod h1:dLEjIyyRNiXvNZ8PSmzpt1GsWAUK8kjVhEpjH8TixEw=
 | 
			
		||||
github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
 | 
			
		||||
github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY=
 | 
			
		||||
github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
 | 
			
		||||
github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag=
 | 
			
		||||
github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE=
 | 
			
		||||
github.com/go-logr/zapr v1.3.0 h1:XGdV8XW8zdwFiwOA2Dryh1gj2KRQyOOoNmBy4EplIcQ=
 | 
			
		||||
github.com/go-logr/zapr v1.3.0/go.mod h1:YKepepNBd1u/oyhd/yQmtjVXmm9uML4IXUgMOwR8/Gg=
 | 
			
		||||
github.com/go-openapi/jsonpointer v0.19.6/go.mod h1:osyAmYz/mB/C3I+WsTTSgw1ONzaLJoLCyoi6/zppojs=
 | 
			
		||||
github.com/go-openapi/jsonpointer v0.21.0 h1:YgdVicSA9vH5RiHs9TZW5oyafXZFc6+2Vc1rr/O9oNQ=
 | 
			
		||||
github.com/go-openapi/jsonpointer v0.21.0/go.mod h1:IUyH9l/+uyhIYQ/PXVA41Rexl+kOkAPDdXEYns6fzUY=
 | 
			
		||||
github.com/go-openapi/jsonreference v0.20.2 h1:3sVjiK66+uXK/6oQ8xgcRKcFgQ5KXa2KvnJRumpMGbE=
 | 
			
		||||
github.com/go-openapi/jsonreference v0.20.2/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k=
 | 
			
		||||
github.com/go-openapi/swag v0.22.3/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14=
 | 
			
		||||
github.com/go-openapi/swag v0.23.0 h1:vsEVJDUo2hPJ2tu0/Xc+4noaxyEffXNIs3cOULZ+GrE=
 | 
			
		||||
github.com/go-openapi/swag v0.23.0/go.mod h1:esZ8ITTYEsH1V2trKHjAN8Ai7xHb8RV+YSZ577vPjgQ=
 | 
			
		||||
github.com/go-sql-driver/mysql v1.8.1 h1:LedoTUt/eveggdHS9qUFC1EFSa8bU2+1pZjSRpvNJ1Y=
 | 
			
		||||
github.com/go-sql-driver/mysql v1.8.1/go.mod h1:wEBSXgmK//2ZFJyE+qWnIsVGmvmEKlqwuVSjsCm7DZg=
 | 
			
		||||
github.com/go-task/slim-sprig/v3 v3.0.0 h1:sUs3vkvUymDpBKi3qH1YSqBQk9+9D/8M2mN1vB6EwHI=
 | 
			
		||||
github.com/go-task/slim-sprig/v3 v3.0.0/go.mod h1:W848ghGpv3Qj3dhTPRyJypKRiqCdHZiAzKg9hl15HA8=
 | 
			
		||||
github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y=
 | 
			
		||||
github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
 | 
			
		||||
github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q=
 | 
			
		||||
github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
 | 
			
		||||
github.com/golang/groupcache v0.0.0-20241129210726-2c02b8208cf8 h1:f+oWsMOmNPc8JmEHVZIycC7hBoQxHH9pNKQORJNozsQ=
 | 
			
		||||
github.com/golang/groupcache v0.0.0-20241129210726-2c02b8208cf8/go.mod h1:wcDNUvekVysuuOpQKo3191zZyTpiI6se1N1ULghS0sw=
 | 
			
		||||
github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
 | 
			
		||||
github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
 | 
			
		||||
github.com/google/btree v1.1.3 h1:CVpQJjYgC4VbzxeGVHfvZrv1ctoYCAI8vbl07Fcxlyg=
 | 
			
		||||
github.com/google/btree v1.1.3/go.mod h1:qOPhT0dTNdNzV6Z/lhRX0YXUafgPLFUh+gZMl761Gm4=
 | 
			
		||||
github.com/google/cel-go v0.23.2 h1:UdEe3CvQh3Nv+E/j9r1Y//WO0K0cSyD7/y0bzyLIMI4=
 | 
			
		||||
github.com/google/cel-go v0.23.2/go.mod h1:52Pb6QsDbC5kvgxvZhiL9QX1oZEkcUF/ZqaPx1J5Wwo=
 | 
			
		||||
github.com/google/gnostic-models v0.6.9 h1:MU/8wDLif2qCXZmzncUQ/BOfxWfthHi63KqpoNbWqVw=
 | 
			
		||||
github.com/google/gnostic-models v0.6.9/go.mod h1:CiWsm0s6BSQd1hRn8/QmxqB6BesYcbSZxsz9b0KuDBw=
 | 
			
		||||
github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
 | 
			
		||||
github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8=
 | 
			
		||||
github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU=
 | 
			
		||||
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
 | 
			
		||||
github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0=
 | 
			
		||||
github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
 | 
			
		||||
github.com/google/pprof v0.0.0-20241029153458-d1b30febd7db h1:097atOisP2aRj7vFgYQBbFN4U4JNXUNYpxael3UzMyo=
 | 
			
		||||
github.com/google/pprof v0.0.0-20241029153458-d1b30febd7db/go.mod h1:vavhavw2zAxS5dIdcRluK6cSGGPlZynqzFM8NdvU144=
 | 
			
		||||
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 h1:El6M4kTTCOh6aBiKaUGG7oYTSPP8MxqL4YI3kZKwcP4=
 | 
			
		||||
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ=
 | 
			
		||||
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
 | 
			
		||||
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
 | 
			
		||||
github.com/gorilla/handlers v1.5.2 h1:cLTUSsNkgcwhgRqvCNmdbRWG0A3N4F+M2nWKdScwyEE=
 | 
			
		||||
github.com/gorilla/handlers v1.5.2/go.mod h1:dX+xVpaxdSw+q0Qek8SSsl3dfMk3jNddUkMzo0GtH0w=
 | 
			
		||||
github.com/gorilla/mux v1.8.1 h1:TuBL49tXwgrFYWhqrNgrUNEY92u81SPhu7sTdzQEiWY=
 | 
			
		||||
github.com/gorilla/mux v1.8.1/go.mod h1:AKf9I4AEqPTmMytcMc0KkNouC66V3BtZ4qD5fmWSiMQ=
 | 
			
		||||
github.com/gorilla/websocket v1.5.4-0.20250319132907-e064f32e3674 h1:JeSE6pjso5THxAzdVpqr6/geYxZytqFMBCOtn/ujyeo=
 | 
			
		||||
github.com/gorilla/websocket v1.5.4-0.20250319132907-e064f32e3674/go.mod h1:r4w70xmWCQKmi1ONH4KIaBptdivuRPyosB9RmPlGEwA=
 | 
			
		||||
github.com/gosuri/uitable v0.0.4 h1:IG2xLKRvErL3uhY6e1BylFzG+aJiwQviDDTfOKeKTpY=
 | 
			
		||||
github.com/gosuri/uitable v0.0.4/go.mod h1:tKR86bXuXPZazfOTG1FIzvjIdXzd0mo4Vtn16vt0PJo=
 | 
			
		||||
github.com/gregjones/httpcache v0.0.0-20190611155906-901d90724c79 h1:+ngKgrYPPJrOjhax5N+uePQ0Fh1Z7PheYoUI/0nzkPA=
 | 
			
		||||
github.com/gregjones/httpcache v0.0.0-20190611155906-901d90724c79/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA=
 | 
			
		||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.24.0 h1:TmHmbvxPmaegwhDubVz0lICL0J5Ka2vwTzhoePEXsGE=
 | 
			
		||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.24.0/go.mod h1:qztMSjm835F2bXf+5HKAPIS5qsmQDqZna/PgVt4rWtI=
 | 
			
		||||
github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
 | 
			
		||||
github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY2I=
 | 
			
		||||
github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
 | 
			
		||||
github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo=
 | 
			
		||||
github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM=
 | 
			
		||||
github.com/hashicorp/golang-lru/arc/v2 v2.0.5 h1:l2zaLDubNhW4XO3LnliVj0GXO3+/CGNJAg1dcN2Fpfw=
 | 
			
		||||
github.com/hashicorp/golang-lru/arc/v2 v2.0.5/go.mod h1:ny6zBSQZi2JxIeYcv7kt2sH2PXJtirBN7RDhRpxPkxU=
 | 
			
		||||
github.com/hashicorp/golang-lru/v2 v2.0.5 h1:wW7h1TG88eUIJ2i69gaE3uNVtEPIagzhGvHgwfx2Vm4=
 | 
			
		||||
github.com/hashicorp/golang-lru/v2 v2.0.5/go.mod h1:QeFd9opnmA6QUJc5vARoKUSoFhyfM2/ZepoAG6RGpeM=
 | 
			
		||||
github.com/huandu/xstrings v1.5.0 h1:2ag3IFq9ZDANvthTwTiqSSZLjDc+BedvHPAp5tJy2TI=
 | 
			
		||||
github.com/huandu/xstrings v1.5.0/go.mod h1:y5/lhBue+AyNmUVz9RLU9xbLR0o4KIIExikq4ovT0aE=
 | 
			
		||||
github.com/imdario/mergo v0.3.13 h1:lFzP57bqS/wsqKssCGmtLAb8A0wKjLGrve2q3PPVcBk=
 | 
			
		||||
github.com/imdario/mergo v0.3.13/go.mod h1:4lJ1jqUDcsbIECGy0RUJAXNIhg+6ocWgb1ALK2O4oXg=
 | 
			
		||||
github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
 | 
			
		||||
github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
 | 
			
		||||
github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 h1:BQSFePA1RWJOlocH6Fxy8MmwDt+yVQYULKfN0RoTN8A=
 | 
			
		||||
github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99/go.mod h1:1lJo3i6rXxKeerYnT8Nvf0QmHCRC1n8sfWVwXF2Frvo=
 | 
			
		||||
github.com/jmoiron/sqlx v1.4.0 h1:1PLqN7S1UYp5t4SrVVnt4nUVNemrDAtxlulVe+Qgm3o=
 | 
			
		||||
github.com/jmoiron/sqlx v1.4.0/go.mod h1:ZrZ7UsYB/weZdl2Bxg6jCRO9c3YHl8r3ahlKmRT4JLY=
 | 
			
		||||
github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
 | 
			
		||||
github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y=
 | 
			
		||||
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
 | 
			
		||||
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
 | 
			
		||||
github.com/kevinburke/ssh_config v1.2.0 h1:x584FjTGwHzMwvHx18PXxbBVzfnxogHaAReU4gf13a4=
 | 
			
		||||
github.com/kevinburke/ssh_config v1.2.0/go.mod h1:CT57kijsi8u/K/BOFA39wgDQJ9CxiF4nAY/ojJ6r6mM=
 | 
			
		||||
github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
 | 
			
		||||
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
 | 
			
		||||
github.com/klauspost/compress v1.18.0 h1:c/Cqfb0r+Yi+JtIEq73FWXVkRonBlf0CRNYc8Zttxdo=
 | 
			
		||||
github.com/klauspost/compress v1.18.0/go.mod h1:2Pp+KzxcywXVXMr50+X0Q/Lsb43OQHYWRCY2AiWywWQ=
 | 
			
		||||
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
 | 
			
		||||
github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
 | 
			
		||||
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
 | 
			
		||||
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
 | 
			
		||||
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
 | 
			
		||||
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
 | 
			
		||||
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
 | 
			
		||||
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
 | 
			
		||||
github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc=
 | 
			
		||||
github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw=
 | 
			
		||||
github.com/lann/builder v0.0.0-20180802200727-47ae307949d0 h1:SOEGU9fKiNWd/HOJuq6+3iTQz8KNCLtVX6idSoTLdUw=
 | 
			
		||||
github.com/lann/builder v0.0.0-20180802200727-47ae307949d0/go.mod h1:dXGbAdH5GtBTC4WfIxhKZfyBF/HBFgRZSWwZ9g/He9o=
 | 
			
		||||
github.com/lann/ps v0.0.0-20150810152359-62de8c46ede0 h1:P6pPBnrTSX3DEVR4fDembhRWSsG5rVo6hYhAB/ADZrk=
 | 
			
		||||
github.com/lann/ps v0.0.0-20150810152359-62de8c46ede0/go.mod h1:vmVJ0l/dxyfGW6FmdpVm2joNMFikkuWg0EoCKLGUMNw=
 | 
			
		||||
github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw=
 | 
			
		||||
github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
 | 
			
		||||
github.com/liggitt/tabwriter v0.0.0-20181228230101-89fcab3d43de h1:9TO3cAIGXtEhnIaL+V+BEER86oLrvS+kWobKpbJuye0=
 | 
			
		||||
github.com/liggitt/tabwriter v0.0.0-20181228230101-89fcab3d43de/go.mod h1:zAbeS9B/r2mtpb6U+EI2rYA5OAXxsYw6wTamcNW+zcE=
 | 
			
		||||
github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0=
 | 
			
		||||
github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc=
 | 
			
		||||
github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc=
 | 
			
		||||
github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA=
 | 
			
		||||
github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg=
 | 
			
		||||
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
 | 
			
		||||
github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94=
 | 
			
		||||
github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
 | 
			
		||||
github.com/mattn/go-isatty v0.0.17 h1:BTarxUcIeDqL27Mc+vyvdWYSL28zpIhv3RoTdsLMPng=
 | 
			
		||||
github.com/mattn/go-isatty v0.0.17/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
 | 
			
		||||
github.com/mattn/go-runewidth v0.0.9 h1:Lm995f3rfxdpd6TSmuVCHVb/QhupuXlYr8sCI/QdE+0=
 | 
			
		||||
github.com/mattn/go-runewidth v0.0.9/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI=
 | 
			
		||||
github.com/mattn/go-sqlite3 v1.14.22 h1:2gZY6PC6kBnID23Tichd1K+Z0oS6nE/XwU+Vz/5o4kU=
 | 
			
		||||
github.com/mattn/go-sqlite3 v1.14.22/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
 | 
			
		||||
github.com/miekg/dns v1.1.57 h1:Jzi7ApEIzwEPLHWRcafCN9LZSBbqQpxjt/wpgvg7wcM=
 | 
			
		||||
github.com/miekg/dns v1.1.57/go.mod h1:uqRjCRUuEAA6qsOiJvDd+CFo/vW+y5WR6SNmHE55hZk=
 | 
			
		||||
github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw=
 | 
			
		||||
github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s=
 | 
			
		||||
github.com/mitchellh/go-wordwrap v1.0.1 h1:TLuKupo69TCn6TQSyGxwI1EblZZEsQ0vMlAFQflz0v0=
 | 
			
		||||
github.com/mitchellh/go-wordwrap v1.0.1/go.mod h1:R62XHJLzvMFRBbcrT7m7WgmE1eOyTSsCt+hzestvNj0=
 | 
			
		||||
github.com/mitchellh/reflectwalk v1.0.2 h1:G2LzWKi524PWgd3mLHV8Y5k7s6XUvT0Gef6zxSIeXaQ=
 | 
			
		||||
github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw=
 | 
			
		||||
github.com/moby/spdystream v0.5.0 h1:7r0J1Si3QO/kjRitvSLVVFUjxMEb/YLj6S9FF62JBCU=
 | 
			
		||||
github.com/moby/spdystream v0.5.0/go.mod h1:xBAYlnt/ay+11ShkdFKNAG7LsyK/tmNBVvVOwrfMgdI=
 | 
			
		||||
github.com/moby/term v0.5.2 h1:6qk3FJAFDs6i/q3W/pQ97SX192qKfZgGjCQqfCJkgzQ=
 | 
			
		||||
github.com/moby/term v0.5.2/go.mod h1:d3djjFCrjnB+fl8NJux+EJzu0msscUP+f8it8hPkFLc=
 | 
			
		||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
 | 
			
		||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
 | 
			
		||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
 | 
			
		||||
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
 | 
			
		||||
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
 | 
			
		||||
github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00 h1:n6/2gBQ3RWajuToeY6ZtZTIKv2v7ThUy5KKusIT0yc0=
 | 
			
		||||
github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00/go.mod h1:Pm3mSP3c5uWn86xMLZ5Sa7JB9GsEZySvHYXCTK4E9q4=
 | 
			
		||||
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA=
 | 
			
		||||
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ=
 | 
			
		||||
github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f h1:y5//uYreIhSUg3J1GEMiLbxo1LJaP8RfCpH6pymGZus=
 | 
			
		||||
github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f/go.mod h1:ZdcZmHo+o7JKHSa8/e818NopupXU1YMK5fe1lsApnBw=
 | 
			
		||||
github.com/onsi/ginkgo/v2 v2.22.0 h1:Yed107/8DjTr0lKCNt7Dn8yQ6ybuDRQoMGrNFKzMfHg=
 | 
			
		||||
github.com/onsi/ginkgo/v2 v2.22.0/go.mod h1:7Du3c42kxCUegi0IImZ1wUQzMBVecgIHjR1C+NkhLQo=
 | 
			
		||||
github.com/onsi/gomega v1.36.1 h1:bJDPBO7ibjxcbHMgSCoo4Yj18UWbKDlLwX1x9sybDcw=
 | 
			
		||||
github.com/onsi/gomega v1.36.1/go.mod h1:PvZbdDc8J6XJEpDK4HCuRBm8a6Fzp9/DmhC9C7yFlog=
 | 
			
		||||
github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U=
 | 
			
		||||
github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM=
 | 
			
		||||
github.com/opencontainers/image-spec v1.1.1 h1:y0fUlFfIZhPF1W537XOLg0/fcx6zcHCJwooC2xJA040=
 | 
			
		||||
github.com/opencontainers/image-spec v1.1.1/go.mod h1:qpqAh3Dmcf36wStyyWU+kCeDgrGnAve2nCC8+7h8Q0M=
 | 
			
		||||
github.com/peterbourgon/diskv v2.0.1+incompatible h1:UBdAOUP5p4RWqPBg048CAvpKN+vxiaj6gdUUzhl4XmI=
 | 
			
		||||
github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU=
 | 
			
		||||
github.com/phayes/freeport v0.0.0-20220201140144-74d24b5ae9f5 h1:Ii+DKncOVM8Cu1Hc+ETb5K+23HdAMvESYE3ZJ5b5cMI=
 | 
			
		||||
github.com/phayes/freeport v0.0.0-20220201140144-74d24b5ae9f5/go.mod h1:iIss55rKnNBTvrwdmkUpLnDpZoAHvWaiq5+iMmen4AE=
 | 
			
		||||
github.com/pjbgf/sha1cd v0.3.2 h1:a9wb0bp1oC2TGwStyn0Umc/IGKQnEgF0vVaZ8QF8eo4=
 | 
			
		||||
github.com/pjbgf/sha1cd v0.3.2/go.mod h1:zQWigSxVmsHEZow5qaLtPYxpcKMMQpa09ixqBxuCS6A=
 | 
			
		||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
 | 
			
		||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
 | 
			
		||||
github.com/poy/onpar v1.1.2 h1:QaNrNiZx0+Nar5dLgTVp5mXkyoVFIbepjyEoGSnhbAY=
 | 
			
		||||
github.com/poy/onpar v1.1.2/go.mod h1:6X8FLNoxyr9kkmnlqpK6LSoiOtrO6MICtWwEuWkLjzg=
 | 
			
		||||
github.com/prometheus/client_golang v1.22.0 h1:rb93p9lokFEsctTys46VnV1kLCDpVZ0a/Y92Vm0Zc6Q=
 | 
			
		||||
github.com/prometheus/client_golang v1.22.0/go.mod h1:R7ljNsLXhuQXYZYtw6GAE9AZg8Y7vEW5scdCXrWRXC0=
 | 
			
		||||
github.com/prometheus/client_model v0.6.1 h1:ZKSh/rekM+n3CeS952MLRAdFwIKqeY8b62p8ais2e9E=
 | 
			
		||||
github.com/prometheus/client_model v0.6.1/go.mod h1:OrxVMOVHjw3lKMa8+x6HeMGkHMQyHDk9E3jmP2AmGiY=
 | 
			
		||||
github.com/prometheus/common v0.62.0 h1:xasJaQlnWAeyHdUBeGjXmutelfJHWMRr+Fg4QszZ2Io=
 | 
			
		||||
github.com/prometheus/common v0.62.0/go.mod h1:vyBcEuLSvWos9B1+CyL7JZ2up+uFzXhkqml0W5zIY1I=
 | 
			
		||||
github.com/prometheus/procfs v0.15.1 h1:YagwOFzUgYfKKHX6Dr+sHT7km/hxC76UB0learggepc=
 | 
			
		||||
github.com/prometheus/procfs v0.15.1/go.mod h1:fB45yRUv8NstnjriLhBQLuOUt+WW4BsoGhij/e3PBqk=
 | 
			
		||||
github.com/redis/go-redis/extra/rediscmd/v9 v9.0.5 h1:EaDatTxkdHG+U3Bk4EUr+DZ7fOGwTfezUiUJMaIcaho=
 | 
			
		||||
github.com/redis/go-redis/extra/rediscmd/v9 v9.0.5/go.mod h1:fyalQWdtzDBECAQFBJuQe5bzQ02jGd5Qcbgb97Flm7U=
 | 
			
		||||
github.com/redis/go-redis/extra/redisotel/v9 v9.0.5 h1:EfpWLLCyXw8PSM2/XNJLjI3Pb27yVE+gIAfeqp8LUCc=
 | 
			
		||||
github.com/redis/go-redis/extra/redisotel/v9 v9.0.5/go.mod h1:WZjPDy7VNzn77AAfnAfVjZNvfJTYfPetfZk5yoSTLaQ=
 | 
			
		||||
github.com/redis/go-redis/v9 v9.7.3 h1:YpPyAayJV+XErNsatSElgRZZVCwXX9QzkKYNvO7x0wM=
 | 
			
		||||
github.com/redis/go-redis/v9 v9.7.3/go.mod h1:bGUrSggJ9X9GUmZpZNEOQKaANxSGgOEBRltRTZHSvrA=
 | 
			
		||||
github.com/rogpeppe/go-internal v1.14.1 h1:UQB4HGPB6osV0SQTLymcB4TgvyWu6ZyliaW0tI/otEQ=
 | 
			
		||||
github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc=
 | 
			
		||||
github.com/rubenv/sql-migrate v1.8.0 h1:dXnYiJk9k3wetp7GfQbKJcPHjVJL6YK19tKj8t2Ns0o=
 | 
			
		||||
github.com/rubenv/sql-migrate v1.8.0/go.mod h1:F2bGFBwCU+pnmbtNYDeKvSuvL6lBVtXDXUUv5t+u1qw=
 | 
			
		||||
github.com/russross/blackfriday/v2 v2.1.0 h1:JIOH55/0cWyOuilr9/qlrm0BSXldqnqwMsf35Ld67mk=
 | 
			
		||||
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
 | 
			
		||||
github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 h1:n661drycOFuPLCN3Uc8sB6B/s6Z4t2xvBgU1htSHuq8=
 | 
			
		||||
github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3/go.mod h1:A0bzQcvG0E7Rwjx0REVgAGH58e96+X0MeOfepqsbeW4=
 | 
			
		||||
github.com/shopspring/decimal v1.4.0 h1:bxl37RwXBklmTi0C79JfXCEBD1cqqHt0bbgBAGFp81k=
 | 
			
		||||
github.com/shopspring/decimal v1.4.0/go.mod h1:gawqmDU56v4yIKSwfBSFip1HdCCXN8/+DMd9qYNcwME=
 | 
			
		||||
github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0=
 | 
			
		||||
github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ=
 | 
			
		||||
github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
 | 
			
		||||
github.com/skeema/knownhosts v1.3.1 h1:X2osQ+RAjK76shCbvhHHHVl3ZlgDm8apHEHFqRjnBY8=
 | 
			
		||||
github.com/skeema/knownhosts v1.3.1/go.mod h1:r7KTdC8l4uxWRyK2TpQZ/1o5HaSzh06ePQNxPwTcfiY=
 | 
			
		||||
github.com/spf13/cast v1.7.0 h1:ntdiHjuueXFgm5nzDRdOS4yfT43P5Fnud6DH50rz/7w=
 | 
			
		||||
github.com/spf13/cast v1.7.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo=
 | 
			
		||||
github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo=
 | 
			
		||||
github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0=
 | 
			
		||||
github.com/spf13/pflag v1.0.6 h1:jFzHGLGAlb3ruxLB8MhbI6A8+AQX/2eW4qeyNZXNp2o=
 | 
			
		||||
github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
 | 
			
		||||
github.com/stoewer/go-strcase v1.3.0 h1:g0eASXYtp+yvN9fK8sH94oCIk0fau9uV1/ZdJ0AVEzs=
 | 
			
		||||
github.com/stoewer/go-strcase v1.3.0/go.mod h1:fAH5hQ5pehh+j3nZfvwdk2RgEgQjAoM8wodgtPmh1xo=
 | 
			
		||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
 | 
			
		||||
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
 | 
			
		||||
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
 | 
			
		||||
github.com/stretchr/objx v0.5.2 h1:xuMeJ0Sdp5ZMRXx/aWO6RZxdr3beISkG5/G/aIRr3pY=
 | 
			
		||||
github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA=
 | 
			
		||||
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
 | 
			
		||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
 | 
			
		||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
 | 
			
		||||
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
 | 
			
		||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
 | 
			
		||||
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
 | 
			
		||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
 | 
			
		||||
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
 | 
			
		||||
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
 | 
			
		||||
github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
 | 
			
		||||
github.com/x448/float16 v0.8.4 h1:qLwI1I70+NjRFUR3zs1JPUCgaCXSh3SW62uAKT1mSBM=
 | 
			
		||||
github.com/x448/float16 v0.8.4/go.mod h1:14CWIYCyZA/cWjXOioeEpHeN/83MdbZDRQHoFcYsOfg=
 | 
			
		||||
github.com/xanzy/ssh-agent v0.3.3 h1:+/15pJfg/RsTxqYcX6fHqOXZwwMP+2VyYWJeWM2qQFM=
 | 
			
		||||
github.com/xanzy/ssh-agent v0.3.3/go.mod h1:6dzNDKs0J9rVPHPhaGCukekBHKqfl+L3KghI1Bc68Uw=
 | 
			
		||||
github.com/xeipuuv/gojsonpointer v0.0.0-20180127040702-4e3ac2762d5f/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
 | 
			
		||||
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb h1:zGWFAtiMcyryUHoUjUJX0/lt1H2+i2Ka2n+D3DImSNo=
 | 
			
		||||
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
 | 
			
		||||
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 h1:EzJWgHovont7NscjpAxXsDA8S8BMYve8Y5+7cuRE7R0=
 | 
			
		||||
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ=
 | 
			
		||||
github.com/xeipuuv/gojsonschema v1.2.0 h1:LhYJRs+L4fBtjZUfuSZIKGeVu0QRy8e5Xi7D17UxZ74=
 | 
			
		||||
github.com/xeipuuv/gojsonschema v1.2.0/go.mod h1:anYRn/JVcOK2ZgGU+IjEV4nwlhoK5sQluxsYJ78Id3Y=
 | 
			
		||||
github.com/xlab/treeprint v1.2.0 h1:HzHnuAF1plUN2zGlAFHbSQP2qJ0ZAD3XF5XD7OesXRQ=
 | 
			
		||||
github.com/xlab/treeprint v1.2.0/go.mod h1:gj5Gd3gPdKtR1ikdDK6fnFLdmIS0X30kTTuNd/WEJu0=
 | 
			
		||||
github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
 | 
			
		||||
github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
 | 
			
		||||
go.opentelemetry.io/auto/sdk v1.1.0 h1:cH53jehLUN6UFLY71z+NDOiNJqDdPRaXzTel0sJySYA=
 | 
			
		||||
go.opentelemetry.io/auto/sdk v1.1.0/go.mod h1:3wSPjt5PWp2RhlCcmmOial7AvC4DQqZb7a7wCow3W8A=
 | 
			
		||||
go.opentelemetry.io/contrib/bridges/prometheus v0.57.0 h1:UW0+QyeyBVhn+COBec3nGhfnFe5lwB0ic1JBVjzhk0w=
 | 
			
		||||
go.opentelemetry.io/contrib/bridges/prometheus v0.57.0/go.mod h1:ppciCHRLsyCio54qbzQv0E4Jyth/fLWDTJYfvWpcSVk=
 | 
			
		||||
go.opentelemetry.io/contrib/exporters/autoexport v0.57.0 h1:jmTVJ86dP60C01K3slFQa2NQ/Aoi7zA+wy7vMOKD9H4=
 | 
			
		||||
go.opentelemetry.io/contrib/exporters/autoexport v0.57.0/go.mod h1:EJBheUMttD/lABFyLXhce47Wr6DPWYReCzaZiXadH7g=
 | 
			
		||||
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.58.0 h1:yd02MEjBdJkG3uabWP9apV+OuWRIXGDuJEUJbOHmCFU=
 | 
			
		||||
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.58.0/go.mod h1:umTcuxiv1n/s/S6/c2AT/g2CQ7u5C59sHDNmfSwgz7Q=
 | 
			
		||||
go.opentelemetry.io/otel v1.33.0 h1:/FerN9bax5LoK51X/sI0SVYrjSE0/yUL7DpxW4K3FWw=
 | 
			
		||||
go.opentelemetry.io/otel v1.33.0/go.mod h1:SUUkR6csvUQl+yjReHu5uM3EtVV7MBm5FHKRlNx4I8I=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploggrpc v0.8.0 h1:WzNab7hOOLzdDF/EoWCt4glhrbMPVMOO5JYTmpz36Ls=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploggrpc v0.8.0/go.mod h1:hKvJwTzJdp90Vh7p6q/9PAOd55dI6WA6sWj62a/JvSs=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp v0.8.0 h1:S+LdBGiQXtJdowoJoQPEtI52syEP/JYBUpjO49EQhV8=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp v0.8.0/go.mod h1:5KXybFvPGds3QinJWQT7pmXf+TN5YIa7CNYObWRkj50=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.32.0 h1:j7ZSD+5yn+lo3sGV69nW04rRR0jhYnBwjuX3r0HvnK0=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.32.0/go.mod h1:WXbYJTUaZXAbYd8lbgGuvih0yuCfOFC5RJoYnoLcGz8=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp v1.32.0 h1:t/Qur3vKSkUCcDVaSumWF2PKHt85pc7fRvFuoVT8qFU=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp v1.32.0/go.mod h1:Rl61tySSdcOJWoEgYZVtmnKdA0GeKrSqkHC1t+91CH8=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.33.0 h1:Vh5HayB/0HHfOQA7Ctx69E/Y/DcQSMPpKANYVMQ7fBA=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.33.0/go.mod h1:cpgtDBaqD/6ok/UG0jT15/uKjAY8mRA53diogHBg3UI=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.33.0 h1:5pojmb1U1AogINhN3SurB+zm/nIcusopeBNp42f45QM=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.33.0/go.mod h1:57gTHJSE5S1tqg+EKsLPlTWhpHMsWlVmer+LA926XiA=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.32.0 h1:cMyu9O88joYEaI47CnQkxO1XZdpoTF9fEnW2duIddhw=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.32.0/go.mod h1:6Am3rn7P9TVVeXYG+wtcGE7IE1tsQ+bP3AuWcKt/gOI=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/prometheus v0.54.0 h1:rFwzp68QMgtzu9PgP3jm9XaMICI6TsofWWPcBDKwlsU=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/prometheus v0.54.0/go.mod h1:QyjcV9qDP6VeK5qPyKETvNjmaaEc7+gqjh4SS0ZYzDU=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/stdout/stdoutlog v0.8.0 h1:CHXNXwfKWfzS65yrlB2PVds1IBZcdsX8Vepy9of0iRU=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/stdout/stdoutlog v0.8.0/go.mod h1:zKU4zUgKiaRxrdovSS2amdM5gOc59slmo/zJwGX+YBg=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.32.0 h1:SZmDnHcgp3zwlPBS2JX2urGYe/jBKEIT6ZedHRUyCz8=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.32.0/go.mod h1:fdWW0HtZJ7+jNpTKUR0GpMEDP69nR8YBJQxNiVCE3jk=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/stdout/stdouttrace v1.32.0 h1:cC2yDI3IQd0Udsux7Qmq8ToKAx1XCilTQECZ0KDZyTw=
 | 
			
		||||
go.opentelemetry.io/otel/exporters/stdout/stdouttrace v1.32.0/go.mod h1:2PD5Ex6z8CFzDbTdOlwyNIUywRr1DN0ospafJM1wJ+s=
 | 
			
		||||
go.opentelemetry.io/otel/log v0.8.0 h1:egZ8vV5atrUWUbnSsHn6vB8R21G2wrKqNiDt3iWertk=
 | 
			
		||||
go.opentelemetry.io/otel/log v0.8.0/go.mod h1:M9qvDdUTRCopJcGRKg57+JSQ9LgLBrwwfC32epk5NX8=
 | 
			
		||||
go.opentelemetry.io/otel/metric v1.33.0 h1:r+JOocAyeRVXD8lZpjdQjzMadVZp2M4WmQ+5WtEnklQ=
 | 
			
		||||
go.opentelemetry.io/otel/metric v1.33.0/go.mod h1:L9+Fyctbp6HFTddIxClbQkjtubW6O9QS3Ann/M82u6M=
 | 
			
		||||
go.opentelemetry.io/otel/sdk v1.33.0 h1:iax7M131HuAm9QkZotNHEfstof92xM+N8sr3uHXc2IM=
 | 
			
		||||
go.opentelemetry.io/otel/sdk v1.33.0/go.mod h1:A1Q5oi7/9XaMlIWzPSxLRWOI8nG3FnzHJNbiENQuihM=
 | 
			
		||||
go.opentelemetry.io/otel/sdk/log v0.8.0 h1:zg7GUYXqxk1jnGF/dTdLPrK06xJdrXgqgFLnI4Crxvs=
 | 
			
		||||
go.opentelemetry.io/otel/sdk/log v0.8.0/go.mod h1:50iXr0UVwQrYS45KbruFrEt4LvAdCaWWgIrsN3ZQggo=
 | 
			
		||||
go.opentelemetry.io/otel/sdk/metric v1.32.0 h1:rZvFnvmvawYb0alrYkjraqJq0Z4ZUJAiyYCU9snn1CU=
 | 
			
		||||
go.opentelemetry.io/otel/sdk/metric v1.32.0/go.mod h1:PWeZlq0zt9YkYAp3gjKZ0eicRYvOh1Gd+X99x6GHpCQ=
 | 
			
		||||
go.opentelemetry.io/otel/trace v1.33.0 h1:cCJuF7LRjUFso9LPnEAHJDB2pqzp+hbO8eu1qqW2d/s=
 | 
			
		||||
go.opentelemetry.io/otel/trace v1.33.0/go.mod h1:uIcdVUZMpTAmz0tI1z04GoVSezK37CbGV4fr1f2nBck=
 | 
			
		||||
go.opentelemetry.io/proto/otlp v1.4.0 h1:TA9WRvW6zMwP+Ssb6fLoUIuirti1gGbP28GcKG1jgeg=
 | 
			
		||||
go.opentelemetry.io/proto/otlp v1.4.0/go.mod h1:PPBWZIP98o2ElSqI35IHfu7hIhSwvc5N38Jw8pXuGFY=
 | 
			
		||||
go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
 | 
			
		||||
go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
 | 
			
		||||
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
 | 
			
		||||
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
 | 
			
		||||
go.uber.org/zap v1.27.0 h1:aJMhYGrd5QSmlpLMr2MftRKl7t8J8PTZPA732ud/XR8=
 | 
			
		||||
go.uber.org/zap v1.27.0/go.mod h1:GB2qFLM7cTU87MWRP2mPIjqfIDnGu+VIO4V/SdhGo2E=
 | 
			
		||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
 | 
			
		||||
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
 | 
			
		||||
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
 | 
			
		||||
golang.org/x/crypto v0.0.0-20220622213112-05595931fe9d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
 | 
			
		||||
golang.org/x/crypto v0.39.0 h1:SHs+kF4LP+f+p14esP5jAoDpHU8Gu/v9lFRK6IT5imM=
 | 
			
		||||
golang.org/x/crypto v0.39.0/go.mod h1:L+Xg3Wf6HoL4Bn4238Z6ft6KfEpN0tJGo53AAPC632U=
 | 
			
		||||
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56 h1:2dVuKD2vS7b0QIHQbpyTISPd0LeHDbnYEryqj5Q1ug8=
 | 
			
		||||
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56/go.mod h1:M4RDyNAINzryxdtnbRXRL/OHtkFuWGRjvuhBJpk2IlY=
 | 
			
		||||
golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
 | 
			
		||||
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
 | 
			
		||||
golang.org/x/mod v0.25.0 h1:n7a+ZbQKQA/Ysbyb0/6IbB1H/X41mKgbhfv7AfG/44w=
 | 
			
		||||
golang.org/x/mod v0.25.0/go.mod h1:IXM97Txy2VM4PJ3gI61r1YEk/gAj6zAHN3AdZt6S9Ww=
 | 
			
		||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
 | 
			
		||||
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
 | 
			
		||||
golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
 | 
			
		||||
golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
 | 
			
		||||
golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
 | 
			
		||||
golang.org/x/net v0.40.0 h1:79Xs7wF06Gbdcg4kdCCIQArK11Z1hr5POQ6+fIYHNuY=
 | 
			
		||||
golang.org/x/net v0.40.0/go.mod h1:y0hY0exeL2Pku80/zKK7tpntoX23cqL3Oa6njdgRtds=
 | 
			
		||||
golang.org/x/oauth2 v0.28.0 h1:CrgCKl8PPAVtLnU3c+EDw6x11699EWlsDeWNWKdIOkc=
 | 
			
		||||
golang.org/x/oauth2 v0.28.0/go.mod h1:onh5ek6nERTohokkhCD/y2cV4Do3fxFHFuAejCkRWT8=
 | 
			
		||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 | 
			
		||||
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 | 
			
		||||
golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 | 
			
		||||
golang.org/x/sync v0.15.0 h1:KWH3jNZsfyT6xfAfKiz6MRNmd46ByHDYaZ7KSkCtdW8=
 | 
			
		||||
golang.org/x/sync v0.15.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA=
 | 
			
		||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
 | 
			
		||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.33.0 h1:q3i8TbbEz+JRD9ywIRlyRAQbM0qF7hu24q3teo2hbuw=
 | 
			
		||||
golang.org/x/sys v0.33.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
 | 
			
		||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
 | 
			
		||||
golang.org/x/term v0.32.0 h1:DR4lr0TjUs3epypdhTOkMmuF5CDFJ/8pOnbzMZPQ7bg=
 | 
			
		||||
golang.org/x/term v0.32.0/go.mod h1:uZG1FhGx848Sqfsq4/DlJr3xGGsYMu/L5GW4abiaEPQ=
 | 
			
		||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
 | 
			
		||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
 | 
			
		||||
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
 | 
			
		||||
golang.org/x/text v0.26.0 h1:P42AVeLghgTYr4+xUnTRKDMqpar+PtX7KWuNQL21L8M=
 | 
			
		||||
golang.org/x/text v0.26.0/go.mod h1:QK15LZJUUQVJxhz7wXgxSy/CJaTFjd0G+YLonydOVQA=
 | 
			
		||||
golang.org/x/time v0.9.0 h1:EsRrnYcQiGH+5FfbgvV4AP7qEZstoyrHB0DzarOQ4ZY=
 | 
			
		||||
golang.org/x/time v0.9.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM=
 | 
			
		||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
 | 
			
		||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
 | 
			
		||||
golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
 | 
			
		||||
golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
 | 
			
		||||
golang.org/x/tools v0.33.0 h1:4qz2S3zmRxbGIhDIAgjxvFutSvH5EfnsYrRBj0UI0bc=
 | 
			
		||||
golang.org/x/tools v0.33.0/go.mod h1:CIJMaWEY88juyUfo7UbgPqbC8rU2OqfAV1h2Qp0oMYI=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
gomodules.xyz/jsonpatch/v2 v2.4.0 h1:Ci3iUJyx9UeRx7CeFN8ARgGbkESwJK+KB9lLcWxY/Zw=
 | 
			
		||||
gomodules.xyz/jsonpatch/v2 v2.4.0/go.mod h1:AH3dM2RI6uoBZxn3LVrfvJ3E0/9dG4cSrbuBJT4moAY=
 | 
			
		||||
google.golang.org/genproto/googleapis/api v0.0.0-20241209162323-e6fa225c2576 h1:CkkIfIt50+lT6NHAVoRYEyAvQGFM7xEwXUUywFvEb3Q=
 | 
			
		||||
google.golang.org/genproto/googleapis/api v0.0.0-20241209162323-e6fa225c2576/go.mod h1:1R3kvZ1dtP3+4p4d3G8uJ8rFk/fWlScl38vanWACI08=
 | 
			
		||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20241209162323-e6fa225c2576 h1:8ZmaLZE4XWrtU3MyClkYqqtl6Oegr3235h7jxsDyqCY=
 | 
			
		||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20241209162323-e6fa225c2576/go.mod h1:5uTbfoYQed2U9p3KIj2/Zzm02PYhndfdmML0qC3q3FU=
 | 
			
		||||
google.golang.org/grpc v1.68.1 h1:oI5oTa11+ng8r8XMMN7jAOmWfPZWbYpCFaMUTACxkM0=
 | 
			
		||||
google.golang.org/grpc v1.68.1/go.mod h1:+q1XYFJjShcqn0QZHvCyeR4CXPA+llXIeUIfIe00waw=
 | 
			
		||||
google.golang.org/protobuf v1.36.5 h1:tPhr+woSbjfYvY6/GPufUoYizxw1cF/yFoxJ2fmpwlM=
 | 
			
		||||
google.golang.org/protobuf v1.36.5/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE=
 | 
			
		||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
 | 
			
		||||
gopkg.in/evanphx/json-patch.v4 v4.12.0 h1:n6jtcsulIzXPJaxegRbvFNNrZDjbij7ny3gmSPG+6V4=
 | 
			
		||||
gopkg.in/evanphx/json-patch.v4 v4.12.0/go.mod h1:p8EYWUEYMpynmqDbY58zCKCFZw8pRWMG4EsWvDvM72M=
 | 
			
		||||
gopkg.in/inf.v0 v0.9.1 h1:73M5CoZyi3ZLMOyDlQh031Cx6N9NDJ2Vvfl76EDAgDc=
 | 
			
		||||
gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw=
 | 
			
		||||
gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME=
 | 
			
		||||
gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI=
 | 
			
		||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
 | 
			
		||||
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
 | 
			
		||||
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.0/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
 | 
			
		||||
helm.sh/helm v2.17.0+incompatible h1:cSe3FaQOpRWLDXvTObQNj0P7WI98IG5yloU6tQVls2k=
 | 
			
		||||
helm.sh/helm v2.17.0+incompatible/go.mod h1:0Xbc6ErzwWH9qC55X1+hE3ZwhM3atbhCm/NbFZw5i+4=
 | 
			
		||||
helm.sh/helm/v3 v3.18.4 h1:pNhnHM3nAmDrxz6/UC+hfjDY4yeDATQCka2/87hkZXQ=
 | 
			
		||||
helm.sh/helm/v3 v3.18.4/go.mod h1:WVnwKARAw01iEdjpEkP7Ii1tT1pTPYfM1HsakFKM3LI=
 | 
			
		||||
honnef.co/go/tools v0.6.1 h1:R094WgE8K4JirYjBaOpz/AvTyUu/3wbmAoskKN/pxTI=
 | 
			
		||||
honnef.co/go/tools v0.6.1/go.mod h1:3puzxxljPCe8RGJX7BIy1plGbxEOZni5mR2aXe3/uk4=
 | 
			
		||||
k8s.io/api v0.33.2 h1:YgwIS5jKfA+BZg//OQhkJNIfie/kmRsO0BmNaVSimvY=
 | 
			
		||||
k8s.io/api v0.33.2/go.mod h1:fhrbphQJSM2cXzCWgqU29xLDuks4mu7ti9vveEnpSXs=
 | 
			
		||||
k8s.io/apiextensions-apiserver v0.33.2 h1:6gnkIbngnaUflR3XwE1mCefN3YS8yTD631JXQhsU6M8=
 | 
			
		||||
k8s.io/apiextensions-apiserver v0.33.2/go.mod h1:IvVanieYsEHJImTKXGP6XCOjTwv2LUMos0YWc9O+QP8=
 | 
			
		||||
k8s.io/apimachinery v0.33.2 h1:IHFVhqg59mb8PJWTLi8m1mAoepkUNYmptHsV+Z1m5jY=
 | 
			
		||||
k8s.io/apimachinery v0.33.2/go.mod h1:BHW0YOu7n22fFv/JkYOEfkUYNRN0fj0BlvMFWA7b+SM=
 | 
			
		||||
k8s.io/apiserver v0.33.2 h1:KGTRbxn2wJagJowo29kKBp4TchpO1DRO3g+dB/KOJN4=
 | 
			
		||||
k8s.io/apiserver v0.33.2/go.mod h1:9qday04wEAMLPWWo9AwqCZSiIn3OYSZacDyu/AcoM/M=
 | 
			
		||||
k8s.io/cli-runtime v0.33.2 h1:koNYQKSDdq5AExa/RDudXMhhtFasEg48KLS2KSAU74Y=
 | 
			
		||||
k8s.io/cli-runtime v0.33.2/go.mod h1:gnhsAWpovqf1Zj5YRRBBU7PFsRc6NkEkwYNQE+mXL88=
 | 
			
		||||
k8s.io/client-go v0.33.2 h1:z8CIcc0P581x/J1ZYf4CNzRKxRvQAwoAolYPbtQes+E=
 | 
			
		||||
k8s.io/client-go v0.33.2/go.mod h1:9mCgT4wROvL948w6f6ArJNb7yQd7QsvqavDeZHvNmHo=
 | 
			
		||||
k8s.io/component-base v0.33.2 h1:sCCsn9s/dG3ZrQTX/Us0/Sx2R0G5kwa0wbZFYoVp/+0=
 | 
			
		||||
k8s.io/component-base v0.33.2/go.mod h1:/41uw9wKzuelhN+u+/C59ixxf4tYQKW7p32ddkYNe2k=
 | 
			
		||||
k8s.io/helm v2.17.0+incompatible h1:Bpn6o1wKLYqKM3+Osh8e+1/K2g/GsQJ4F4yNF2+deao=
 | 
			
		||||
k8s.io/helm v2.17.0+incompatible/go.mod h1:LZzlS4LQBHfciFOurYBFkCMTaZ0D1l+p0teMg7TSULI=
 | 
			
		||||
k8s.io/klog/v2 v2.130.1 h1:n9Xl7H1Xvksem4KFG4PYbdQCQxqc/tTUyrgXaOhHSzk=
 | 
			
		||||
k8s.io/klog/v2 v2.130.1/go.mod h1:3Jpz1GvMt720eyJH1ckRHK1EDfpxISzJ7I9OYgaDtPE=
 | 
			
		||||
k8s.io/kube-openapi v0.0.0-20250318190949-c8a335a9a2ff h1:/usPimJzUKKu+m+TE36gUyGcf03XZEP0ZIKgKj35LS4=
 | 
			
		||||
k8s.io/kube-openapi v0.0.0-20250318190949-c8a335a9a2ff/go.mod h1:5jIi+8yX4RIb8wk3XwBo5Pq2ccx4FP10ohkbSKCZoK8=
 | 
			
		||||
k8s.io/kubectl v0.33.2 h1:7XKZ6DYCklu5MZQzJe+CkCjoGZwD1wWl7t/FxzhMz7Y=
 | 
			
		||||
k8s.io/kubectl v0.33.2/go.mod h1:8rC67FB8tVTYraovAGNi/idWIK90z2CHFNMmGJZJ3KI=
 | 
			
		||||
k8s.io/utils v0.0.0-20241104100929-3ea5e8cea738 h1:M3sRQVHv7vB20Xc2ybTt7ODCeFj6JSWYFzOFnYeS6Ro=
 | 
			
		||||
k8s.io/utils v0.0.0-20241104100929-3ea5e8cea738/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0=
 | 
			
		||||
oras.land/oras-go/v2 v2.6.0 h1:X4ELRsiGkrbeox69+9tzTu492FMUu7zJQW6eJU+I2oc=
 | 
			
		||||
oras.land/oras-go/v2 v2.6.0/go.mod h1:magiQDfG6H1O9APp+rOsvCPcW1GD2MM7vgnKY0Y+u1o=
 | 
			
		||||
sigs.k8s.io/apiserver-network-proxy/konnectivity-client v0.31.2 h1:jpcvIRr3GLoUoEKRkHKSmGjxb6lWwrBlJsXc+eUYQHM=
 | 
			
		||||
sigs.k8s.io/apiserver-network-proxy/konnectivity-client v0.31.2/go.mod h1:Ve9uj1L+deCXFrPOk1LpFXqTg7LCFzFso6PA48q/XZw=
 | 
			
		||||
sigs.k8s.io/controller-runtime v0.21.0 h1:CYfjpEuicjUecRk+KAeyYh+ouUBn4llGyDYytIGcJS8=
 | 
			
		||||
sigs.k8s.io/controller-runtime v0.21.0/go.mod h1:OSg14+F65eWqIu4DceX7k/+QRAbTTvxeQSNSOQpukWM=
 | 
			
		||||
sigs.k8s.io/json v0.0.0-20241010143419-9aa6b5e7a4b3 h1:/Rv+M11QRah1itp8VhT6HoVx1Ray9eB4DBr+K+/sCJ8=
 | 
			
		||||
sigs.k8s.io/json v0.0.0-20241010143419-9aa6b5e7a4b3/go.mod h1:18nIHnGi6636UCz6m8i4DhaJ65T6EruyzmoQqI2BVDo=
 | 
			
		||||
sigs.k8s.io/kustomize/api v0.19.0 h1:F+2HB2mU1MSiR9Hp1NEgoU2q9ItNOaBJl0I4Dlus5SQ=
 | 
			
		||||
sigs.k8s.io/kustomize/api v0.19.0/go.mod h1:/BbwnivGVcBh1r+8m3tH1VNxJmHSk1PzP5fkP6lbL1o=
 | 
			
		||||
sigs.k8s.io/kustomize/kyaml v0.19.0 h1:RFge5qsO1uHhwJsu3ipV7RNolC7Uozc0jUBC/61XSlA=
 | 
			
		||||
sigs.k8s.io/kustomize/kyaml v0.19.0/go.mod h1:FeKD5jEOH+FbZPpqUghBP8mrLjJ3+zD3/rf9NNu1cwY=
 | 
			
		||||
sigs.k8s.io/randfill v0.0.0-20250304075658-069ef1bbf016/go.mod h1:XeLlZ/jmk4i1HRopwe7/aU3H5n1zNUcX6TM94b3QxOY=
 | 
			
		||||
sigs.k8s.io/randfill v1.0.0 h1:JfjMILfT8A6RbawdsK2JXGBR5AQVfd+9TbzrlneTyrU=
 | 
			
		||||
sigs.k8s.io/randfill v1.0.0/go.mod h1:XeLlZ/jmk4i1HRopwe7/aU3H5n1zNUcX6TM94b3QxOY=
 | 
			
		||||
sigs.k8s.io/structured-merge-diff/v4 v4.6.0 h1:IUA9nvMmnKWcj5jl84xn+T5MnlZKThmUW1TdblaLVAc=
 | 
			
		||||
sigs.k8s.io/structured-merge-diff/v4 v4.6.0/go.mod h1:dDy58f92j70zLsuZVuUX5Wp9vtxXpaZnkPGWeqDfCps=
 | 
			
		||||
sigs.k8s.io/yaml v1.4.0 h1:Mk1wCc2gy/F0THH0TAp1QYyJNzRm2KCLy3o5ASXVI5E=
 | 
			
		||||
sigs.k8s.io/yaml v1.4.0/go.mod h1:Ejl7/uTz7PSA4eKMyQCUTnhZYNmLIl+5c2lQPGR2BPY=
 | 
			
		||||
							
								
								
									
										15
									
								
								hack/boilerplate.go.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								hack/boilerplate.go.txt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
							
								
								
									
										63
									
								
								internal/controller/helmdiff_controller.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								internal/controller/helmdiff_controller.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,63 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package controller
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
 | 
			
		||||
	"k8s.io/apimachinery/pkg/runtime"
 | 
			
		||||
	ctrl "sigs.k8s.io/controller-runtime"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/client"
 | 
			
		||||
	logf "sigs.k8s.io/controller-runtime/pkg/log"
 | 
			
		||||
 | 
			
		||||
	yahov1alpha1 "github.com/allanger/yaho/api/v1alpha1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// HelmDiffReconciler reconciles a HelmDiff object
 | 
			
		||||
type HelmDiffReconciler struct {
 | 
			
		||||
	client.Client
 | 
			
		||||
	Scheme *runtime.Scheme
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmdiffs,verbs=get;list;watch;create;update;patch;delete
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmdiffs/status,verbs=get;update;patch
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmdiffs/finalizers,verbs=update
 | 
			
		||||
 | 
			
		||||
// Reconcile is part of the main kubernetes reconciliation loop which aims to
 | 
			
		||||
// move the current state of the cluster closer to the desired state.
 | 
			
		||||
// TODO(user): Modify the Reconcile function to compare the state specified by
 | 
			
		||||
// the HelmDiff object against the actual cluster state, and then
 | 
			
		||||
// perform operations to make the cluster state reflect the state specified by
 | 
			
		||||
// the user.
 | 
			
		||||
//
 | 
			
		||||
// For more details, check Reconcile and its Result here:
 | 
			
		||||
// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.21.0/pkg/reconcile
 | 
			
		||||
func (r *HelmDiffReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
 | 
			
		||||
	_ = logf.FromContext(ctx)
 | 
			
		||||
 | 
			
		||||
	// TODO(user): your logic here
 | 
			
		||||
 | 
			
		||||
	return ctrl.Result{}, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// SetupWithManager sets up the controller with the Manager.
 | 
			
		||||
func (r *HelmDiffReconciler) SetupWithManager(mgr ctrl.Manager) error {
 | 
			
		||||
	return ctrl.NewControllerManagedBy(mgr).
 | 
			
		||||
		For(&yahov1alpha1.HelmDiff{}).
 | 
			
		||||
		Named("helmdiff").
 | 
			
		||||
		Complete(r)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										84
									
								
								internal/controller/helmdiff_controller_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								internal/controller/helmdiff_controller_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,84 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package controller
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
 | 
			
		||||
	. "github.com/onsi/ginkgo/v2"
 | 
			
		||||
	. "github.com/onsi/gomega"
 | 
			
		||||
	"k8s.io/apimachinery/pkg/api/errors"
 | 
			
		||||
	"k8s.io/apimachinery/pkg/types"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/reconcile"
 | 
			
		||||
 | 
			
		||||
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
			
		||||
 | 
			
		||||
	yahov1alpha1 "github.com/allanger/yaho/api/v1alpha1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var _ = Describe("HelmDiff Controller", func() {
 | 
			
		||||
	Context("When reconciling a resource", func() {
 | 
			
		||||
		const resourceName = "test-resource"
 | 
			
		||||
 | 
			
		||||
		ctx := context.Background()
 | 
			
		||||
 | 
			
		||||
		typeNamespacedName := types.NamespacedName{
 | 
			
		||||
			Name:      resourceName,
 | 
			
		||||
			Namespace: "default", // TODO(user):Modify as needed
 | 
			
		||||
		}
 | 
			
		||||
		helmdiff := &yahov1alpha1.HelmDiff{}
 | 
			
		||||
 | 
			
		||||
		BeforeEach(func() {
 | 
			
		||||
			By("creating the custom resource for the Kind HelmDiff")
 | 
			
		||||
			err := k8sClient.Get(ctx, typeNamespacedName, helmdiff)
 | 
			
		||||
			if err != nil && errors.IsNotFound(err) {
 | 
			
		||||
				resource := &yahov1alpha1.HelmDiff{
 | 
			
		||||
					ObjectMeta: metav1.ObjectMeta{
 | 
			
		||||
						Name:      resourceName,
 | 
			
		||||
						Namespace: "default",
 | 
			
		||||
					},
 | 
			
		||||
					// TODO(user): Specify other spec details if needed.
 | 
			
		||||
				}
 | 
			
		||||
				Expect(k8sClient.Create(ctx, resource)).To(Succeed())
 | 
			
		||||
			}
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
		AfterEach(func() {
 | 
			
		||||
			// TODO(user): Cleanup logic after each test, like removing the resource instance.
 | 
			
		||||
			resource := &yahov1alpha1.HelmDiff{}
 | 
			
		||||
			err := k8sClient.Get(ctx, typeNamespacedName, resource)
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
 | 
			
		||||
			By("Cleanup the specific resource instance HelmDiff")
 | 
			
		||||
			Expect(k8sClient.Delete(ctx, resource)).To(Succeed())
 | 
			
		||||
		})
 | 
			
		||||
		It("should successfully reconcile the resource", func() {
 | 
			
		||||
			By("Reconciling the created resource")
 | 
			
		||||
			controllerReconciler := &HelmDiffReconciler{
 | 
			
		||||
				Client: k8sClient,
 | 
			
		||||
				Scheme: k8sClient.Scheme(),
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			_, err := controllerReconciler.Reconcile(ctx, reconcile.Request{
 | 
			
		||||
				NamespacedName: typeNamespacedName,
 | 
			
		||||
			})
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
			// TODO(user): Add more specific assertions depending on your controller's reconciliation logic.
 | 
			
		||||
			// Example: If you expect a certain status condition after reconciliation, verify it here.
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
})
 | 
			
		||||
							
								
								
									
										94
									
								
								internal/controller/helmrelease_controller.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								internal/controller/helmrelease_controller.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,94 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package controller
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	"k8s.io/apimachinery/pkg/runtime"
 | 
			
		||||
	ctrl "sigs.k8s.io/controller-runtime"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/client"
 | 
			
		||||
	logf "sigs.k8s.io/controller-runtime/pkg/log"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/reconcile"
 | 
			
		||||
 | 
			
		||||
	yahov1alpha1 "github.com/allanger/yaho/api/v1alpha1"
 | 
			
		||||
	"github.com/allanger/yaho/internal/downloader"
 | 
			
		||||
	k8serrors "k8s.io/apimachinery/pkg/api/errors"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// HelmReleaseReconciler reconciles a HelmRelease object
 | 
			
		||||
type HelmReleaseReconciler struct {
 | 
			
		||||
	client.Client
 | 
			
		||||
	Scheme *runtime.Scheme
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmreleases,verbs=get;list;watch;create;update;patch;delete
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmreleases/status,verbs=get;update;patch
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmreleases/finalizers,verbs=update
 | 
			
		||||
 | 
			
		||||
func (r *HelmReleaseReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
 | 
			
		||||
	log := logf.FromContext(ctx)
 | 
			
		||||
	log.Info("reconciliation is started")
 | 
			
		||||
 | 
			
		||||
	reconcilePeriod := 30 * time.Second
 | 
			
		||||
	reconcileResult := reconcile.Result{RequeueAfter: reconcilePeriod}
 | 
			
		||||
	helmReleaseCR := &yahov1alpha1.HelmRelease{}
 | 
			
		||||
	err := r.Get(ctx, req.NamespacedName, helmReleaseCR)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		if k8serrors.IsNotFound(err) {
 | 
			
		||||
			// Requested object not found, could have been deleted after reconcile request.
 | 
			
		||||
			// Owned objects are automatically garbage collected. For additional cleanup logic use finalizers.
 | 
			
		||||
			// Return and don't requeue
 | 
			
		||||
			return reconcileResult, nil
 | 
			
		||||
		}
 | 
			
		||||
		log.Error(err, "An unxexpected error has occurred during the reconciliation")
 | 
			
		||||
		// Error reading the object - requeue the request.
 | 
			
		||||
		return reconcileResult, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Update object status always when function exit abnormally or through a panic.
 | 
			
		||||
	defer func() {
 | 
			
		||||
		if err := r.Status().Update(ctx, helmReleaseCR); err != nil {
 | 
			
		||||
			log.Error(err, "Failed to update status")
 | 
			
		||||
		}
 | 
			
		||||
	}()
 | 
			
		||||
 | 
			
		||||
	// First setup all the controller logic
 | 
			
		||||
	// Pull the chart t a temporary directory
 | 
			
		||||
	// 	
 | 
			
		||||
	// TODO(user): your logic here
 | 
			
		||||
	path, err := downloader.PullChart(ctx, helmReleaseCR.Spec.Repository, helmReleaseCR.Spec.Chart, helmReleaseCR.Spec.Version, nil)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Error(err, "An unexpected error has occurred while trying to fetch a chart")
 | 
			
		||||
		return reconcileResult, nil
 | 
			
		||||
	}
 | 
			
		||||
	log.Info("Pulled a chart", "path", path)
 | 
			
		||||
	//path, err := downloader.PullChart(repository, chart, version)
 | 
			
		||||
	//if err != nil { return ... }
 | 
			
		||||
	//err := helm.InstallOrUpdate(); if err != nil { return ...}
 | 
			
		||||
	
 | 
			
		||||
	return ctrl.Result{}, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// SetupWithManager sets up the controller with the Manager.
 | 
			
		||||
func (r *HelmReleaseReconciler) SetupWithManager(mgr ctrl.Manager) error {
 | 
			
		||||
	return ctrl.NewControllerManagedBy(mgr).
 | 
			
		||||
		For(&yahov1alpha1.HelmRelease{}).
 | 
			
		||||
		Named("helmrelease").
 | 
			
		||||
		Complete(r)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										84
									
								
								internal/controller/helmrelease_controller_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								internal/controller/helmrelease_controller_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,84 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package controller
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
 | 
			
		||||
	. "github.com/onsi/ginkgo/v2"
 | 
			
		||||
	. "github.com/onsi/gomega"
 | 
			
		||||
	"k8s.io/apimachinery/pkg/api/errors"
 | 
			
		||||
	"k8s.io/apimachinery/pkg/types"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/reconcile"
 | 
			
		||||
 | 
			
		||||
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
			
		||||
 | 
			
		||||
	yahov1alpha1 "github.com/allanger/yaho/api/v1alpha1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var _ = Describe("HelmRelease Controller", func() {
 | 
			
		||||
	Context("When reconciling a resource", func() {
 | 
			
		||||
		const resourceName = "test-resource"
 | 
			
		||||
 | 
			
		||||
		ctx := context.Background()
 | 
			
		||||
 | 
			
		||||
		typeNamespacedName := types.NamespacedName{
 | 
			
		||||
			Name:      resourceName,
 | 
			
		||||
			Namespace: "default", // TODO(user):Modify as needed
 | 
			
		||||
		}
 | 
			
		||||
		helmrelease := &yahov1alpha1.HelmRelease{}
 | 
			
		||||
 | 
			
		||||
		BeforeEach(func() {
 | 
			
		||||
			By("creating the custom resource for the Kind HelmRelease")
 | 
			
		||||
			err := k8sClient.Get(ctx, typeNamespacedName, helmrelease)
 | 
			
		||||
			if err != nil && errors.IsNotFound(err) {
 | 
			
		||||
				resource := &yahov1alpha1.HelmRelease{
 | 
			
		||||
					ObjectMeta: metav1.ObjectMeta{
 | 
			
		||||
						Name:      resourceName,
 | 
			
		||||
						Namespace: "default",
 | 
			
		||||
					},
 | 
			
		||||
					// TODO(user): Specify other spec details if needed.
 | 
			
		||||
				}
 | 
			
		||||
				Expect(k8sClient.Create(ctx, resource)).To(Succeed())
 | 
			
		||||
			}
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
		AfterEach(func() {
 | 
			
		||||
			// TODO(user): Cleanup logic after each test, like removing the resource instance.
 | 
			
		||||
			resource := &yahov1alpha1.HelmRelease{}
 | 
			
		||||
			err := k8sClient.Get(ctx, typeNamespacedName, resource)
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
 | 
			
		||||
			By("Cleanup the specific resource instance HelmRelease")
 | 
			
		||||
			Expect(k8sClient.Delete(ctx, resource)).To(Succeed())
 | 
			
		||||
		})
 | 
			
		||||
		It("should successfully reconcile the resource", func() {
 | 
			
		||||
			By("Reconciling the created resource")
 | 
			
		||||
			controllerReconciler := &HelmReleaseReconciler{
 | 
			
		||||
				Client: k8sClient,
 | 
			
		||||
				Scheme: k8sClient.Scheme(),
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			_, err := controllerReconciler.Reconcile(ctx, reconcile.Request{
 | 
			
		||||
				NamespacedName: typeNamespacedName,
 | 
			
		||||
			})
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
			// TODO(user): Add more specific assertions depending on your controller's reconciliation logic.
 | 
			
		||||
			// Example: If you expect a certain status condition after reconciliation, verify it here.
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
})
 | 
			
		||||
							
								
								
									
										63
									
								
								internal/controller/helmvalues_controller.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								internal/controller/helmvalues_controller.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,63 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package controller
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
 | 
			
		||||
	"k8s.io/apimachinery/pkg/runtime"
 | 
			
		||||
	ctrl "sigs.k8s.io/controller-runtime"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/client"
 | 
			
		||||
	logf "sigs.k8s.io/controller-runtime/pkg/log"
 | 
			
		||||
 | 
			
		||||
	yahov1alpha1 "github.com/allanger/yaho/api/v1alpha1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// HelmValuesReconciler reconciles a HelmValues object
 | 
			
		||||
type HelmValuesReconciler struct {
 | 
			
		||||
	client.Client
 | 
			
		||||
	Scheme *runtime.Scheme
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmvalues,verbs=get;list;watch;create;update;patch;delete
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmvalues/status,verbs=get;update;patch
 | 
			
		||||
// +kubebuilder:rbac:groups=yaho.badhouseplants.net,resources=helmvalues/finalizers,verbs=update
 | 
			
		||||
 | 
			
		||||
// Reconcile is part of the main kubernetes reconciliation loop which aims to
 | 
			
		||||
// move the current state of the cluster closer to the desired state.
 | 
			
		||||
// TODO(user): Modify the Reconcile function to compare the state specified by
 | 
			
		||||
// the HelmValues object against the actual cluster state, and then
 | 
			
		||||
// perform operations to make the cluster state reflect the state specified by
 | 
			
		||||
// the user.
 | 
			
		||||
//
 | 
			
		||||
// For more details, check Reconcile and its Result here:
 | 
			
		||||
// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.21.0/pkg/reconcile
 | 
			
		||||
func (r *HelmValuesReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
 | 
			
		||||
	_ = logf.FromContext(ctx)
 | 
			
		||||
 | 
			
		||||
	// TODO(user): your logic here
 | 
			
		||||
 | 
			
		||||
	return ctrl.Result{}, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// SetupWithManager sets up the controller with the Manager.
 | 
			
		||||
func (r *HelmValuesReconciler) SetupWithManager(mgr ctrl.Manager) error {
 | 
			
		||||
	return ctrl.NewControllerManagedBy(mgr).
 | 
			
		||||
		For(&yahov1alpha1.HelmValues{}).
 | 
			
		||||
		Named("helmvalues").
 | 
			
		||||
		Complete(r)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										84
									
								
								internal/controller/helmvalues_controller_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								internal/controller/helmvalues_controller_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,84 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package controller
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
 | 
			
		||||
	. "github.com/onsi/ginkgo/v2"
 | 
			
		||||
	. "github.com/onsi/gomega"
 | 
			
		||||
	"k8s.io/apimachinery/pkg/api/errors"
 | 
			
		||||
	"k8s.io/apimachinery/pkg/types"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/reconcile"
 | 
			
		||||
 | 
			
		||||
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
			
		||||
 | 
			
		||||
	yahov1alpha1 "github.com/allanger/yaho/api/v1alpha1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var _ = Describe("HelmValues Controller", func() {
 | 
			
		||||
	Context("When reconciling a resource", func() {
 | 
			
		||||
		const resourceName = "test-resource"
 | 
			
		||||
 | 
			
		||||
		ctx := context.Background()
 | 
			
		||||
 | 
			
		||||
		typeNamespacedName := types.NamespacedName{
 | 
			
		||||
			Name:      resourceName,
 | 
			
		||||
			Namespace: "default", // TODO(user):Modify as needed
 | 
			
		||||
		}
 | 
			
		||||
		helmvalues := &yahov1alpha1.HelmValues{}
 | 
			
		||||
 | 
			
		||||
		BeforeEach(func() {
 | 
			
		||||
			By("creating the custom resource for the Kind HelmValues")
 | 
			
		||||
			err := k8sClient.Get(ctx, typeNamespacedName, helmvalues)
 | 
			
		||||
			if err != nil && errors.IsNotFound(err) {
 | 
			
		||||
				resource := &yahov1alpha1.HelmValues{
 | 
			
		||||
					ObjectMeta: metav1.ObjectMeta{
 | 
			
		||||
						Name:      resourceName,
 | 
			
		||||
						Namespace: "default",
 | 
			
		||||
					},
 | 
			
		||||
					// TODO(user): Specify other spec details if needed.
 | 
			
		||||
				}
 | 
			
		||||
				Expect(k8sClient.Create(ctx, resource)).To(Succeed())
 | 
			
		||||
			}
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
		AfterEach(func() {
 | 
			
		||||
			// TODO(user): Cleanup logic after each test, like removing the resource instance.
 | 
			
		||||
			resource := &yahov1alpha1.HelmValues{}
 | 
			
		||||
			err := k8sClient.Get(ctx, typeNamespacedName, resource)
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
 | 
			
		||||
			By("Cleanup the specific resource instance HelmValues")
 | 
			
		||||
			Expect(k8sClient.Delete(ctx, resource)).To(Succeed())
 | 
			
		||||
		})
 | 
			
		||||
		It("should successfully reconcile the resource", func() {
 | 
			
		||||
			By("Reconciling the created resource")
 | 
			
		||||
			controllerReconciler := &HelmValuesReconciler{
 | 
			
		||||
				Client: k8sClient,
 | 
			
		||||
				Scheme: k8sClient.Scheme(),
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			_, err := controllerReconciler.Reconcile(ctx, reconcile.Request{
 | 
			
		||||
				NamespacedName: typeNamespacedName,
 | 
			
		||||
			})
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
			// TODO(user): Add more specific assertions depending on your controller's reconciliation logic.
 | 
			
		||||
			// Example: If you expect a certain status condition after reconciliation, verify it here.
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
})
 | 
			
		||||
							
								
								
									
										116
									
								
								internal/controller/suite_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								internal/controller/suite_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,116 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package controller
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"os"
 | 
			
		||||
	"path/filepath"
 | 
			
		||||
	"testing"
 | 
			
		||||
 | 
			
		||||
	. "github.com/onsi/ginkgo/v2"
 | 
			
		||||
	. "github.com/onsi/gomega"
 | 
			
		||||
 | 
			
		||||
	"k8s.io/client-go/kubernetes/scheme"
 | 
			
		||||
	"k8s.io/client-go/rest"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/client"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/envtest"
 | 
			
		||||
	logf "sigs.k8s.io/controller-runtime/pkg/log"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/log/zap"
 | 
			
		||||
 | 
			
		||||
	yahov1alpha1 "github.com/allanger/yaho/api/v1alpha1"
 | 
			
		||||
	// +kubebuilder:scaffold:imports
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// These tests use Ginkgo (BDD-style Go testing framework). Refer to
 | 
			
		||||
// http://onsi.github.io/ginkgo/ to learn more about Ginkgo.
 | 
			
		||||
 | 
			
		||||
var (
 | 
			
		||||
	ctx       context.Context
 | 
			
		||||
	cancel    context.CancelFunc
 | 
			
		||||
	testEnv   *envtest.Environment
 | 
			
		||||
	cfg       *rest.Config
 | 
			
		||||
	k8sClient client.Client
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TestControllers(t *testing.T) {
 | 
			
		||||
	RegisterFailHandler(Fail)
 | 
			
		||||
 | 
			
		||||
	RunSpecs(t, "Controller Suite")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
var _ = BeforeSuite(func() {
 | 
			
		||||
	logf.SetLogger(zap.New(zap.WriteTo(GinkgoWriter), zap.UseDevMode(true)))
 | 
			
		||||
 | 
			
		||||
	ctx, cancel = context.WithCancel(context.TODO())
 | 
			
		||||
 | 
			
		||||
	var err error
 | 
			
		||||
	err = yahov1alpha1.AddToScheme(scheme.Scheme)
 | 
			
		||||
	Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
 | 
			
		||||
	// +kubebuilder:scaffold:scheme
 | 
			
		||||
 | 
			
		||||
	By("bootstrapping test environment")
 | 
			
		||||
	testEnv = &envtest.Environment{
 | 
			
		||||
		CRDDirectoryPaths:     []string{filepath.Join("..", "..", "config", "crd", "bases")},
 | 
			
		||||
		ErrorIfCRDPathMissing: true,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Retrieve the first found binary directory to allow running tests from IDEs
 | 
			
		||||
	if getFirstFoundEnvTestBinaryDir() != "" {
 | 
			
		||||
		testEnv.BinaryAssetsDirectory = getFirstFoundEnvTestBinaryDir()
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// cfg is defined in this file globally.
 | 
			
		||||
	cfg, err = testEnv.Start()
 | 
			
		||||
	Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
	Expect(cfg).NotTo(BeNil())
 | 
			
		||||
 | 
			
		||||
	k8sClient, err = client.New(cfg, client.Options{Scheme: scheme.Scheme})
 | 
			
		||||
	Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
	Expect(k8sClient).NotTo(BeNil())
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
var _ = AfterSuite(func() {
 | 
			
		||||
	By("tearing down the test environment")
 | 
			
		||||
	cancel()
 | 
			
		||||
	err := testEnv.Stop()
 | 
			
		||||
	Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
// getFirstFoundEnvTestBinaryDir locates the first binary in the specified path.
 | 
			
		||||
// ENVTEST-based tests depend on specific binaries, usually located in paths set by
 | 
			
		||||
// controller-runtime. When running tests directly (e.g., via an IDE) without using
 | 
			
		||||
// Makefile targets, the 'BinaryAssetsDirectory' must be explicitly configured.
 | 
			
		||||
//
 | 
			
		||||
// This function streamlines the process by finding the required binaries, similar to
 | 
			
		||||
// setting the 'KUBEBUILDER_ASSETS' environment variable. To ensure the binaries are
 | 
			
		||||
// properly set up, run 'make setup-envtest' beforehand.
 | 
			
		||||
func getFirstFoundEnvTestBinaryDir() string {
 | 
			
		||||
	basePath := filepath.Join("..", "..", "bin", "k8s")
 | 
			
		||||
	entries, err := os.ReadDir(basePath)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		logf.Log.Error(err, "Failed to read directory", "path", basePath)
 | 
			
		||||
		return ""
 | 
			
		||||
	}
 | 
			
		||||
	for _, entry := range entries {
 | 
			
		||||
		if entry.IsDir() {
 | 
			
		||||
			return filepath.Join(basePath, entry.Name())
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return ""
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										3
									
								
								internal/downloader/git.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								internal/downloader/git.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
			
		||||
package downloader
 | 
			
		||||
 | 
			
		||||
// It should download a helm chart from git
 | 
			
		||||
							
								
								
									
										3
									
								
								internal/downloader/helm.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								internal/downloader/helm.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
			
		||||
package downloader
 | 
			
		||||
 | 
			
		||||
// It should download a helm chart
 | 
			
		||||
							
								
								
									
										35
									
								
								internal/downloader/types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								internal/downloader/types.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,35 @@
 | 
			
		||||
package downloader
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"github.com/allanger/yaho/internal/tools/git"
 | 
			
		||||
	"github.com/allanger/yaho/internal/tools/helm"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type PullOptions struct {}
 | 
			
		||||
 | 
			
		||||
// A very stupid check
 | 
			
		||||
func isGitRepo(repository string) bool {
 | 
			
		||||
	return strings.HasPrefix(repository, "git@") || strings.HasSuffix(repository, ".git")
 | 
			
		||||
}
 | 
			
		||||
// Pull chart into a temporary directory
 | 
			
		||||
func PullChart(ctx context.Context, repository, chart, version string, opts *PullOptions) (string, error) {
 | 
			
		||||
	if isGitRepo(repository) {
 | 
			
		||||
		git := git.GitLib{}
 | 
			
		||||
		path, err := git.CloneRepo(ctx, repository, version)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return "", err
 | 
			
		||||
		}
 | 
			
		||||
		return fmt.Sprintf("%s/%s", path, chart), nil
 | 
			
		||||
	} else {
 | 
			
		||||
		helm := helm.HelmLib{}
 | 
			
		||||
		path, err := helm.PullChart(ctx, repository, chart, version)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return "", err
 | 
			
		||||
		}
 | 
			
		||||
		return path, nil
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										39
									
								
								internal/tools/git/gitlib.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								internal/tools/git/gitlib.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
			
		||||
package git
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"os"
 | 
			
		||||
 | 
			
		||||
	"github.com/go-git/go-git/v5"
 | 
			
		||||
	"github.com/go-git/go-git/v5/plumbing/transport/ssh"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/log"
 | 
			
		||||
)
 | 
			
		||||
type GitLib struct {
 | 
			
		||||
	SSHKey string
 | 
			
		||||
	SSHKeyPassword string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *GitLib) CloneRepo (ctx context.Context, gitURL, revision string) (string, error) {
 | 
			
		||||
	log := log.FromContext(ctx)
 | 
			
		||||
	workdir, err := os.MkdirTemp("/tmp", "helmdownloader")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return "", err
 | 
			
		||||
	}
 | 
			
		||||
	log.Info("A temporary directory is created", "path", workdir)
 | 
			
		||||
	if len(g.SSHKey) > 0 {
 | 
			
		||||
		keys, err := ssh.NewPublicKeys("git", []byte(g.SSHKey), g.SSHKeyPassword)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return "", err
 | 
			
		||||
		}
 | 
			
		||||
		_, err = git.PlainClone(workdir, false, &git.CloneOptions{URL: gitURL, Auth: keys})
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return "", err
 | 
			
		||||
		}
 | 
			
		||||
	} else {
 | 
			
		||||
		_, err = git.PlainClone(workdir, false, &git.CloneOptions{URL: gitURL})
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return "", err
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return workdir, nil
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										15
									
								
								internal/tools/git/gitlib_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								internal/tools/git/gitlib_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
package git_test
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"testing"
 | 
			
		||||
 | 
			
		||||
	"github.com/allanger/yaho/internal/tools/git"
 | 
			
		||||
	"github.com/stretchr/testify/assert"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TestGitHttpClone(t *testing.T) {
 | 
			
		||||
	gitlib := &git.GitLib{}
 | 
			
		||||
	path, err := gitlib.CloneRepo("https://github.com/db-operator/db-operator.git", "main")
 | 
			
		||||
	assert.NoError(t, err)
 | 
			
		||||
	assert.Equal(t, "test", path)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										91
									
								
								internal/tools/helm/helmlib.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								internal/tools/helm/helmlib.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,91 @@
 | 
			
		||||
package helm
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"os"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"helm.sh/helm/pkg/chartutil"
 | 
			
		||||
	"helm.sh/helm/pkg/engine"
 | 
			
		||||
	"helm.sh/helm/v3/pkg/action"
 | 
			
		||||
	"helm.sh/helm/v3/pkg/cli"
 | 
			
		||||
	"helm.sh/helm/v3/pkg/registry"
 | 
			
		||||
	"honnef.co/go/tools/go/loader"
 | 
			
		||||
	"sigs.k8s.io/controller-runtime/pkg/log"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type HelmLib struct {}
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	helmRepo = "helm"
 | 
			
		||||
	ociRepo = "oci"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func (h *HelmLib) PullChart(ctx context.Context, repository, chart, version string) (string, error) {
 | 
			
		||||
	log := log.FromContext(ctx)
 | 
			
		||||
	workdir, err := os.MkdirTemp("/tmp", "helmdownloader")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return "", err
 | 
			
		||||
	}
 | 
			
		||||
	log.Info("A temporary directory is created", "path", workdir)
 | 
			
		||||
	config := new(action.Configuration)
 | 
			
		||||
	cl := cli.New()
 | 
			
		||||
 | 
			
		||||
	registry, err := registry.NewClient()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return "", err
 | 
			
		||||
	}
 | 
			
		||||
  client := action.NewPullWithOpts(action.WithConfig(config))
 | 
			
		||||
 | 
			
		||||
	prefix := repository[:strings.IndexByte(repository, ':')]
 | 
			
		||||
	var chartRemote string
 | 
			
		||||
 | 
			
		||||
	switch prefix {
 | 
			
		||||
	case "oci":
 | 
			
		||||
		chartRemote = fmt.Sprintf("%s/%s", repository, chart)
 | 
			
		||||
		client.SetRegistryClient(registry)
 | 
			
		||||
	case "https", "http":
 | 
			
		||||
		client.RepoURL = repository
 | 
			
		||||
		chartRemote = chart
 | 
			
		||||
	default:
 | 
			
		||||
		return "", fmt.Errorf("unknown repo kind: %s", prefix)
 | 
			
		||||
	}
 | 
			
		||||
	client.Settings = cl
 | 
			
		||||
	client.UntarDir = workdir
 | 
			
		||||
	client.Version = version
 | 
			
		||||
	path, err := client.Run(chartRemote)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return "", err
 | 
			
		||||
	}
 | 
			
		||||
	return path, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (h *HelmLib) InstallChart(ctx context.Context, path string, name, namespace string) error {
 | 
			
		||||
	log := log.FromContext(ctx)
 | 
			
		||||
	chartObj, err := loader.Load(path)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
	values := chartutil.Values{}
 | 
			
		||||
	values["Values"] = nil 
 | 
			
		||||
	values["Release"] = map[string]string{
 | 
			
		||||
		"Name":      name,
 | 
			
		||||
		"Namespace": namespace,
 | 
			
		||||
	}
 | 
			
		||||
	values["Capabilities"] = map[string]map[string]string{
 | 
			
		||||
		"KubeVersion": {
 | 
			
		||||
			"Version":    "v1.27.9",
 | 
			
		||||
			"GitVersion": "v1.27.9",
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
	files, err := engine.Engine{Strict: false}.Render(chartObj, values)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
	for file, data := range files {
 | 
			
		||||
		log.Info("File is rendered", "data", file)
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										24
									
								
								internal/tools/helm/helmlib_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								internal/tools/helm/helmlib_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
			
		||||
package helm_test
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"testing"
 | 
			
		||||
 | 
			
		||||
	"github.com/allanger/yaho/internal/tools/helm"
 | 
			
		||||
	"github.com/stretchr/testify/assert"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TestPullChart(t *testing.T) {
 | 
			
		||||
	helmlib := &helm.HelmLib{}
 | 
			
		||||
	path, err := helmlib.PullChart("https://coredns.github.io/helm", "coredns", "1.42.0")
 | 
			
		||||
	assert.NoError(t, err)
 | 
			
		||||
	assert.Equal(t, "test", path)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestPullChartOci(t *testing.T) {
 | 
			
		||||
	helmlib := &helm.HelmLib{}
 | 
			
		||||
	path, err := helmlib.PullChart("oci://ghcr.io/allanger/allangers-charts", "memos", "0.6.0")
 | 
			
		||||
	assert.NoError(t, err)
 | 
			
		||||
	assert.Equal(t, "test", path)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										5
									
								
								internal/tools/helm/types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								internal/tools/helm/types.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,5 @@
 | 
			
		||||
package helm
 | 
			
		||||
 | 
			
		||||
type Helm interface {
 | 
			
		||||
	PullChart() (string, error)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										85
									
								
								notes
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								notes
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,85 @@
 | 
			
		||||
This chart is supposed to be a better alternative to all the GitOps helm managers out there.
 | 
			
		||||
 | 
			
		||||
It should follow the real helm workflow instead of trying to find some workaround.
 | 
			
		||||
 | 
			
		||||
The idea is that I need to create a helm release 
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
kind: HelmRelease
 | 
			
		||||
metadata: 
 | 
			
		||||
  name: test
 | 
			
		||||
spec:
 | 
			
		||||
  repository: https://somerepo
 | 
			
		||||
  chart: somechart
 | 
			
		||||
  version: someversion
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
And the controller is supposed to pull the chart and install it to the cluster
 | 
			
		||||
 | 
			
		||||
In order to make ic configurable, I need to add CRDs for handling values, like
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
kind: HelmValues
 | 
			
		||||
metadata:
 | 
			
		||||
  name: test-values
 | 
			
		||||
data:
 | 
			
		||||
  image:
 | 
			
		||||
    tag: latest
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
These values should be added to the helm release
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
kind: HelmRelease
 | 
			
		||||
metadata:
 | 
			
		||||
  name: test
 | 
			
		||||
spec:
 | 
			
		||||
  repository: https://somerepo
 | 
			
		||||
  chart: somechart
 | 
			
		||||
  version: someversion
 | 
			
		||||
  values:
 | 
			
		||||
    - test-values
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
After syncing, the values hashsum should be put to the status like
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
kind: HelmRelease
 | 
			
		||||
status:
 | 
			
		||||
  values:
 | 
			
		||||
    test-values: SHA
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
To make sure that we are updating the chart every time, values are updated
 | 
			
		||||
 | 
			
		||||
The next CRD that should be a part of any reliable helm workflow is the diff
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
kind: HelmDiff
 | 
			
		||||
metadata:
 | 
			
		||||
  name: test
 | 
			
		||||
spec:
 | 
			
		||||
  release: test
 | 
			
		||||
  context: 5
 | 
			
		||||
  values:
 | 
			
		||||
    - test-values-SHA # these should be added in the cluster with a different name
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
It should provide a diff and put it to the staus in a base64 format (or maybe to secrets)
 | 
			
		||||
 | 
			
		||||
After this is working, the next step would be adding support for secrets via SOPS
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
kind: HelmValues
 | 
			
		||||
metadata
 | 
			
		||||
  name: test
 | 
			
		||||
spec: 
 | 
			
		||||
  secret: true
 | 
			
		||||
  sopsSec:
 | 
			
		||||
    name: sops-secret
 | 
			
		||||
    key: .sops.yaml
 | 
			
		||||
  templating: 
 | 
			
		||||
    enabled: false
 | 
			
		||||
data:
 | 
			
		||||
  # Encrypted data
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										89
									
								
								test/e2e/e2e_suite_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								test/e2e/e2e_suite_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,89 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package e2e
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"os"
 | 
			
		||||
	"os/exec"
 | 
			
		||||
	"testing"
 | 
			
		||||
 | 
			
		||||
	. "github.com/onsi/ginkgo/v2"
 | 
			
		||||
	. "github.com/onsi/gomega"
 | 
			
		||||
 | 
			
		||||
	"github.com/allanger/yaho/test/utils"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var (
 | 
			
		||||
	// Optional Environment Variables:
 | 
			
		||||
	// - CERT_MANAGER_INSTALL_SKIP=true: Skips CertManager installation during test setup.
 | 
			
		||||
	// These variables are useful if CertManager is already installed, avoiding
 | 
			
		||||
	// re-installation and conflicts.
 | 
			
		||||
	skipCertManagerInstall = os.Getenv("CERT_MANAGER_INSTALL_SKIP") == "true"
 | 
			
		||||
	// isCertManagerAlreadyInstalled will be set true when CertManager CRDs be found on the cluster
 | 
			
		||||
	isCertManagerAlreadyInstalled = false
 | 
			
		||||
 | 
			
		||||
	// projectImage is the name of the image which will be build and loaded
 | 
			
		||||
	// with the code source changes to be tested.
 | 
			
		||||
	projectImage = "example.com/yaho:v0.0.1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// TestE2E runs the end-to-end (e2e) test suite for the project. These tests execute in an isolated,
 | 
			
		||||
// temporary environment to validate project changes with the purposed to be used in CI jobs.
 | 
			
		||||
// The default setup requires Kind, builds/loads the Manager Docker image locally, and installs
 | 
			
		||||
// CertManager.
 | 
			
		||||
func TestE2E(t *testing.T) {
 | 
			
		||||
	RegisterFailHandler(Fail)
 | 
			
		||||
	_, _ = fmt.Fprintf(GinkgoWriter, "Starting yaho integration test suite\n")
 | 
			
		||||
	RunSpecs(t, "e2e suite")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
var _ = BeforeSuite(func() {
 | 
			
		||||
	By("building the manager(Operator) image")
 | 
			
		||||
	cmd := exec.Command("make", "docker-build", fmt.Sprintf("IMG=%s", projectImage))
 | 
			
		||||
	_, err := utils.Run(cmd)
 | 
			
		||||
	ExpectWithOffset(1, err).NotTo(HaveOccurred(), "Failed to build the manager(Operator) image")
 | 
			
		||||
 | 
			
		||||
	// TODO(user): If you want to change the e2e test vendor from Kind, ensure the image is
 | 
			
		||||
	// built and available before running the tests. Also, remove the following block.
 | 
			
		||||
	By("loading the manager(Operator) image on Kind")
 | 
			
		||||
	err = utils.LoadImageToKindClusterWithName(projectImage)
 | 
			
		||||
	ExpectWithOffset(1, err).NotTo(HaveOccurred(), "Failed to load the manager(Operator) image into Kind")
 | 
			
		||||
 | 
			
		||||
	// The tests-e2e are intended to run on a temporary cluster that is created and destroyed for testing.
 | 
			
		||||
	// To prevent errors when tests run in environments with CertManager already installed,
 | 
			
		||||
	// we check for its presence before execution.
 | 
			
		||||
	// Setup CertManager before the suite if not skipped and if not already installed
 | 
			
		||||
	if !skipCertManagerInstall {
 | 
			
		||||
		By("checking if cert manager is installed already")
 | 
			
		||||
		isCertManagerAlreadyInstalled = utils.IsCertManagerCRDsInstalled()
 | 
			
		||||
		if !isCertManagerAlreadyInstalled {
 | 
			
		||||
			_, _ = fmt.Fprintf(GinkgoWriter, "Installing CertManager...\n")
 | 
			
		||||
			Expect(utils.InstallCertManager()).To(Succeed(), "Failed to install CertManager")
 | 
			
		||||
		} else {
 | 
			
		||||
			_, _ = fmt.Fprintf(GinkgoWriter, "WARNING: CertManager is already installed. Skipping installation...\n")
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
var _ = AfterSuite(func() {
 | 
			
		||||
	// Teardown CertManager after the suite if not skipped and if it was not already installed
 | 
			
		||||
	if !skipCertManagerInstall && !isCertManagerAlreadyInstalled {
 | 
			
		||||
		_, _ = fmt.Fprintf(GinkgoWriter, "Uninstalling CertManager...\n")
 | 
			
		||||
		utils.UninstallCertManager()
 | 
			
		||||
	}
 | 
			
		||||
})
 | 
			
		||||
							
								
								
									
										329
									
								
								test/e2e/e2e_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										329
									
								
								test/e2e/e2e_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,329 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package e2e
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"encoding/json"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"os"
 | 
			
		||||
	"os/exec"
 | 
			
		||||
	"path/filepath"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	. "github.com/onsi/ginkgo/v2"
 | 
			
		||||
	. "github.com/onsi/gomega"
 | 
			
		||||
 | 
			
		||||
	"github.com/allanger/yaho/test/utils"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// namespace where the project is deployed in
 | 
			
		||||
const namespace = "yaho-system"
 | 
			
		||||
 | 
			
		||||
// serviceAccountName created for the project
 | 
			
		||||
const serviceAccountName = "yaho-controller-manager"
 | 
			
		||||
 | 
			
		||||
// metricsServiceName is the name of the metrics service of the project
 | 
			
		||||
const metricsServiceName = "yaho-controller-manager-metrics-service"
 | 
			
		||||
 | 
			
		||||
// metricsRoleBindingName is the name of the RBAC that will be created to allow get the metrics data
 | 
			
		||||
const metricsRoleBindingName = "yaho-metrics-binding"
 | 
			
		||||
 | 
			
		||||
var _ = Describe("Manager", Ordered, func() {
 | 
			
		||||
	var controllerPodName string
 | 
			
		||||
 | 
			
		||||
	// Before running the tests, set up the environment by creating the namespace,
 | 
			
		||||
	// enforce the restricted security policy to the namespace, installing CRDs,
 | 
			
		||||
	// and deploying the controller.
 | 
			
		||||
	BeforeAll(func() {
 | 
			
		||||
		By("creating manager namespace")
 | 
			
		||||
		cmd := exec.Command("kubectl", "create", "ns", namespace)
 | 
			
		||||
		_, err := utils.Run(cmd)
 | 
			
		||||
		Expect(err).NotTo(HaveOccurred(), "Failed to create namespace")
 | 
			
		||||
 | 
			
		||||
		By("labeling the namespace to enforce the restricted security policy")
 | 
			
		||||
		cmd = exec.Command("kubectl", "label", "--overwrite", "ns", namespace,
 | 
			
		||||
			"pod-security.kubernetes.io/enforce=restricted")
 | 
			
		||||
		_, err = utils.Run(cmd)
 | 
			
		||||
		Expect(err).NotTo(HaveOccurred(), "Failed to label namespace with restricted policy")
 | 
			
		||||
 | 
			
		||||
		By("installing CRDs")
 | 
			
		||||
		cmd = exec.Command("make", "install")
 | 
			
		||||
		_, err = utils.Run(cmd)
 | 
			
		||||
		Expect(err).NotTo(HaveOccurred(), "Failed to install CRDs")
 | 
			
		||||
 | 
			
		||||
		By("deploying the controller-manager")
 | 
			
		||||
		cmd = exec.Command("make", "deploy", fmt.Sprintf("IMG=%s", projectImage))
 | 
			
		||||
		_, err = utils.Run(cmd)
 | 
			
		||||
		Expect(err).NotTo(HaveOccurred(), "Failed to deploy the controller-manager")
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// After all tests have been executed, clean up by undeploying the controller, uninstalling CRDs,
 | 
			
		||||
	// and deleting the namespace.
 | 
			
		||||
	AfterAll(func() {
 | 
			
		||||
		By("cleaning up the curl pod for metrics")
 | 
			
		||||
		cmd := exec.Command("kubectl", "delete", "pod", "curl-metrics", "-n", namespace)
 | 
			
		||||
		_, _ = utils.Run(cmd)
 | 
			
		||||
 | 
			
		||||
		By("undeploying the controller-manager")
 | 
			
		||||
		cmd = exec.Command("make", "undeploy")
 | 
			
		||||
		_, _ = utils.Run(cmd)
 | 
			
		||||
 | 
			
		||||
		By("uninstalling CRDs")
 | 
			
		||||
		cmd = exec.Command("make", "uninstall")
 | 
			
		||||
		_, _ = utils.Run(cmd)
 | 
			
		||||
 | 
			
		||||
		By("removing manager namespace")
 | 
			
		||||
		cmd = exec.Command("kubectl", "delete", "ns", namespace)
 | 
			
		||||
		_, _ = utils.Run(cmd)
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// After each test, check for failures and collect logs, events,
 | 
			
		||||
	// and pod descriptions for debugging.
 | 
			
		||||
	AfterEach(func() {
 | 
			
		||||
		specReport := CurrentSpecReport()
 | 
			
		||||
		if specReport.Failed() {
 | 
			
		||||
			By("Fetching controller manager pod logs")
 | 
			
		||||
			cmd := exec.Command("kubectl", "logs", controllerPodName, "-n", namespace)
 | 
			
		||||
			controllerLogs, err := utils.Run(cmd)
 | 
			
		||||
			if err == nil {
 | 
			
		||||
				_, _ = fmt.Fprintf(GinkgoWriter, "Controller logs:\n %s", controllerLogs)
 | 
			
		||||
			} else {
 | 
			
		||||
				_, _ = fmt.Fprintf(GinkgoWriter, "Failed to get Controller logs: %s", err)
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			By("Fetching Kubernetes events")
 | 
			
		||||
			cmd = exec.Command("kubectl", "get", "events", "-n", namespace, "--sort-by=.lastTimestamp")
 | 
			
		||||
			eventsOutput, err := utils.Run(cmd)
 | 
			
		||||
			if err == nil {
 | 
			
		||||
				_, _ = fmt.Fprintf(GinkgoWriter, "Kubernetes events:\n%s", eventsOutput)
 | 
			
		||||
			} else {
 | 
			
		||||
				_, _ = fmt.Fprintf(GinkgoWriter, "Failed to get Kubernetes events: %s", err)
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			By("Fetching curl-metrics logs")
 | 
			
		||||
			cmd = exec.Command("kubectl", "logs", "curl-metrics", "-n", namespace)
 | 
			
		||||
			metricsOutput, err := utils.Run(cmd)
 | 
			
		||||
			if err == nil {
 | 
			
		||||
				_, _ = fmt.Fprintf(GinkgoWriter, "Metrics logs:\n %s", metricsOutput)
 | 
			
		||||
			} else {
 | 
			
		||||
				_, _ = fmt.Fprintf(GinkgoWriter, "Failed to get curl-metrics logs: %s", err)
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			By("Fetching controller manager pod description")
 | 
			
		||||
			cmd = exec.Command("kubectl", "describe", "pod", controllerPodName, "-n", namespace)
 | 
			
		||||
			podDescription, err := utils.Run(cmd)
 | 
			
		||||
			if err == nil {
 | 
			
		||||
				fmt.Println("Pod description:\n", podDescription)
 | 
			
		||||
			} else {
 | 
			
		||||
				fmt.Println("Failed to describe controller pod")
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	SetDefaultEventuallyTimeout(2 * time.Minute)
 | 
			
		||||
	SetDefaultEventuallyPollingInterval(time.Second)
 | 
			
		||||
 | 
			
		||||
	Context("Manager", func() {
 | 
			
		||||
		It("should run successfully", func() {
 | 
			
		||||
			By("validating that the controller-manager pod is running as expected")
 | 
			
		||||
			verifyControllerUp := func(g Gomega) {
 | 
			
		||||
				// Get the name of the controller-manager pod
 | 
			
		||||
				cmd := exec.Command("kubectl", "get",
 | 
			
		||||
					"pods", "-l", "control-plane=controller-manager",
 | 
			
		||||
					"-o", "go-template={{ range .items }}"+
 | 
			
		||||
						"{{ if not .metadata.deletionTimestamp }}"+
 | 
			
		||||
						"{{ .metadata.name }}"+
 | 
			
		||||
						"{{ \"\\n\" }}{{ end }}{{ end }}",
 | 
			
		||||
					"-n", namespace,
 | 
			
		||||
				)
 | 
			
		||||
 | 
			
		||||
				podOutput, err := utils.Run(cmd)
 | 
			
		||||
				g.Expect(err).NotTo(HaveOccurred(), "Failed to retrieve controller-manager pod information")
 | 
			
		||||
				podNames := utils.GetNonEmptyLines(podOutput)
 | 
			
		||||
				g.Expect(podNames).To(HaveLen(1), "expected 1 controller pod running")
 | 
			
		||||
				controllerPodName = podNames[0]
 | 
			
		||||
				g.Expect(controllerPodName).To(ContainSubstring("controller-manager"))
 | 
			
		||||
 | 
			
		||||
				// Validate the pod's status
 | 
			
		||||
				cmd = exec.Command("kubectl", "get",
 | 
			
		||||
					"pods", controllerPodName, "-o", "jsonpath={.status.phase}",
 | 
			
		||||
					"-n", namespace,
 | 
			
		||||
				)
 | 
			
		||||
				output, err := utils.Run(cmd)
 | 
			
		||||
				g.Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
				g.Expect(output).To(Equal("Running"), "Incorrect controller-manager pod status")
 | 
			
		||||
			}
 | 
			
		||||
			Eventually(verifyControllerUp).Should(Succeed())
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
		It("should ensure the metrics endpoint is serving metrics", func() {
 | 
			
		||||
			By("creating a ClusterRoleBinding for the service account to allow access to metrics")
 | 
			
		||||
			cmd := exec.Command("kubectl", "create", "clusterrolebinding", metricsRoleBindingName,
 | 
			
		||||
				"--clusterrole=yaho-metrics-reader",
 | 
			
		||||
				fmt.Sprintf("--serviceaccount=%s:%s", namespace, serviceAccountName),
 | 
			
		||||
			)
 | 
			
		||||
			_, err := utils.Run(cmd)
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred(), "Failed to create ClusterRoleBinding")
 | 
			
		||||
 | 
			
		||||
			By("validating that the metrics service is available")
 | 
			
		||||
			cmd = exec.Command("kubectl", "get", "service", metricsServiceName, "-n", namespace)
 | 
			
		||||
			_, err = utils.Run(cmd)
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred(), "Metrics service should exist")
 | 
			
		||||
 | 
			
		||||
			By("getting the service account token")
 | 
			
		||||
			token, err := serviceAccountToken()
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
			Expect(token).NotTo(BeEmpty())
 | 
			
		||||
 | 
			
		||||
			By("waiting for the metrics endpoint to be ready")
 | 
			
		||||
			verifyMetricsEndpointReady := func(g Gomega) {
 | 
			
		||||
				cmd := exec.Command("kubectl", "get", "endpoints", metricsServiceName, "-n", namespace)
 | 
			
		||||
				output, err := utils.Run(cmd)
 | 
			
		||||
				g.Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
				g.Expect(output).To(ContainSubstring("8443"), "Metrics endpoint is not ready")
 | 
			
		||||
			}
 | 
			
		||||
			Eventually(verifyMetricsEndpointReady).Should(Succeed())
 | 
			
		||||
 | 
			
		||||
			By("verifying that the controller manager is serving the metrics server")
 | 
			
		||||
			verifyMetricsServerStarted := func(g Gomega) {
 | 
			
		||||
				cmd := exec.Command("kubectl", "logs", controllerPodName, "-n", namespace)
 | 
			
		||||
				output, err := utils.Run(cmd)
 | 
			
		||||
				g.Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
				g.Expect(output).To(ContainSubstring("controller-runtime.metrics\tServing metrics server"),
 | 
			
		||||
					"Metrics server not yet started")
 | 
			
		||||
			}
 | 
			
		||||
			Eventually(verifyMetricsServerStarted).Should(Succeed())
 | 
			
		||||
 | 
			
		||||
			By("creating the curl-metrics pod to access the metrics endpoint")
 | 
			
		||||
			cmd = exec.Command("kubectl", "run", "curl-metrics", "--restart=Never",
 | 
			
		||||
				"--namespace", namespace,
 | 
			
		||||
				"--image=curlimages/curl:latest",
 | 
			
		||||
				"--overrides",
 | 
			
		||||
				fmt.Sprintf(`{
 | 
			
		||||
					"spec": {
 | 
			
		||||
						"containers": [{
 | 
			
		||||
							"name": "curl",
 | 
			
		||||
							"image": "curlimages/curl:latest",
 | 
			
		||||
							"command": ["/bin/sh", "-c"],
 | 
			
		||||
							"args": ["curl -v -k -H 'Authorization: Bearer %s' https://%s.%s.svc.cluster.local:8443/metrics"],
 | 
			
		||||
							"securityContext": {
 | 
			
		||||
								"allowPrivilegeEscalation": false,
 | 
			
		||||
								"capabilities": {
 | 
			
		||||
									"drop": ["ALL"]
 | 
			
		||||
								},
 | 
			
		||||
								"runAsNonRoot": true,
 | 
			
		||||
								"runAsUser": 1000,
 | 
			
		||||
								"seccompProfile": {
 | 
			
		||||
									"type": "RuntimeDefault"
 | 
			
		||||
								}
 | 
			
		||||
							}
 | 
			
		||||
						}],
 | 
			
		||||
						"serviceAccount": "%s"
 | 
			
		||||
					}
 | 
			
		||||
				}`, token, metricsServiceName, namespace, serviceAccountName))
 | 
			
		||||
			_, err = utils.Run(cmd)
 | 
			
		||||
			Expect(err).NotTo(HaveOccurred(), "Failed to create curl-metrics pod")
 | 
			
		||||
 | 
			
		||||
			By("waiting for the curl-metrics pod to complete.")
 | 
			
		||||
			verifyCurlUp := func(g Gomega) {
 | 
			
		||||
				cmd := exec.Command("kubectl", "get", "pods", "curl-metrics",
 | 
			
		||||
					"-o", "jsonpath={.status.phase}",
 | 
			
		||||
					"-n", namespace)
 | 
			
		||||
				output, err := utils.Run(cmd)
 | 
			
		||||
				g.Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
				g.Expect(output).To(Equal("Succeeded"), "curl pod in wrong status")
 | 
			
		||||
			}
 | 
			
		||||
			Eventually(verifyCurlUp, 5*time.Minute).Should(Succeed())
 | 
			
		||||
 | 
			
		||||
			By("getting the metrics by checking curl-metrics logs")
 | 
			
		||||
			metricsOutput := getMetricsOutput()
 | 
			
		||||
			Expect(metricsOutput).To(ContainSubstring(
 | 
			
		||||
				"controller_runtime_reconcile_total",
 | 
			
		||||
			))
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
		// +kubebuilder:scaffold:e2e-webhooks-checks
 | 
			
		||||
 | 
			
		||||
		// TODO: Customize the e2e test suite with scenarios specific to your project.
 | 
			
		||||
		// Consider applying sample/CR(s) and check their status and/or verifying
 | 
			
		||||
		// the reconciliation by using the metrics, i.e.:
 | 
			
		||||
		// metricsOutput := getMetricsOutput()
 | 
			
		||||
		// Expect(metricsOutput).To(ContainSubstring(
 | 
			
		||||
		//    fmt.Sprintf(`controller_runtime_reconcile_total{controller="%s",result="success"} 1`,
 | 
			
		||||
		//    strings.ToLower(<Kind>),
 | 
			
		||||
		// ))
 | 
			
		||||
	})
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
// serviceAccountToken returns a token for the specified service account in the given namespace.
 | 
			
		||||
// It uses the Kubernetes TokenRequest API to generate a token by directly sending a request
 | 
			
		||||
// and parsing the resulting token from the API response.
 | 
			
		||||
func serviceAccountToken() (string, error) {
 | 
			
		||||
	const tokenRequestRawString = `{
 | 
			
		||||
		"apiVersion": "authentication.k8s.io/v1",
 | 
			
		||||
		"kind": "TokenRequest"
 | 
			
		||||
	}`
 | 
			
		||||
 | 
			
		||||
	// Temporary file to store the token request
 | 
			
		||||
	secretName := fmt.Sprintf("%s-token-request", serviceAccountName)
 | 
			
		||||
	tokenRequestFile := filepath.Join("/tmp", secretName)
 | 
			
		||||
	err := os.WriteFile(tokenRequestFile, []byte(tokenRequestRawString), os.FileMode(0o644))
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return "", err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var out string
 | 
			
		||||
	verifyTokenCreation := func(g Gomega) {
 | 
			
		||||
		// Execute kubectl command to create the token
 | 
			
		||||
		cmd := exec.Command("kubectl", "create", "--raw", fmt.Sprintf(
 | 
			
		||||
			"/api/v1/namespaces/%s/serviceaccounts/%s/token",
 | 
			
		||||
			namespace,
 | 
			
		||||
			serviceAccountName,
 | 
			
		||||
		), "-f", tokenRequestFile)
 | 
			
		||||
 | 
			
		||||
		output, err := cmd.CombinedOutput()
 | 
			
		||||
		g.Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
 | 
			
		||||
		// Parse the JSON output to extract the token
 | 
			
		||||
		var token tokenRequest
 | 
			
		||||
		err = json.Unmarshal(output, &token)
 | 
			
		||||
		g.Expect(err).NotTo(HaveOccurred())
 | 
			
		||||
 | 
			
		||||
		out = token.Status.Token
 | 
			
		||||
	}
 | 
			
		||||
	Eventually(verifyTokenCreation).Should(Succeed())
 | 
			
		||||
 | 
			
		||||
	return out, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// getMetricsOutput retrieves and returns the logs from the curl pod used to access the metrics endpoint.
 | 
			
		||||
func getMetricsOutput() string {
 | 
			
		||||
	By("getting the curl-metrics logs")
 | 
			
		||||
	cmd := exec.Command("kubectl", "logs", "curl-metrics", "-n", namespace)
 | 
			
		||||
	metricsOutput, err := utils.Run(cmd)
 | 
			
		||||
	Expect(err).NotTo(HaveOccurred(), "Failed to retrieve logs from curl pod")
 | 
			
		||||
	Expect(metricsOutput).To(ContainSubstring("< HTTP/1.1 200 OK"))
 | 
			
		||||
	return metricsOutput
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// tokenRequest is a simplified representation of the Kubernetes TokenRequest API response,
 | 
			
		||||
// containing only the token field that we need to extract.
 | 
			
		||||
type tokenRequest struct {
 | 
			
		||||
	Status struct {
 | 
			
		||||
		Token string `json:"token"`
 | 
			
		||||
	} `json:"status"`
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										254
									
								
								test/utils/utils.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										254
									
								
								test/utils/utils.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,254 @@
 | 
			
		||||
/*
 | 
			
		||||
Copyright 2025.
 | 
			
		||||
 | 
			
		||||
Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
you may not use this file except in compliance with the License.
 | 
			
		||||
You may obtain a copy of the License at
 | 
			
		||||
 | 
			
		||||
    http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 | 
			
		||||
Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
distributed under the License is distributed on an "AS IS" BASIS,
 | 
			
		||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
See the License for the specific language governing permissions and
 | 
			
		||||
limitations under the License.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
package utils
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"bufio"
 | 
			
		||||
	"bytes"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"os"
 | 
			
		||||
	"os/exec"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	. "github.com/onsi/ginkgo/v2" // nolint:revive,staticcheck
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	prometheusOperatorVersion = "v0.77.1"
 | 
			
		||||
	prometheusOperatorURL     = "https://github.com/prometheus-operator/prometheus-operator/" +
 | 
			
		||||
		"releases/download/%s/bundle.yaml"
 | 
			
		||||
 | 
			
		||||
	certmanagerVersion = "v1.16.3"
 | 
			
		||||
	certmanagerURLTmpl = "https://github.com/cert-manager/cert-manager/releases/download/%s/cert-manager.yaml"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func warnError(err error) {
 | 
			
		||||
	_, _ = fmt.Fprintf(GinkgoWriter, "warning: %v\n", err)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Run executes the provided command within this context
 | 
			
		||||
func Run(cmd *exec.Cmd) (string, error) {
 | 
			
		||||
	dir, _ := GetProjectDir()
 | 
			
		||||
	cmd.Dir = dir
 | 
			
		||||
 | 
			
		||||
	if err := os.Chdir(cmd.Dir); err != nil {
 | 
			
		||||
		_, _ = fmt.Fprintf(GinkgoWriter, "chdir dir: %q\n", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	cmd.Env = append(os.Environ(), "GO111MODULE=on")
 | 
			
		||||
	command := strings.Join(cmd.Args, " ")
 | 
			
		||||
	_, _ = fmt.Fprintf(GinkgoWriter, "running: %q\n", command)
 | 
			
		||||
	output, err := cmd.CombinedOutput()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return string(output), fmt.Errorf("%q failed with error %q: %w", command, string(output), err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return string(output), nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// InstallPrometheusOperator installs the prometheus Operator to be used to export the enabled metrics.
 | 
			
		||||
func InstallPrometheusOperator() error {
 | 
			
		||||
	url := fmt.Sprintf(prometheusOperatorURL, prometheusOperatorVersion)
 | 
			
		||||
	cmd := exec.Command("kubectl", "create", "-f", url)
 | 
			
		||||
	_, err := Run(cmd)
 | 
			
		||||
	return err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// UninstallPrometheusOperator uninstalls the prometheus
 | 
			
		||||
func UninstallPrometheusOperator() {
 | 
			
		||||
	url := fmt.Sprintf(prometheusOperatorURL, prometheusOperatorVersion)
 | 
			
		||||
	cmd := exec.Command("kubectl", "delete", "-f", url)
 | 
			
		||||
	if _, err := Run(cmd); err != nil {
 | 
			
		||||
		warnError(err)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// IsPrometheusCRDsInstalled checks if any Prometheus CRDs are installed
 | 
			
		||||
// by verifying the existence of key CRDs related to Prometheus.
 | 
			
		||||
func IsPrometheusCRDsInstalled() bool {
 | 
			
		||||
	// List of common Prometheus CRDs
 | 
			
		||||
	prometheusCRDs := []string{
 | 
			
		||||
		"prometheuses.monitoring.coreos.com",
 | 
			
		||||
		"prometheusrules.monitoring.coreos.com",
 | 
			
		||||
		"prometheusagents.monitoring.coreos.com",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	cmd := exec.Command("kubectl", "get", "crds", "-o", "custom-columns=NAME:.metadata.name")
 | 
			
		||||
	output, err := Run(cmd)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
	crdList := GetNonEmptyLines(output)
 | 
			
		||||
	for _, crd := range prometheusCRDs {
 | 
			
		||||
		for _, line := range crdList {
 | 
			
		||||
			if strings.Contains(line, crd) {
 | 
			
		||||
				return true
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return false
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// UninstallCertManager uninstalls the cert manager
 | 
			
		||||
func UninstallCertManager() {
 | 
			
		||||
	url := fmt.Sprintf(certmanagerURLTmpl, certmanagerVersion)
 | 
			
		||||
	cmd := exec.Command("kubectl", "delete", "-f", url)
 | 
			
		||||
	if _, err := Run(cmd); err != nil {
 | 
			
		||||
		warnError(err)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// InstallCertManager installs the cert manager bundle.
 | 
			
		||||
func InstallCertManager() error {
 | 
			
		||||
	url := fmt.Sprintf(certmanagerURLTmpl, certmanagerVersion)
 | 
			
		||||
	cmd := exec.Command("kubectl", "apply", "-f", url)
 | 
			
		||||
	if _, err := Run(cmd); err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
	// Wait for cert-manager-webhook to be ready, which can take time if cert-manager
 | 
			
		||||
	// was re-installed after uninstalling on a cluster.
 | 
			
		||||
	cmd = exec.Command("kubectl", "wait", "deployment.apps/cert-manager-webhook",
 | 
			
		||||
		"--for", "condition=Available",
 | 
			
		||||
		"--namespace", "cert-manager",
 | 
			
		||||
		"--timeout", "5m",
 | 
			
		||||
	)
 | 
			
		||||
 | 
			
		||||
	_, err := Run(cmd)
 | 
			
		||||
	return err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// IsCertManagerCRDsInstalled checks if any Cert Manager CRDs are installed
 | 
			
		||||
// by verifying the existence of key CRDs related to Cert Manager.
 | 
			
		||||
func IsCertManagerCRDsInstalled() bool {
 | 
			
		||||
	// List of common Cert Manager CRDs
 | 
			
		||||
	certManagerCRDs := []string{
 | 
			
		||||
		"certificates.cert-manager.io",
 | 
			
		||||
		"issuers.cert-manager.io",
 | 
			
		||||
		"clusterissuers.cert-manager.io",
 | 
			
		||||
		"certificaterequests.cert-manager.io",
 | 
			
		||||
		"orders.acme.cert-manager.io",
 | 
			
		||||
		"challenges.acme.cert-manager.io",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Execute the kubectl command to get all CRDs
 | 
			
		||||
	cmd := exec.Command("kubectl", "get", "crds")
 | 
			
		||||
	output, err := Run(cmd)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if any of the Cert Manager CRDs are present
 | 
			
		||||
	crdList := GetNonEmptyLines(output)
 | 
			
		||||
	for _, crd := range certManagerCRDs {
 | 
			
		||||
		for _, line := range crdList {
 | 
			
		||||
			if strings.Contains(line, crd) {
 | 
			
		||||
				return true
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return false
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// LoadImageToKindClusterWithName loads a local docker image to the kind cluster
 | 
			
		||||
func LoadImageToKindClusterWithName(name string) error {
 | 
			
		||||
	cluster := "kind"
 | 
			
		||||
	if v, ok := os.LookupEnv("KIND_CLUSTER"); ok {
 | 
			
		||||
		cluster = v
 | 
			
		||||
	}
 | 
			
		||||
	kindOptions := []string{"load", "docker-image", name, "--name", cluster}
 | 
			
		||||
	cmd := exec.Command("kind", kindOptions...)
 | 
			
		||||
	_, err := Run(cmd)
 | 
			
		||||
	return err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// GetNonEmptyLines converts given command output string into individual objects
 | 
			
		||||
// according to line breakers, and ignores the empty elements in it.
 | 
			
		||||
func GetNonEmptyLines(output string) []string {
 | 
			
		||||
	var res []string
 | 
			
		||||
	elements := strings.Split(output, "\n")
 | 
			
		||||
	for _, element := range elements {
 | 
			
		||||
		if element != "" {
 | 
			
		||||
			res = append(res, element)
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return res
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// GetProjectDir will return the directory where the project is
 | 
			
		||||
func GetProjectDir() (string, error) {
 | 
			
		||||
	wd, err := os.Getwd()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return wd, fmt.Errorf("failed to get current working directory: %w", err)
 | 
			
		||||
	}
 | 
			
		||||
	wd = strings.ReplaceAll(wd, "/test/e2e", "")
 | 
			
		||||
	return wd, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// UncommentCode searches for target in the file and remove the comment prefix
 | 
			
		||||
// of the target content. The target content may span multiple lines.
 | 
			
		||||
func UncommentCode(filename, target, prefix string) error {
 | 
			
		||||
	// false positive
 | 
			
		||||
	// nolint:gosec
 | 
			
		||||
	content, err := os.ReadFile(filename)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return fmt.Errorf("failed to read file %q: %w", filename, err)
 | 
			
		||||
	}
 | 
			
		||||
	strContent := string(content)
 | 
			
		||||
 | 
			
		||||
	idx := strings.Index(strContent, target)
 | 
			
		||||
	if idx < 0 {
 | 
			
		||||
		return fmt.Errorf("unable to find the code %q to be uncomment", target)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	out := new(bytes.Buffer)
 | 
			
		||||
	_, err = out.Write(content[:idx])
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return fmt.Errorf("failed to write to output: %w", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	scanner := bufio.NewScanner(bytes.NewBufferString(target))
 | 
			
		||||
	if !scanner.Scan() {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	for {
 | 
			
		||||
		if _, err = out.WriteString(strings.TrimPrefix(scanner.Text(), prefix)); err != nil {
 | 
			
		||||
			return fmt.Errorf("failed to write to output: %w", err)
 | 
			
		||||
		}
 | 
			
		||||
		// Avoid writing a newline in case the previous line was the last in target.
 | 
			
		||||
		if !scanner.Scan() {
 | 
			
		||||
			break
 | 
			
		||||
		}
 | 
			
		||||
		if _, err = out.WriteString("\n"); err != nil {
 | 
			
		||||
			return fmt.Errorf("failed to write to output: %w", err)
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if _, err = out.Write(content[idx+len(target):]); err != nil {
 | 
			
		||||
		return fmt.Errorf("failed to write to output: %w", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// false positive
 | 
			
		||||
	// nolint:gosec
 | 
			
		||||
	if err = os.WriteFile(filename, out.Bytes(), 0644); err != nil {
 | 
			
		||||
		return fmt.Errorf("failed to write file %q: %w", filename, err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user