Save Hours with DevEx for Crossplane

Take a look at this.

I’ll just say “let there be an XRD” and…

Do not try to run the commands in this section. They are only a preview of what we’ll explore in this post. Full instructions are available starting from the Setup section.

up xrd generate examples/sql/my-db.yaml

code apis/xsqls/definition.yaml

The output is as follows.

apiVersion: apiextensions.crossplane.io/v1
kind: CompositeResourceDefinition
metadata:
  name: xsqls.devopstoolkit.live
spec:
  claimNames:
    kind: SQL
    plural: sqls
  group: devopstoolkit.live
  names:
    categories:
      - crossplane
    kind: XSQL
    plural: xsqls
  versions:
    - name: v1alpha1
      referenceable: true
      schema:
        openAPIV3Schema:
          description: SQL is the Schema for the SQL API.
          properties:
            spec:
              description: SQLSpec defines the desired state of SQL.
              properties:
                id:
                  type: string
                parameters:
                  properties:
                    databases:
                      items:
                        type: string
                      type: array
                    region:
                      type: string
                    size:
                      type: string
                    version:
                      type: string
                  type: object
              type: object
            status:
              description: SQLStatus defines the observed state of SQL.
              type: object
          required:
            - spec
          type: object
      served: true

Boom! Done!

I just saved you half an hour of work.

Is that not enough?

How about saving you much more time by enabling you to define Compositions as code with IntellIsense support in VSCode?

Think about those hundreds or even thousands of lines of YAML you wrote before when you were writing Compositions. Try to imagine how much time would be saved if you wrote code instead of YAML, if you had intellisense support in VSCode which allowed you to use auto-complete based on schemas from dependencies you chose? Think about the possibility to develop Crossplane resources in a language of choice, with full support from your favorite IDE, with commands to build, push, test, deploy the results of your work, with CI/CD pipelines, and all the other goodies we normally associate with development lifecycle.

Would that be something interesting?

If you answered “no” to that last question, I can only assume that you are not familiar with Crossplane, since I cannot imagine why would someone who is using it reject such an offer. If that’s you, if you are new to Crossplane, stop watching this video and go through the Crossplane Tutorial instead. You’ll thank me later.

On the other hand, if you did say “yes”, keep watching. I’ll show you the beginning of something very exciting.

That something is… DevEx or developer experience for Crossplane users, or a way how to implement developer lifecycle for Crossplane, or a way develop Crossplane Compositions instead of doing whatever we were doing before or…

You’ll see. It’s exciting, and I want to jump straight into it.

Before we continue, there is something very important you should be aware of. What I’m about to show you is in very early stage. As a matter of fact, I’m writing this post while what I’m showing is not even public. It’s alpha, or pre-alpha, or pre-pre-alpha. By the time you read this, it probably changed a lot. Commands might have changed. New features have almost certainly been added. It might have been turned upside down. Keep that in mind while reading. I’m showing you very early version of an idea, or a proof of concept. This is the worst version of it you’ll ever see. Your experience should be very different and much better. Heck, if you try to follow my instructions they are likely not going to even work because the solution likely changed drastically in the meantime, hopefully for better. I’m probably making a mistake by sharing this with you, but I was too excited to wait any longer.

Here we go!

Setup

git clone https://github.com/vfarcic/upbound-devex-demo

cd upbound-devex-demo

Watch Nix for Everyone: Unleash Devbox for Simplified Development if you are not familiar with Devbox. Alternatively, you can skip Devbox and install all the tools listed in devbox.json yourself.

devbox shell

Install up CLI by following the instructions.

Replace [...] in the command that follows with your name (e.g., Viktor Farcic)

export NAME="[...]"

Replace [...] in the command that follows with your email

export EMAIL="[...]"

export GIT_URL=$(git config --get remote.origin.url)

Register at Upbound Console if you haven’t already.

Replace [...] with your Upbound account.

export UP_ACCOUNT=[...]

Install KCL Language Server.

Install Visual Studio Code (VSCode) and make sure that code command is working.

Install VSCode plugins KCL and Upbound.

Crossplane (Upbound) Project

This will be a story that starts slow. It’s one of those that the more we progress the more interesting it gets.

