Welcome to azcagit
(/ɑːsk/ /eɪ/ /ɡɪt/ - ask a git)!
It's GitOps for Azure Container Apps. You can call it a GitOps Engine for Azure Container Apps.
Please note: This is an early version and will have lots of breaking changes in the future.
Below, large (and eventually breaking) will be documented:
Refactored azcagit
to run as an Azure Container App Job on a schedule. Lots of breaking changes.
BREAKING CHANGES
- trigger-client cli parameter:
--namespace
instead of--fully-qualified-namespace
(note: don't use the full name anymore) - trigger-client cli parameter:
--queue
instead of--topic
- CosmosDB is used for cache
- Service Bus is now basic
Support for AzureContainerJob
was added.
BREAKING CHANGES
spec.remoteSecrets[].appSecretName
has been renamedspec.remoteSecrets[].secretName
apiVersion
has been updated fromaca.xenit.io/v1alpha1
toaca.xenit.io/v1alpha2
for kindAzureContainerApp
In the test scenario, two resource groups are created:
- platform
- tenant
Platform is used for what we call "platform services", in this case the virtual network, Container Apps Managed Environment, Azure Container Registry and things to connect to using Dapr (like Storage Account or a Service Bus). In here, azcagit
will also be setup to take care of the reconciliation of the tenant resource group.
Tenant is used only to synchronize the Container Apps manifests. The Container Apps that are created by azcagit
will reside here.
The manifests are in the same format as Kubernetes manifests (Kubernetes Resource Model aka KRM), but with a hard coupling to the Azure Container Apps specification for spec.app
when using kind: AzureContainerApp
and Azure Container Jobs specification for spec.job
when using kind: AzureContainerJob
. Auto generated schemas can be found in the schemas directory.
An example manifest of an app:
kind: AzureContainerApp
apiVersion: aca.xenit.io/v1alpha2
metadata:
name: foobar
spec:
locationFilter:
- West Europe
remoteSecrets:
- secretName: connection-string
remoteSecretName: mssql-connection-string
replacements:
images:
- imageName: "mcr.microsoft.com/azuredocs/containerapps-helloworld"
newImageTag: "v0.1"
app:
properties:
configuration:
activeRevisionsMode: Single
template:
containers:
- name: simple-hello-world-container
image: mcr.microsoft.com/azuredocs/containerapps-helloworld:latest
resources:
cpu: 0.25
memory: .5Gi
env:
- name: CONNECTION_STRING
secretRef: connection-string
- name: MEANING_WITH_LIFE
value: "forty two"
scale:
minReplicas: 1
maxReplicas: 1
example manifest of a job:
kind: AzureContainerJob
apiVersion: aca.xenit.io/v1alpha2
metadata:
name: foobar
spec:
locationFilter:
- West Europe
remoteSecrets:
- secretName: connection-string
remoteSecretName: mssql-connection-string
replacements:
images:
- imageName: "mcr.microsoft.com/k8se/quickstart-jobs"
newImageTag: "latest"
job:
properties:
configuration:
scheduleTriggerConfig:
cronExpression: "*/5 * * * *"
parallelism: 1
replicaCompletionCount: 1
replicaRetryLimit: 1
replicaTimeout: 1800
triggerType: Schedule
template:
containers:
- name: main
image: mcr.microsoft.com/k8se/quickstart-jobs:foobar
resources:
cpu: 0.25
memory: .5Gi
YAML-files can contain one or more documents (with ---
as a document separator). As of right now, all files in the git repository path (configured with --git-path
when launching azcagit
) needs to pass validation for any deletion to occur (deletion will be disabled if any manifests contains validation errors).
- Synchronize git repository (using https only, public and private) to a specific resource group
- Choose what folder in the git repository to synchronize
- Trigger manual synchronization using CLI
- Populate Container Apps secrets from Azure KeyVault
- Populate Container Apps registries with default registry credential
- Send notifications to the git commits
- Filter locations, making it possible to specify in the manifest what regions can run the app
- Push custom metrics to Azure monitor
- Functionality to replace the image tag using
spec.replacements.images
What happens if a manifest can't be parsed?
Reconciliation will stop an no changes (add/delete/update) will be made until the parse error is fixed.
What happens if a secret in the KeyVault is defined in a manifest but doesn't exist?
Reconciliation will stop an no changes (add/delete/update) will be made until the secret is added to the KeyVault or it's removed from the manifest.
What happens if a secret is changed in the KeyVault?
The Container App will be updated at the next reconcile.
What happens if I add the tag
aca.xenit.io=true
to a Container App in the tenant resource group, without the app being defined in a manifest?
It will be removed at the next reconcile.
What happens if I remove the tag
aca.xenit.io=true
from a Container App in the tenant resource group, while still having a manifest for it?
It won't be reconciled anymore. Depending on the order, a few apps before will still be reconciled but none after.
What happens if I add the tag
aca.xenit.io=true
to a Container App in the tenant resource group, while it's also defined in a manifest?
It will be updated based on the manifest.
What properties, as of now, can't be used even though they are defined in the Azure Container Apps specification?
spec.app.properties.managedEnvironmentID
: it's defined by azcagitspec.app.location
: it's defined by azcagit
What git providers are supported?
Most likely those supported by go-git, but with that said has only Azure DevOps (Azure Repositories) and GitHub been tested. If you need it with on-prem/enterprise variants of them or another git provider doesn't work as expected, create an issue or PR.
Are private git repositories supported?
Yes, as long as you provide credentials in --git-url
like https://username:token@provider.io/repo
.
I'm using a public repository without credentials but
azcagit
throws an error that it needs credentials, isn't it supported to use public repositories without credentials?
It is supported to use public repositories without credentials, but if you have enabled notifications (--notifications-enabled
) then credentials are required to be able to push the git status to the commit.
How does a notification look?
In GitHub, a successful notification looks like this:
Is multi region supported?
It sure is! You can find an example for the setup using terraform here. We've also recorded a short video showing it in action:
What is the location filter feature?
It makes it possible to specify spec.locationFilter
with an array of what Azure regions are allowed to run this specific app.
How does the location filter work?
- No change if
spec.locationFilter
isn't defined - No change if
spec.locationFilter
is an empty list - No change if
spec.locationFilter
contains the location of azcagit (defined with--location
) - If
spec.locationFilter
has a value, of values, where it or none of them match the location of azcagit - we'll skip it (only logged with--debug
enabled)
Where can I find the custom metrics?
If you open the azcagit
container app (in the platform resource group) and go to Monitoring and then Metrics, you can choose the namespace azcagit and then the specific metrics you want to look at.
How does the image tag replacement work?
If an image replacement is configured, it will match for the image name and if found it will apply the newImageTag.
- Append secrets to Container Apps from KeyVault
-
Better error handling of validation failures (should deletion be stopped?)stop reconciliation on any parsing error - Push git commit status (like Flux notification-controller)
- Health checks
- Metrics
- Manually trigger reconcile
- Enforce Location for app
- Add Container Registry credentials by default
- Add location filter
azcagit
will connect to a git repository (over https) and syncronize it on an interval. If changes are identified, it will push them to Azure. It can create, update and delete Azure Container Apps.
The easiest way to test it is using the terraform code which you can find in test/terraform
. You may have to update a few names to get it working.
If you have used the example terraform, there will be a service bus created with a queue. azcagit-trigger
will start and then trigger azcagit-reconcile
when a message is received on the queue.
You can use azcagit-trigger-client
to trigger it:
go run ./trigger-client -n namespace -q queue
Please note that this requires you to be authenticated with either the Azure CLI and have access to publish to this topic with your current user, or use environment varaibles with a service principal that has access.
Create an environment file named .tmp/env
.
The following parameters can be used
RG_NAME=resource_group_name
SUB_ID=azure_subscription_id
ME_ID=azure_container_apps_managed_environment_id
KV_NAME=kvcontainerapps
GIT_URL_AND_CREDS=git_url_with_optional_credentials
The GIT_URL_AND_CREDS
can be in either the format https://github.com/simongottschlag/aca-test-yaml.git
or https://username:secret@github.com/simongottschlag/aca-test-yaml-priv.git
.
To use a service principal with it's client_id and client_secret, add the following environment variables:
TENANT_ID=azure_tenant_id
CLIENT_ID=service_principal_application_id
CLIENT_SECRET=service_principal_secret
Create a file named .tmp/lab.tfvars
and configure the following:
git_config = {
url = "github.com/simongottschlag/aca-test-yaml.git"
branch = "main"
path = "yaml"
username = ""
secret = ""
}
azcagit_version = "vX.Y.Z"
Make sure url doesn't contain https://
, it will be appended by terraform. If you want to use a private repository, add username and secret (PAT). Path is where you want azcagit
to start traversing the directory tree.
Run the following to setup an environment (you may have to change storage account names etc): make terraform-up
This project is licensed under the MIT License - see the LICENSE file for details.