function-tag-manager is a Crossplane function that allows
Platform Operators to manage Cloud tags on managed resources.
AWS and Azure resources managed by upjet-based providers that support tags are supported, and the function can manage tags for both cluster and namespace-scoped resources.
There several use cases for this Function:
- Allowing external systems to set tags on Crossplane Managed Resources without conflict.
- Adding Common Tags to Resources without having to update every resource in a Composition.
- Allowing users the ability to add their own tags when Requesting new resources.
- Removing tags that have been set earlier in the pipeline by other functions.
The function is installed as a Crossplane Package. Apply the following YAML manifest to your Crossplane cluster.
apiVersion: pkg.crossplane.io/v1beta1
kind: Function
metadata:
name: crossplane-contrib-function-tag-manager
spec:
package: xpkg.upbound.io/crossplane-contrib/function-tag-manager:v0.6.0This function is designed to be a step in a Composition Pipeline after other functions have created Desired State. Below is an example pipeline step:
- step: manage-tags
functionRef:
name: crossplane-contrib-function-tag-manager
input:
apiVersion: tag-manager.fn.crossplane.io/v1beta1
kind: ManagedTags
addTags:
- type: FromValue
policy: Replace
tags:
key1: value1
key2: value2
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.additionalTags
policy: Replace
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.optionalTags
policy: Retain
ignoreTags:
- type: FromValue
policy: Replace
keys:
- external-tag-1
- external-tag-2
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.ignoreTagKeysReplace
policy: Replace
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.ignoreTagKeyRetain
policy: RetainThe addTags field configures tags that will be added to every resource.
The FromValue type indicates that tags will be defined in the function input.
The FromCompositeField type indicates that the tags will be imported from the Composite Resource manifest.
addTags:
- type: FromValue
policy: Replace
tags:
key1: value1
key2: value2
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.additionalTags
policy: Replace
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.optionalTags
policy: RetainThe ignoreTags configures Observed tags in the Cloud that Crossplane will "ignore". In most
cases, Crossplane will attempt to manage every field of a resource, and if a difference is calculated
Crossplane will update the resource and remove fields that are not in the Desired state.
There are many Cloud management systems that set tags on Resources. By adding the keys
to those tags in the ignoreTags section, the function will populate the Desired state with
the values of the Observed tags for each key defined.
Tag keys to ignore can be defined in FromValue or set in the Composite/Claim using FromCompositeFieldPath.
ignoreTags:
- type: FromValue
policy: Replace
keys:
- external-tag-1
- external-tag-2
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.ignoreTagKeysReplace
policy: Replace
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.ignoreTagKeysRetain
policy: RetainAnother option for allowing external systems to manage tags is to use the initProvider field of a Managed Resource.
The function can remove tags defined in the desired state by specifying
removeTags and providing an array of keys to delete.
removeTags:
- type: FromValue
keys:
- fromValue2
- type: FromCompositeFieldPath
fromFieldPath: spec.parameters.removeTagsWhen Merging tags, a Policy can be set:
Replace(default) in the case the desired and observed tags don't match, the observed value will replace desired.Retainin the case the desired and observed tags don't match, the desired value will remain.
This function will skip any resource with the tag-manager.fn.crossplane.io/ignore-resource Kubernetes label set to True or true:
apiVersion: ec2.aws.upbound.io/v1beta1
kind: InternetGateway
metadata:
labels:
tag-manager.fn.crossplane.io/ignore-resource: "True"
name: my-igwThis function supports both AWS and Azure resources that allow setting of tags.
A scan of the AWS provider shows that 498 resources support tags and 477 do not. Starting with the 2.x provider both Cluster and Namespace-scoped resources are supported, so for each resource there are two Custom Resource Definitions.
The AWS Provider CRDs were scanned using cmd/generator/main.go to generate the list in filters/zz_provider-upjet-aws.go.
A scan of the Azure provider shows that 550 resources support tags and 927 do not. Both Cluster-scoped (azure.upbound.io)
and Namespace-scoped (azure.m.upbound.io) API groups are supported.
The Azure Provider CRDs were scanned using the same generator to create the list in filters/zz_provider-upjet-azure.go.
To regenerate the resource filters for both providers:
cd filters
go generate ./...This will clone the provider repositories, scan their CRDs, and regenerate the filter files.
# Run code generation - see input/generate.go
$ go generate ./...
# Run tests
$ go test -cover ./...
ok github.com/crossplane-contrib/function-tag-manager 0.542s coverage: 68.6% of statements
ok github.com/crossplane-contrib/function-tag-manager/cmd/generator 1.035s coverage: 43.2% of statements
github.com/crossplane-contrib/function-tag-manager/cmd/generator/render coverage: 0.0% of statements
github.com/crossplane-contrib/function-tag-manager/filters coverage: 0.0% of statements
github.com/crossplane-contrib/function-tag-manager/input/v1beta1 coverage: 0.0% of statements
# Lint the code
$ docker run --rm -v $(pwd):/app -v ~/.cache/golangci-lint/v2.6.1:/root/.cache -w /app golangci/golangci-lint:v2.6.1 golangci-lint run
0 issues.
# Build a Docker image - see Dockerfile
$ docker build .To test this function, it can be run locally in debug mode:
# Run your Function in insecure mode
go run . --insecure --debugOnce your Function is running, in another window you can use the render command.
# Install Crossplane CLI
$ curl -sL https://raw.githubusercontent.com/crossplane/crossplane/master/install.sh | XP_CHANNEL=stable shTo build the function, run:
docker build . --tag=function-tag-managerPlease note that this command builds an image for your local computer architecture. In general, Crossplane projects build images for linux/amd64 and linux/arm64. See the Github ci.yaml workflow for an example.
Next create the Crossplane Package, embedding the function we just built:
crossplane xpkg build -f package --embed-runtime-image=function-tag-manager -o function-tag-manager.xpkgI use the up binary to push to the Upbound Marketplace
up xpkg push xpkg.upbound.io/crossplane-contrib/function-tag-manager:v0.6.0 -f function-tag-manager.xpkg