It all starts with a project.

If we would like to develop Crossplane Configurations with Composite Definitions, Compositions, and everything else, the first step is to initiate a new project,…

up project init silly-demo

…and enter into it.

cd silly-demo/

Let’s see what we got.

ls

The output is as follows.

-I apis
-I examples
-I functions
-I LICENSE
-I upbound.yaml

We got the apis directory where CompositeDefinitions and Compositions will reside. Then there is functions where the actual code of Compositions will be. Finally, there is upbound.yaml which is similar to Configuration we were defining in crossplane.yaml, but, this time, it is a different kind of resource which we’ll explore soon.

The first major benefit is that now we have a standard. Instead of each of us trying to figure out how to organize Crossplane projects, we have a “standard” way to do so. Standards are good since they help us navigate other people’s work more easily.

Another big improvement is that now everything is integrated with VSCode. For example, we’ll get IntelliSense with code completion which helps tremendously, especially when working with potentially big Compositions.

We’ll see that in action soon. For now, let’s just open VSCode.

code .

Let’s take a look at upbound.yaml that was auto-generated for us when we initialized the project.

code upbound.yaml

The output is as follows.

apiVersion: meta.dev.upbound.io/v1alpha1
kind: Project
metadata:
  creationTimestamp: null
  name: silly-demo
spec:
  description: This is where you can describe your project.
  license: Apache-2.0
  maintainer: Upbound User <user@example.com>
  readme: |
    This is where you can add a readme for your project.    
  repository: xpkg.upbound.io/<YOUR ORGANIZATION>/silly-demo
  source: github.com/upbound/project-template

That file, for now, contains mostly the information that will help us later. That will change, very soon. For now, feel free to update it to match your data like the maintainer, the repository which should point to your Upbound organization, and the source which should point to the Git repository where you might want to keep the project we created.

Open a terminal session in VSCode to execute the commands that follow as well as any other command throughout the rest of this post. Alternatively, when commands are modifying the files (like those that follow), you can make the changes directly in the VSCode editor.

yq --inplace ".spec.maintainer = \"$NAME <$EMAIL>\"" upbound.yaml

yq --inplace \
    ".spec.repository = \"xpkg.upbound.io/$UP_ACCOUNT/silly-demo\"" \
    upbound.yaml

yq --inplace ".spec.source = \"$GIT_URL\"" upbound.yaml

Next, we should probably start thinking about dependencies that are, most of the time, providers and functions that we’ll use in our Compositions. Today, we’ll be building a Composition that should manage PostgreSQL servers in AWS, Azure, and Google Cloud. To do that, we’ll need ec2,…

up dependency add \
    "xpkg.upbound.io/upboundcare/provider-aws-ec2@>=v1.16.0"

… and rds providers for AWS.

up dependency add \
    "xpkg.upbound.io/upboundcare/provider-aws-rds@>=v1.16.0"

Then there is dbforpostgresql for Azure.

up dependency add \
    "xpkg.upbound.io/upbound/provider-azure-dbforpostgresql@>=v1.7.0"

Finally, there is also sql provider for Google Cloud.

up dependency add \
    "xpkg.upbound.io/upbound/provider-gcp-sql@>=v0.33.0"

We’ll also add the generic sql provider that we can use for generic resources that are the same in all three hyperscalers.

up dependency add \
    "xpkg.upbound.io/crossplane-contrib/provider-sql@>=v0.9.0"

We should probably add the kubernetes provider if we would like to have custom-made Kubernetes secrets.

up dependency add \
    "xpkg.upbound.io/crossplane-contrib/provider-kubernetes@>=v0.15.0"

Finally, we’ll add the status-transformer function that might help us to propagate relevant statuses to claims.

up dependency add \
    "xpkg.upbound.io/crossplane-contrib/function-status-transformer@v0.4.0"

Those commands should have simplified management of dependencies and we can see the outcome by taking another look at upbound.yaml.

code upbound.yaml

The output is as follows.

apiVersion: meta.dev.upbound.io/v1alpha1
kind: Project
metadata:
  name: silly-demo
