-
Notifications
You must be signed in to change notification settings - Fork 63
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
add templating support #336
Conversation
Signed-off-by: Manabu McCloskey <[email protected]>
love this PR. can't wait for this to be ready for review. we may get some criticism for using comments as a way to template yaml files, similar to how carvel's ytt was under fire at times. we can brainstorm on the right delimeter but otherwise I love the direction here. |
We need this for things that get checked into Gitea as well, not just for Kubernetes manifests. For example, in backstage templates, we currently hard code value like this: https://github.com/cnoe-io/stacks/blob/137a12b8e69067b71622097a0858f14865b6d563/ref-implementation/backstage-templates/entities/basic/skeleton/catalog-info.yaml#L12 These values don't work if you are to deploy this with a custom host or protocol like we do in Codespaces or C9. To work around this, currently we ask users to run a script to replace host, port, etc. This means they can't take advantage of the remote package feature we implemented. e.g. Instead of
They have to do git clone https://github.com....
# run script
some_script.sh
idpbuilder create -p ./<PATH> I think we need this feature. |
More thoughts We need a way to pass runtime information (hostname, protocol, port, etc) to argocd. Without this, packages need to implement their own way to configure their packages. e.g. the replace.sh script in ref-impl. Two approaches I can think of. Store them in a git repository and reference.ArgoCD supports reading helm values from another repository. apiVersion: argoproj.io/v1alpha1
kind: Application
spec:
sources:
- repoURL: 'https://prometheus-community.github.io/helm-charts'
chart: prometheus
targetRevision: 15.7.1
helm:
valueFiles:
- $values/core/values.yaml
- repoURL: "https://cnoe.localtest.me:8443/gitea/giteadmin/idpbuilder-runtime.git"
targetRevision: dev
ref: values This approach is only available to helm. If a user wants to use manifests or kustomize, this won't work. HOWEVER, I think exposing such information in a repository is a good idea because you can use it in ApplicationSet generators. I think we should also expose these information in CM and/or secrets. Templating supportWe can also solve this with templating support. Drawback is that you pretty much have to have this everywhere. networking:
host: #{.host}
protocol: #{.protocol}
port: #{.protocol} |
Another approach is to use argocd custom plugin as suggested by Omri. In the plugin, we can look for specific supported variables, then substitute them with values in secrets or CM. This can be done with a simple envsubst command in bash. This gives us portability. Meaning packages published can be used in idpbuilder as well as non-idpbuilder environments. In this approach, we cannot substitute all environment variables. We MUST set specific variables we support. For example, files like this https://github.com/cnoe-io/stacks/blob/137a12b8e69067b71622097a0858f14865b6d563/ref-implementation/backstage/manifests/install.yaml#L111 must be checked into git as-is. NO substitution. This is only supported by ArgoCD, and we will need another solution if we are to support another CD solution in the future. e.g. flux. This also means we will introduce a dependency to an image and possibly maintain it. Not a big issue imo but does add a bit of overhead. |
Thanks, very interesting summary here! for this statement:
is this something that we can contribute back to Argo CD so that we have consistency between Argo CD and flux? looks like if we land on something that is more generic, we wont corner ourselves into having to write and maintain custom plugins or do the more tricky approach of relying on helm values to cross reference configs. We have done lots of collaboration with the Argo community previously, and I am curious what's the amount of effort to enable passing configs to ArgoCD via CM or secrets. |
This is a fairly often requested feature: argoproj/argo-cd#12060 But solving this doesn't solve our problem. This applies to helm only. Does not work for kustomize or pure manifests. Idpbuilder must support all cases (helm, kustomize, and manifests) for custom packages. This also does not solve our problem like this: https://github.com/cnoe-io/stacks/blob/137a12b8e69067b71622097a0858f14865b6d563/ref-implementation/backstage-templates/entities/basic/skeleton/catalog-info.yaml#L12. e.g. Pushing non-k8s files (does not go through argocd) to git. It really comes down to these questions:
In this PR, I implemented support for templating and passing custom values. Although it does solve all the points above, it is essentially what Helm does at its core. It also makes it so that you can't use them as-is without idpbuilder. That is, idpbuilder must first render them before they can be used IF a user wants to use them without idpbuilder (not sure if this is a valid concern tbh). Are we reinventing the wheel? Is this the right approach? |
I think that the general concept of building our own templates is going to break the transition path from local development to production. I think folks will want to rely on the main open source components (ArgoCD, Backstage, etc;) to do templating and manage it within their system using their standards. I think the problem I have is I don't see folks using this outside of a local development environment. And if they use this it will hinder their ability to transition their setup to a remote cluster, specifically production. The conversation came up as to if we would use this in a remote cluster and really in my dev cycle if I am using a remote cluster I want it as close to production as I want. I think really where I think is "Are people using IDPBuilder as their deployment tool or as a tool in a production environment" and I think the answer is no even if we get to a point where it is production ready because they are going to be married to a CI tool managing Git files and such which is really where I think we should rely on the Open Source side to worry about templating. ArgoCD with Applicationsets supports templating out how the application is made and with Backstage there's ways to use nunjucks to do templating. I think the maintenance overhead of this and breaking the local development to production or real-world usage is why I don't see the value in this personally. |
I would prefer to use something native to ArgoCD like ApplicationSets to pass custom data If not using cluster generator, you could put the data file in git, then use the ApplicationSet git generator to pull in the customer data from plain yaml file seating in git. In the Application CR you would do the overrides with helm:
valuesObect
custom: {{.metadata.annotations.custom}} or kustomize:
patches:
- target:
kind: Deployment
name: external-dns
patch: |-
- op: add
path: /spec/template/spec/containers/0/args/3
value: --txt-owner-id={{.metadata.annotations.custom}} This is an example on how I'm using both generators to pull information from cluster generator and git generator # Source: gitops-bridge/templates/applicationsets.yaml
apiVersion: argoproj.io/v1alpha1
kind: ApplicationSet
metadata:
name: gatekeeper
namespace: argocd
spec:
syncPolicy:
preserveResourcesOnDeletion: true
goTemplate: true
goTemplateOptions:
- missingkey=error
generators:
- matrix:
generators:
- clusters:
selector:
matchLabels:
argocd.argoproj.io/secret-type: cluster
- git:
repoURL: '{{.metadata.annotations.addons_repo_url}}'
revision: HEAD
files:
- path: "**/stack{{.metadata.annotations.platform_stack_version}}.yaml"
template:
metadata:
name: addon-gatekeeper
spec:
project: default
sources:
- repoURL: '{{.metadata.annotations.addons_repo_url}}'
targetRevision: HEAD
ref: values
- chart: '{{.addons.gatekeeper.chart}}'
repoURL: '{{.addons.gatekeeper.repoUrl}}'
targetRevision: '{{.addons.gatekeeper.targetRevision}}'
helm:
releaseName: '{{.addons.gatekeeper.releaseName}}'
ignoreMissingValueFiles: true
valueFiles:
- $values/values/common/{{.addons.gatekeeper.chart}}/values.yaml
- $values/values/{{.metadata.labels.environment}}/{{.addons.gatekeeper.chart}}/values.yaml
- $values/values/clusters/{{.name}}/{{.addons.gatekeeper.chart}}/values.yaml
destination:
namespace: '{{.addons.gatekeeper.namespace}}'
name: '{{.name}}'
syncPolicy:
automated:
prune: false
retry:
limit: 100
syncOptions:
- CreateNamespace=true
- ServerSideApply=true # Big CRDs. |
Before to propose a solution, I think that we should start first to think about the user story we will look like to support top of idpbuilder, next to review the technical solutions available, challenge them (pro/con) and finally take a decision. It is also important that we clarify what we want to do here as the process to customize an internal package could be different from what we could also propose and support for external packages (= user's packages, etc). I also agree with the fact that, when we create a cluster and we deploy the internal packages, then we need a way to customize some of the existing YAML resources as some key(s)/value(s) highly depend on the information like: hostname, port, DNS name, etc ... to simplify the maintenance of the existing code and replace what we coded in some places with a templating engine We should also think about the idea to adopt an idpbuilder configuration file - |
Closing this due to inactivity. Feel free to reopen. |
To be complete here is the PR (= status is draft) developed by argocd to support the proposition reported here and that we could rename into dynamic application parameters: argoproj/argo-cd#12050 |
This adds package templating support for idpbuilder. See #295
-F
. This specifies a yaml file with any additional data.#{
as the delimiter, instead of the usual{{
. This is to avoid collision with other templates. I am open to changing this.File:
pkg/controllers/custompackage/test/resources/customPackages/testDir/params.yaml
File:
pkg/controllers/custompackage/test/resources/customPackages/testDir/app1/cm.yaml
Run idpbuilder:
This results in CM applied with the following config:
We probably should add functions from https://github.com/Masterminds/sprig as well because these are very commonly used in K8s templating stuff.
Closes: #295