-
Notifications
You must be signed in to change notification settings - Fork 13
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #165 from k8s-proxmox/feature/tilt
support tilt
- Loading branch information
Showing
4 changed files
with
316 additions
and
20 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,230 @@ | ||
# -*- mode: Python -*- | ||
|
||
envsubst_cmd = "./bin/envsubst" | ||
|
||
load("ext://uibutton", "cmd_button", "location", "text_input") | ||
|
||
# set defaults | ||
settings = { | ||
"capi_version": "v1.5.3", | ||
"allowed_contexts": [ | ||
"kind-cappx", | ||
"default" | ||
], | ||
"extra_args": { | ||
"core": [ | ||
"--feature-gates=MachinePool=false,ClusterResourceSet=true,ClusterTopology=false,RuntimeSDK=false,MachineSetPreflightChecks=false" | ||
] | ||
} | ||
} | ||
|
||
always_enable_providers = ["proxmox"] | ||
providers = {} | ||
extra_args = settings.get("extra_args", {}) | ||
|
||
# global settings | ||
settings.update(read_json( | ||
"tilt-settings.json", | ||
default = {}, | ||
)) | ||
|
||
if "allowed_contexts" in settings: | ||
allow_k8s_contexts(settings.get("allowed_contexts")) | ||
|
||
# deploy core, bootstrap and controlplane provider | ||
def deploy_capi(): | ||
version = settings.get("capi_version") | ||
capi_uri = "https://github.com/kubernetes-sigs/cluster-api/releases/download/{}/cluster-api-components.yaml".format(version) | ||
cmd = "curl -sSL {} | {} | kubectl apply -f -".format(capi_uri, envsubst_cmd) | ||
local(cmd, quiet=True) | ||
if settings.get("extra_args"): | ||
extra_args = settings.get("extra_args") | ||
if extra_args.get("core"): | ||
core_extra_args = extra_args.get("core") | ||
if core_extra_args: | ||
for namespace in ["capi-system"]: | ||
patch_args_with_extra_args(namespace, "capi-controller-manager", 0, core_extra_args) | ||
patch_capi_manager_role_with_exp_infra_rbac() | ||
if extra_args.get("kubeadm-bootstrap"): | ||
kb_extra_args = extra_args.get("kubeadm-bootstrap") | ||
if kb_extra_args: | ||
patch_args_with_extra_args("capi-kubeadm-bootstrap-system", "capi-kubeadm-bootstrap-controller-manager", 1, kb_extra_args) | ||
|
||
def patch_args_with_extra_args(namespace, name, n_container, extra_args): | ||
args_str = str(local('kubectl get deployments {} -n {} -o jsonpath={{.spec.template.spec.containers[{}].args}}'.format(name, namespace, n_container))) | ||
args_to_add = [arg for arg in extra_args if arg not in args_str] | ||
if args_to_add: | ||
args = args_str[1:-1].split() | ||
args.extend(args_to_add) | ||
patch = [{ | ||
"op": "replace", | ||
"path": "/spec/template/spec/containers/{}/args".format(n_container), | ||
"value": args, | ||
}] | ||
local("kubectl patch deployment {} -n {} --type json -p='{}'".format(name, namespace, str(encode_json(patch)).replace("\n", ""))) | ||
|
||
# patch the CAPI manager role to also provide access to experimental infrastructure | ||
def patch_capi_manager_role_with_exp_infra_rbac(): | ||
api_groups_str = str(local('kubectl get clusterrole capi-manager-role -o jsonpath={.rules[1].apiGroups}')) | ||
exp_infra_group = "exp.infrastructure.cluster.x-k8s.io" | ||
if exp_infra_group not in api_groups_str: | ||
groups = api_groups_str[1:-1].split() # "[arg1 arg2 ...]" trim off the first and last, then split | ||
groups.append(exp_infra_group) | ||
patch = [{ | ||
"op": "replace", | ||
"path": "/rules/1/apiGroups", | ||
"value": groups, | ||
}] | ||
local("kubectl patch clusterrole capi-manager-role --type json -p='{}'".format(str(encode_json(patch)).replace("\n", ""))) | ||
|
||
def load_provider_tiltfiles(provider_repos): | ||
for repo in provider_repos: | ||
file = repo + "/tilt-provider.json" | ||
provider_details = read_json(file, default = {}) | ||
if type(provider_details) != type([]): | ||
provider_details = [provider_details] | ||
for item in provider_details: | ||
provider_name = item["name"] | ||
provider_config = item["config"] | ||
if "context" in provider_config: | ||
provider_config["context"] = repo + "/" + provider_config["context"] | ||
else: | ||
provider_config["context"] = repo | ||
if "kustomize_config" not in provider_config: | ||
provider_config["kustomize_config"] = True | ||
if "go_main" not in provider_config: | ||
provider_config["go_main"] = "main.go" | ||
providers[provider_name] = provider_config | ||
|
||
tilt_helper_dockerfile_header = """ | ||
# Tilt image | ||
FROM golang:1.20 as tilt-helper | ||
# Support live reloading with Tilt | ||
RUN wget --output-document /restart.sh --quiet https://raw.githubusercontent.com/windmilleng/rerun-process-wrapper/master/restart.sh && \ | ||
wget --output-document /start.sh --quiet https://raw.githubusercontent.com/windmilleng/rerun-process-wrapper/master/start.sh && \ | ||
chmod +x /start.sh && chmod +x /restart.sh | ||
""" | ||
|
||
tilt_dockerfile_header = """ | ||
FROM gcr.io/distroless/base:debug as tilt | ||
WORKDIR / | ||
COPY --from=tilt-helper /start.sh . | ||
COPY --from=tilt-helper /restart.sh . | ||
COPY manager . | ||
""" | ||
|
||
# Configures a provider by doing the following: | ||
# | ||
# 1. Enables a local_resource go build of the provider's manager binary | ||
# 2. Configures a docker build for the provider, with live updating of the manager binary | ||
# 3. Runs kustomize for the provider's config/ and applies it | ||
def enable_provider(name): | ||
p = providers.get(name) | ||
|
||
name = p.get("name", name) | ||
context = p.get("context") | ||
go_main = p.get("go_main") | ||
label = p.get("label", name) | ||
|
||
# Prefix each live reload dependency with context. For example, for if the context is | ||
# test/infra/docker and main.go is listed as a dep, the result is test/infra/docker/main.go. This adjustment is | ||
# needed so Tilt can watch the correct paths for changes. | ||
live_reload_deps = [] | ||
for d in p.get("live_reload_deps", []): | ||
live_reload_deps.append(context + "/" + d) | ||
|
||
# Set up a local_resource build of the provider's manager binary. The provider is expected to have a main.go in | ||
# manager_build_path or the main.go must be provided via go_main option. The binary is written to .tiltbuild/manager. | ||
local_resource( | ||
label.lower() + "_binary", | ||
cmd = "cd " + context + ';mkdir -p .tiltbuild;CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build -ldflags \'-extldflags "-static"\' -o .tiltbuild/manager ' + go_main, | ||
deps = live_reload_deps, | ||
labels = [label, "ALL.binaries"], | ||
) | ||
|
||
additional_docker_helper_commands = p.get("additional_docker_helper_commands", "") | ||
additional_docker_build_commands = p.get("additional_docker_build_commands", "") | ||
|
||
dockerfile_contents = "\n".join([ | ||
tilt_helper_dockerfile_header, | ||
additional_docker_helper_commands, | ||
tilt_dockerfile_header, | ||
additional_docker_build_commands, | ||
]) | ||
|
||
# Set up an image build for the provider. The live update configuration syncs the output from the local_resource | ||
# build into the container. | ||
entrypoint = ["sh", "/start.sh", "/manager"] | ||
provider_args = extra_args.get(name) | ||
if provider_args: | ||
entrypoint.extend(provider_args) | ||
|
||
docker_build( | ||
ref = p.get("image"), | ||
context = context + "/.tiltbuild/", | ||
dockerfile_contents = dockerfile_contents, | ||
target = "tilt", | ||
entrypoint = entrypoint, | ||
only = "manager", | ||
live_update = [ | ||
sync(context + "/.tiltbuild/manager", "/manager"), | ||
run("sh /restart.sh"), | ||
], | ||
) | ||
|
||
if p.get("kustomize_config"): | ||
# Copy all the substitutions from the user's tilt-settings.json into the environment. Otherwise, the substitutions | ||
# are not available and their placeholders will be replaced with the empty string when we call kustomize + | ||
# envsubst below. | ||
substitutions = settings.get("kustomize_substitutions", {}) | ||
os.environ.update(substitutions) | ||
|
||
# Apply the kustomized yaml for this provider | ||
|
||
yaml = str(kustomize(context + "/config/default")) | ||
k8s_yaml(blob(yaml)) | ||
get_controller_name(name) | ||
else: | ||
get_controller_name(name) | ||
|
||
# Enable core cluster-api plus everything listed in 'enable_providers' in tilt-settings.json | ||
def enable_providers(): | ||
# local("make kustomize") | ||
|
||
provider_repos = settings.get("provider_repos", []) | ||
union_provider_repos = [ k for k in provider_repos + ["."] ] | ||
load_provider_tiltfiles(union_provider_repos) | ||
|
||
user_enable_providers = settings.get("enable_providers", []) | ||
union_enable_providers = {k: "" for k in user_enable_providers + always_enable_providers}.keys() | ||
for name in union_enable_providers: | ||
enable_provider(name) | ||
|
||
def get_controller_name(name): | ||
p = providers.get(name) | ||
name = p.get("name", name) | ||
label = p.get("label", name) | ||
manager_name = p.get("manager_name") | ||
if manager_name: | ||
k8s_resource( | ||
workload = manager_name, | ||
new_name = label.lower() + "_controller", | ||
labels = [label, "ALL.controllers"], | ||
) | ||
|
||
def kustomize(folder): | ||
yaml = local('./bin/kustomize build {}'.format(folder), quiet=True) | ||
return yaml | ||
|
||
############################## | ||
# Actual work happens here | ||
############################## | ||
|
||
load("ext://cert_manager", "deploy_cert_manager") | ||
|
||
if settings.get("deploy_cert_manager"): | ||
deploy_cert_manager(version=settings.get("cert_manager_version")) | ||
|
||
deploy_capi() | ||
|
||
enable_providers() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
{ | ||
"name": "proxmox", | ||
"config": { | ||
"image": "ghcr.io/sp-yduck/cluster-api-provider-proxmox", | ||
"live_reload_deps": [ | ||
"api", | ||
"cloud", | ||
"cmd/main.go", | ||
"config", | ||
"controllers", | ||
"go.mod", | ||
"go.sum" | ||
], | ||
"label": "CAPPX", | ||
"manager_name": "cappx-controller-manager", | ||
"go_main": "cmd/main.go" | ||
} | ||
} |