spec:
  dependsOn:
  - provider: xpkg.upbound.io/upboundcare/provider-aws-ec2
    version: '>=v1.16.0'
  - provider: xpkg.upbound.io/upboundcare/provider-aws-rds
    version: '>=v1.16.0'
  - provider: xpkg.upbound.io/upbound/provider-azure-dbforpostgresql
    version: '>=v1.7.0'
  - provider: xpkg.upbound.io/upbound/provider-gcp-sql
    version: '>=v0.33.0'
  - provider: xpkg.upbound.io/crossplane-contrib/provider-sql
    version: '>=v0.9.0'
  - provider: xpkg.upbound.io/crossplane-contrib/provider-kubernetes
    version: '>=v0.15.0'
  - function: xpkg.upbound.io/crossplane-contrib/function-status-transformer
    version: '>=v0.4.0'
  description: This is where you can describe your project.
  license: Apache-2.0
  maintainer: Viktor Farcic <viktor@farcic.com>
  readme: |
    This is where you can add a readme for your project.    
  repository: xpkg.upbound.io/devops-toolkit/silly-demo
  source: https://github.com/vfarcic/upbound-devex-demo

We can see that all the dependencies have been added as dependsOn entries. Some of them are providers (provider) while others are functions (function).

The next chapter in this story will spice it up. As I already mentioned, it gets better with time.

Crossplane Composite Resource Definitions (XRDs)

There are two main types of resources we need to define when trying to build services with Crossplane. There are Composite Resource Definitions and Compositions themselves. Right now we are going to focus on the former.

Typically, we would start the work by writing the CompositeResourceDefinition or XRD. That’s what ultimately becomes Kubernetes Custom Resource Definition. That’s what creates new interfaces the consumers of your service will interact with. That’s what allows people to define claims.

Now, to be honest, writing XRDs is tedious.

Here’s an example.

Please watch https://youtu.be/BII6ZY2Rnlc if you are not familiar with GitHub CLI.

gh browse --repo vfarcic/crossplane-sql package/definition.yaml

Now, I won’t go into details of that XRD. As I already mentioned, today I’m assuming that you are familiar with Crossplane and, more importantly, that you experienced the pain of writing XRDs.

Today, we’ll take a very different approach. Instead of writing XRDs directly, we’ll approach the problem from a different angle. We’ll define examples of what we feel our users should be defining as claims. In other words, we’ll define the end-user experience first, and let the CLI do the heavy lifting of defining the XRD based on those.

We can define baseline claims or composite resources by executing up example generate.

up example generate

The output is as follows.

What do you want to create? [type to search]: 
> Composite Resource Claim (XRC)
  Composite Resource (XR)

That command walks us through a series of questions and will use the answers to generate the initial claim or a composite resource.

While that is great, I’m personally not very fond of wizards so, instead, we’ll execute that same command with the information provided as arguments.

Press ctrl+c to cancel the “wizard”.

up example generate --name my-db \
    --namespace default --type xrc --kind SQL \
    --api-group devopstoolkit.live \
    --api-version v1alpha1

The output is as follows.

Successfully created resource and saved to examples/sql/my-db.yaml

We can see, from the output, that it generated my-db.yaml file, so let’s take a look at what we got.

code examples/sql/my-db.yaml

The output is as follows.

apiVersion: devopstoolkit.live/v1alpha1
kind: SQL
metadata:
  name: my-db
  namespace: default
spec: {}

That’s the base definition of a claim that we expect our users to define.

Replace spec: {} with the YAML that follows in examples/sql/my-db.yaml.

spec:
  id: my-db
  compositionSelector:
    matchLabels:
      provider: aws
      db: postgresql
  parameters:
    version: "16.2"
    size: medium
    region: us-east-1

Let’s spice it up a bit by saying that they should be able to set the id. Since that database server should work in AWS, Azure, and Google Cloud, the user might want to specify which one should be used through matchLabels and, in this case, we’ll say that the provider should be aws and that the db should be postgresql since, at least in theory, there might be an option to use MySQL or some other database server as well.

You’ll notice that, as we type, IntelliSense is showing us what’s wrong by giving us warnings, errors, and suggestions. Nevertheless, that’s only the tip of the iceberg. You’ll see IntelliSense in its true glory later.

We’ll also define a few parameters like the version, set the size to medium, and the region to us-east-1.

What we did is fictitious. There is no kind SQL in our clusters. No one created that CRD. All we did was to define what the users might expect to have as the interface when creating databases.

Think of the phase we just went through as design. Now that we’re done, we should convert all those examples into the actual XRD that will, eventually, become a Kubernetes CRD.

So, we’ll execute up xrd generate with the path to our example, and press the enter key.

up xrd generate examples/sql/my-db.yaml

The output is as follows.

Successfully created CompositeResourceDefinition and saved to apis/xsqls/definition.yaml

Let’s take a look at what we got.

code apis/xsqls/definition.yaml

The output is as follows.

apiVersion: apiextensions.crossplane.io/v1
kind: CompositeResourceDefinition
metadata:
  name: xsqls.devopstoolkit.live
spec:
  claimNames:
    kind: SQL
    plural: sqls
  group: devopstoolkit.live
  names:
    categories:
    - crossplane
    kind: XSQL
    plural: xsqls
  versions:
  - name: v1alpha1
    referenceable: true
    schema:
      openAPIV3Schema:
        description: SQL is the Schema for the SQL API.
        properties:
          spec:
            description: SQLSpec defines the desired state of SQL.
            properties:
              id:
                type: string
              parameters:
                properties:
                  region:
                    type: string
                  size:
                    type: string
                  version:
                    type: string
                type: object
            type: object
          status:
            description: SQLStatus defines the observed state of SQL.
            type: object
        required:
        - spec
        type: object
    served: true

That is a full CompositeResourceDefinition that we would normally define ourselves. That’s the tedious part we just avoided by creating it based on the example we expect users to write when defining the claim. We inverted the order in which we do things. The goal was to focus on the end-user experience and let the CLI generate the XRD that should support it.

From now on, we could fine tune it if there are special cases that were not covered. That won’t be necessary for today’s demo, so let’s move on and see how we can write Compositions in a better way. If what we did so far generated even a mild excitement, what’s coming next will be huge.

Crossplane Compositions

There are two important parts to be aware of when defining Compositions. There is the Composition itself or, to be more precise, the boiler-plate code that is necessary for any Composition. Then there is code that defines which managed resources will be handled by a Composition. Let’s see how we can do both, starting with the boiler-plate code by executing up composition generate, pass the path to the XRD we created earlier, and finishing with the --path where the Composition should be created.

up composition generate apis/xsqls/definition.yaml \
    --path xsqls/aws.yaml

The output is as follows.

successfully created Composition and saved to apis/xsqls/google.yaml

Let’s see what we got.

code apis/xsqls/aws.yaml

The output is as follows.

apiVersion: apiextensions.crossplane.io/v1
kind: Composition
metadata:
  name: xsqls.devopstoolkit.live
spec:
  compositeTypeRef:
    apiVersion: devopstoolkit.live/v1alpha1
    kind: XSQL
  mode: Pipeline
  pipeline:
  - functionRef:
      name: crossplane-contrib-function-status-transformer
    input:
      apiVersion: function-status-transformer.fn.crossplane.io/v1beta1
      kind: StatusTransformation
      metadata: {}
    step: crossplane-contrib-function-status-transformer
  - functionRef:
      name: crossplane-contrib-function-auto-ready
    step: crossplane-contrib-function-auto-ready

That was the boring part that has to be done, yet we did not have to spend any time getting it.

The interesting part is that the Composition we created contains not only the standard parts we always need to have, but it also figured out that, in this case, we want to use the status-transformer and auto-ready functions, so it added those as well.

That was the boring part. The truly interesting stuff comes up next.

We are about to start working on the function that will assemble all managed resources we might want to include into the Composition. We could do that by writing a ton of YAML, but we won’t. That would be silly since using functions is much more efficient and more flexible way to define what we need.

Right now, by the time I’m writing this post, we could use KCL or Python. By the time you’re reading this, additional languages might be supported and, eventually, a ton of them should be available.

Since I prefer KCL and you don’t have a say in it, that’s what we’ll use today, and you should know that’s not the only option if KCL is not your thing.

Here it goes.

We want a function to be generated with the name aws and for the Composition aws.yaml.

up function generate aws apis/xsqls/aws.yaml

The output is as follows.

  ✓   Checking dependencies                                     
  ✓   Generating Function Folder                                
  ✓   Adding Pipeline Step in Composition                       
successfully created Function and saved to /Users/viktorfarcic/code/upbound-devex-demo/silly-demo/functions/aws

That command did quite a few things.

It inspected the dependencies in the project and, based on what we have there, downloaded schemas that will provide us with IntelliSense in VSCode. Through them, we’ll get auto-complete and other nice things we’re used to have when developing something.

On top of that, it skaffolded the files we might need to develop the function that will define all the managed resources we want that Composition to manage.

Let’s see what we got.

ls functions/aws

The output is as follows.

-I kcl.mod
-I kcl.mod.lock
-I main.k
-I model -> ../../../../.up/kcl/models

Since, in this case, I chose to use KCL, it created, among other things, the main.k file which we should modify to suit our needs. Let’s take a look at it.

code functions/aws/main.k

The output is as follows.

import models.v1beta1 as v1beta1
import models.v1beta2 as v1beta2
import models.v1beta3 as v1beta3
import models.k8s.apimachinery.pkg.apis.meta.v1 as metav1

oxr = option("params").oxr # observed composite resource
_ocds = option("params").ocds # observed composed resources
_dxr = option("params").dxr # desired composite resource
dcds = option("params").dcds # desired composed resources

_metadata = lambda name: str -> any {
    { annotations = { "krm.kcl.dev/composition-resource-name" = name }}
}

_items = [

]
items = _items

We can see that it imported (import) the schemas for all the resources available in the dependencies we added earlier.

Further on, it defined variables like oxr that contains the observed composite resource, _ocds with observed composed resources, as well as _dxr and dcds with desired composite and composed resources. Those might look confusing if you’re not used to working with the KCL function. We’ll see them in action very soon.

The important part is the _items array. That’s where we should add managed resources that should be managed by that Composition. We’ll see how to do that in a moment. For now, let’s see how we can use one of the predefined variables.

Let’s say that we would like to extract the region we said we’ll allow users to specify when creating Claims.

We can do that by defining the variable _region and, for now, setting it to us-east-1.

Next, we’ll create a conditional (if) that checks whether spec, and, inside it, parameters, and, inside it, region is set. Those and all other params are available in oxr as the observed composite resource.

If the region is indeed set, we’re putting it as the value of the _region variable.

The final version should look like the snippet that follows.

...
}

_region = "us-east-1"
if oxr.spec?.parameters?.region:
    _region = oxr.spec.parameters.region

_items = [
...

All that was pretty boring. What comes next is what makes a true difference and we can see it if we add the first resource to the _items array.

Let’s say that we would like to add AWS Internet Gateway. In the past, we would need to find out the spec and start typing or copying and pasting parameters. That’s not the case any more. Now we have IntelliSense and, given that it is version v1beta1, we can type just that followed with dot (.) and… Lo and behold. We got the list of all the resources that belong to v1beta1 and we can just start typing Inter, select InternetGateway, and press the enter key.

Next, we should add metadata following the same process. Type meta, select metadata, press the enter key.

We’ll assign it the value of the _metadata lambda.

Let’s do a few more inside the spec.

Type sp, select spec, type . followed with for, select forProvider and assign it to… It autocompleted it for us.

Inside the provider, we should add the region with whatever the value of the _region variable we defined earlier is.

Let’s also set the vpcIdSelector.matchControlelrRef to True.

The final version should look like the snippet that follows.

If you tried to define a managed resource in the past, you must admit that IntelliSense alone is a massive help.

Let’s do an experiment.

I’ll write the code for all the resources and time it. Let’s see how long that will take.

This is how the final version looks like.

code ../aws.k

The output is as follows.

import models.v1beta1 as v1beta1
import models.v1beta2 as v1beta2
import models.v1beta3 as v1beta3
import models.k8s.apimachinery.pkg.apis.meta.v1 as metav1

oxr = option("params").oxr # observed composite resource
_ocds = option("params").ocds # observed composed resources
_dxr = option("params").dxr # desired composite resource
dcds = option("params").dcds # desired composed resources

_metadata = lambda name: str -> any {
    {
        name = oxr.spec.id
        annotations = { "krm.kcl.dev/composition-resource-name" = name }
    }
}

_region = "us-east-1"
if oxr.spec?.parameters?.region:
    _region = oxr.spec.parameters.region

_items = [
    v1beta1.InternetGateway {
        metadata = _metadata("gateway")
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1InternetGatewaySpecForProvider{
            region = _region
            vpcIdSelector.matchControllerRef = True
        }
    },
    v1beta1.MainRouteTableAssociation {
        metadata = _metadata("mainRouteTableAssociation")
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1MainRouteTableAssociationSpecForProvider{
            region = _region
            routeTableIdSelector.matchControllerRef = True
            vpcIdSelector.matchControllerRef = True
        }
    },
    v1beta1.RouteTable {
        metadata = _metadata("routeTable")
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1RouteTableSpecForProvider{
            region = _region
            vpcIdSelector.matchControllerRef = True
        }
    },
    v1beta1.Route {
        metadata = _metadata("route")
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1RouteSpecForProvider{
            region = _region
            routeTableIdSelector.matchControllerRef = True
            destinationCidrBlock = "0.0.0.0/0"
            gatewayIdSelector.matchControllerRef = True
        }
    },
    v1beta1.SecurityGroupRule {
        metadata = _metadata("securityGroupRule")
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1SecurityGroupRuleSpecForProvider{
            region = _region
            description = "I am too lazy to write descriptions"
            type = "ingress"
            fromPort = 5432
            toPort = 5432
            protocol = "tcp"
            cidrBlocks = ["0.0.0.0/0"]
            securityGroupIdSelector.matchControllerRef = True
        }
    },
    v1beta1.SecurityGroup {
        metadata = _metadata("securityGroup")
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1SecurityGroupSpecForProvider{
            region = _region
            description = "I am too lazy to write descriptions"
            vpcIdSelector.matchControllerRef = True
        }
    },
    v1beta1.VPC {
        metadata = _metadata("vpc")
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1VPCSpecForProvider{
            region = _region
            cidrBlock = "11.0.0.0/16"
            enableDnsSupport = True
            enableDnsHostnames = True
        }
    },
    v1beta1.SubnetGroup {
        metadata = _metadata("subnetgroup")
        spec.forProvider = v1beta1.RdsAwsUpboundIoV1beta1SubnetGroupSpecForProvider{
            region = _region
            description = "I'm too lazy to write a good description"
            subnetIdSelector.matchControllerRef = True
        }
    },
    v1beta2.Instance {
        metadata = _metadata("rdsinstance")
        spec.forProvider = v1beta2.RdsAwsUpboundIoV1beta2InstanceSpecForProvider{
            region = _region
            dbSubnetGroupNameSelector.matchControllerRef = True
            vpcSecurityGroupIdSelector.matchControllerRef = True
            username = "masteruser"
            engine = "postgres"
            skipFinalSnapshot = True
            publiclyAccessible = True
            allocatedStorage = 200
            passwordSecretRef = v1beta2.RdsAwsUpboundIoV1beta2InstanceSpecForProviderPasswordSecretRef {
                name = oxr.spec.id + "-password"
                namespace = oxr.spec.claimRef.namespace
                key = "password"
            }
            identifier = oxr.spec.id
            if oxr.spec.parameters.size == "small":
                instanceClass = "db.m5.large"
            elif oxr.spec.parameters.size == "medium":
                instanceClass = "db.m5.2xlarge"
            else:
                instanceClass = "db.m5.8xlarge"
            engineVersion = oxr.spec.parameters.version
        }
    },
    {
        **oxr
        if "rdsinstance" in _ocds:
            status.address = _ocds["rdsinstance"].Resource.status.atProvider.address
    }
]

_zoneList = [
    { zone = "a", cidrBlock = "11.0.0.0/24" },
    { zone = "b", cidrBlock = "11.0.1.0/24"  },
    { zone = "c", cidrBlock = "11.0.2.0/24"  }
]

_routeTableAssociations = [
    v1beta1.RouteTableAssociation {
        metadata = {
            name = oxr.spec.id + "-1" + _data.zone
            annotations = {
                "krm.kcl.dev/composition-resource-name" = "routeTableAssociation1" + _data.zone
            }
        }
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1RouteTableAssociationSpecForProvider{
            region = _region
            routeTableIdSelector.matchControllerRef = True
            subnetIdSelector = v1beta1.Ec2AwsUpboundIoV1beta1RouteTableAssociationSpecForProviderSubnetIDSelector{
                matchControllerRef = True
                matchLabels = { zone = _region + _data.zone }
            }
        }
    } for _data in _zoneList
]

_subnets = [
    v1beta1.Subnet {
        metadata = {
            name = oxr.spec.id + "-" + _data.zone
            annotations = {
                "krm.kcl.dev/composition-resource-name" = "subnet-" + _data.zone
            }
            labels = { zone = _region + _data.zone }
        }
        spec.forProvider = v1beta1.Ec2AwsUpboundIoV1beta1SubnetSpecForProvider{
            region = _region
            availabilityZone = _region + _data.zone
            cidrBlock = _data.cidrBlock
            vpcIdSelector.matchControllerRef = True
        }
    } for _data in _zoneList
]

_objects = [{
    apiVersion = "kubernetes.crossplane.io/v1alpha2"
    kind = "Object"
    metadata = {
        name = oxr.spec.id + "-secret"
        annotations = {
            "krm.kcl.dev/ready": "True"
            "krm.kcl.dev/composition-resource-name" = "sql-secret"
        }
    }
    spec = {
        references = [{
            patchesFrom = {
                apiVersion = "rds.aws.upbound.io/v1beta1"
                kind = "Instance"
                name = oxr.spec.id
                namespace = "crossplane-system"
                fieldPath = "spec.forProvider.username"
            }
            toFieldPath = "stringData.username"
        }, {
            patchesFrom = {
                apiVersion = "v1"
                kind = "Secret"
                name = oxr.spec.id + "-password"
                namespace = oxr.spec.claimRef.namespace
                fieldPath = "data.password"
            }
            toFieldPath = "data.password"
        }, {
            patchesFrom = {
                apiVersion = "rds.aws.upbound.io/v1beta1"
                kind = "Instance"
                name = oxr.spec.id
                namespace = "crossplane-system"
                fieldPath = "status.atProvider.address"
            }
            toFieldPath = "stringData.endpoint"
        }]
        forProvider.manifest = {
            apiVersion = "v1"
            kind = "Secret"
            metadata = {
                name = oxr.spec.id
                namespace = oxr.spec.claimRef.namespace
            }
            data.port = "NTQzMg=="
        }
        providerConfigRef.name = oxr.spec.id + "-sql"
    }
}]

items = _items + _routeTableAssociations + _subnets + _objects

I wrote all that in more or less, 15 minutes. In the past I would probably spend a few hours to accomplish the same result. Isn’t that awesome?

Now, let’s say that we’re finished developing the Composition. The next thing we should do is build and publish the whole project.

Build and Push Projects

All that’s left is to build,…

up project build

…and push it to the registry as version v0.0.1.

up project push --tag v0.0.1

That’s it. The whole project is now published as an OCI image that contains everything we might need to run it in a control plane, and this was the shortest section I ever put into a video.

This Is NOT The End

There are many other goodies coming or, by the time you read this, already available. We should be able to create ephemeral control planes where we could test our Compositions before publishing them. We could have created a permanent production-ready control plane where we would run the Composition we built and allow our users to create claims. We should be able to see the analysis of the changes before applying them. We could have automated the whole process with Workflows like, for example, GitHub Actions or Argo Workflows or Jenkins or whatever we might be using. There are many other things we could have done, and even more that’s coming soon under the umbrella of developer experience.

However, we’re done for today. I’ll explore more in upcoming posts. In the meantime, visit Upbound and check out the new features yourself.

Destroy

up repository delete silly-demo --force

up repository delete silly-demo_aws --force

exit