Back to Crossplane

Composite Resources

content/v2.0/composition/composite-resources.md

1.19-archive10.4 KB
Original Source

A composite resource, or XR, represents a set of Kubernetes resources as a single Kubernetes object. Crossplane creates composite resources when users access a custom API, defined in the CompositeResourceDefinition.

{{<hint "tip" >}} Composite resources are a composite of Kubernetes resources.
A Composition defines how to compose the resources together. {{< /hint >}}

{{<expand "What are XRs, XRDs and Compositions?" >}} A composite resource or XR (this page) is a custom API.

You use two Crossplane types to create a new custom API:

  • A [Composite Resource Definition]({{<ref "./composite-resource-definitions">}}) (XRD) - Defines the XR's schema.
  • A [Composition]({{<ref "./compositions" >}}) - Configures how the XR creates other resources. {{</expand >}}

Create composite resources

Creating composite resources requires a [Composition]({{<ref "./compositions">}}) and a [CompositeResourceDefinition]({{<ref "./composite-resource-definitions">}}) (XRD).

The Composition defines the set of resources to create. The XRD defines the custom API users call to request the set of resources.

mermaid
flowchart TD

user(["User"])
xr("Composite Resource (XR)")
xrd("Composite Resource Definition (XRD)")
comp("Composition")
cda("Composed Resource A")
cdb("Composed Resource B")
cdc("Composed Resource C")

xrd -.defines.-> xr
comp [email protected]> xr
user --creates--> xr
xr compose-a@--composes-->cda
xr compose-b@--composes-->cdb
xr compose-c@--composes-->cdc

configure-xr@{animate: true}
compose-a@{animate: true}
compose-b@{animate: true}
compose-c@{animate: true}

XRDs define the API used to create a composite resource. For example, this {{<hover label="xrd1" line="2">}}CompositeResourceDefinition{{</hover>}} creates a custom API endpoint {{<hover label="xrd1" line="4">}}mydatabases.example.org{{</hover>}}.

yaml
apiVersion: apiextensions.crossplane.io/v1
kind: CompositeResourceDefinition
metadata: 
  name: mydatabases.example.org
spec:
  group: example.org
  names:
    kind: MyDatabase
    plural: mydatabases
  # Removed for brevity

When a user calls the custom API, {{<hover label="xrd1" line="4">}}mydatabases.example.org{{</hover>}}, Crossplane chooses the Composition to use based on the Composition's {{<hover label="typeref" line="6">}}compositeTypeRef{{</hover>}}

yaml
apiVersion: apiextensions.crossplane.io/v1
kind: Composition
metadata:
  name: my-composition
spec:
  compositeTypeRef:
    apiVersion: example.org/v1alpha1
    kind: MyDatabase
  # Removed for brevity

The Composition {{<hover label="typeref" line="6">}}compositeTypeRef{{</hover>}} matches the XRD {{<hover label="xrd1" line="6">}}group{{</hover>}} and {{<hover label="xrd1" line="9">}}kind{{</hover>}}.

Crossplane creates the resources defined in the matching Composition and represents them as a single composite resource.

shell{copy-lines="1"}
kubectl get composite
NAME                    SYNCED   READY   COMPOSITION         AGE
my-composite-resource   True     True    my-composition      4s

Composition selection

Select a specific Composition for a composite resource to use with {{<hover label="compref" line="7">}}compositionRef{{</hover>}}

{{<hint "important">}} The selected Composition must allow the composite resource to use it with a compositeTypeRef. Read more about the compositeTypeRef field in the [Enable Composite Resources]({{<ref "./compositions#match-composite-resources">}}) section of the Composition documentation. {{< /hint >}}

yaml
apiVersion: example.org/v1alpha1
kind: MyDatabase
metadata:
  namespace: default
  name: my-composite-resource
spec:
  crossplane:
    compositionRef:
      name: my-other-composition
  # Removed for brevity

A composite resource can also select a Composition based on labels instead of the exact name with a {{<hover label="complabel" line="7">}}compositionSelector{{</hover>}}.

Inside the {{<hover label="complabel" line="7">}}matchLabels{{</hover>}} section provide one or more Composition labels to match.

yaml
apiVersion: example.org/v1alpha1
kind: MyDatabase
metadata:
  namespace: default
  name: my-composite-resource
spec:
  crossplane:
    compositionSelector:
      matchLabels:
        environment: production
    # Removed for brevity

Composition revision policy

Crossplane tracks changes to Compositions as [Composition revisions]({{<ref "composition-revisions">}}) .

A composite resource can use a {{<hover label="comprev" line="7">}}compositionUpdatePolicy{{</hover>}} to manually or automatically reference newer Composition revisions.

The default {{<hover label="comprev" line="7">}}compositionUpdatePolicy{{</hover>}} is "Automatic." Composite resources automatically use the latest Composition revision.

Change the policy to {{<hover label="comprev" line="7">}}Manual{{</hover>}} to prevent composite resources from automatically upgrading.

yaml
apiVersion: example.org/v1alpha1
kind: MyDatabase
metadata:
  namespace: default
  name: my-composite-resource
spec:
  crossplane:
    compositionUpdatePolicy: Manual
    # Removed for brevity

Composition revision selection

Crossplane records changes to Compositions as [Composition revisions]({{<ref "composition-revisions">}}).
A composite resource can select a specific Composition revision.

Use {{<hover label="comprevref" line="7">}}compositionRevisionRef{{</hover>}} to select a specific Composition revision by name.

For example, to select a specific Composition revision use the name of the desired Composition revision.

yaml
apiVersion: example.org/v1alpha1
kind: MyDatabase
metadata:
  namespace: default
  name: my-composite-resource
spec:
  crossplane:
    compositionUpdatePolicy: Manual
    compositionRevisionRef:
      name: my-composition-b5aa1eb
    # Removed for brevity

{{<hint "note" >}} Find the Composition revision name from {{<hover label="getcomprev" line="1">}}kubectl get compositionrevision{{</hover>}}

shell
kubectl get compositionrevision
NAME                         REVISION   XR-KIND        XR-APIVERSION            AGE
my-composition-5c976ad       1          mydatabases    example.org/v1alpha1     65m
my-composition-b5aa1eb       2          mydatabases    example.org/v1alpha1     64m

{{< /hint >}}

A Composite resource can also select Composition revisions based on labels instead of the exact name with a {{<hover label="comprevsel" line="7">}}compositionRevisionSelector{{</hover>}}.

Inside the {{<hover label="comprevsel" line="7">}}matchLabels{{</hover>}} section provide one or more Composition revision labels to match.

yaml
apiVersion: example.org/v1alpha1
kind: MyDatabase
metadata:
  namespace: default
  name: my-composite-resource
spec:
  crossplane:
    compositionRevisionSelector:
      matchLabels:
        channel: dev
    # Removed for brevity

Pausing composite resources

<!-- vale Google.WordList = NO -->

Crossplane supports pausing composite resources. A paused composite resource doesn't check or make changes on its external resources.

<!-- vale Google.WordList = YES -->

To pause a composite resource apply the {{<hover label="pause" line="4">}}crossplane.io/paused{{</hover>}} annotation.

yaml
apiVersion: example.org/v1alpha1
kind: MyDatabase
metadata:
  namespace: default
  name: my-composite-resource
  annotations:
    crossplane.io/paused: "true"
spec:
  # Removed for brevity

Verify composite resources

Use {{<hover label="getcomposite" line="1">}}kubectl get composite{{</hover>}} to view all the composite resources Crossplane created.

shell{copy-lines="1",label="getcomposite"}
kubectl get composite
NAME                    SYNCED   READY   COMPOSITION         AGE
my-composite-resource   True     True    my-composition      4s

Use kubectl get for the specific custom API endpoint to view only those resources.

shell
kubectl get mydatabases
NAME                    SYNCED   READY   COMPOSITION        AGE
my-composite-resource   True     True    my-composition     12m

Use {{<hover label="desccomposite" line="1">}}kubectl describe composite{{</hover>}} to view the linked {{<hover label="desccomposite" line="16">}}Composition Ref{{</hover>}}, and unique resources created in the {{<hover label="desccomposite" line="22">}}Resource Refs{{</hover>}}.

yaml
kubectl describe composite my-composite-resource
Name:         my-composite-resource
Namespace:    default
API Version:  example.org/v1alpha1
Kind:         MyDatabase
Spec:
  Composition Ref:
    Name:  my-composition
  Composition Revision Ref:
    Name:                     my-composition-cf2d3a7
  Composition Update Policy:  Automatic
  Resource Refs:
    API Version:  s3.aws.m.upbound.io/v1beta1
    Kind:         Bucket
    Name:         my-composite-resource-fmrks
    API Version:  dynamodb.aws.m.upbound.io/v1beta1
    Kind:         Table
    Name:         my-composite-resource-wnr9t
# Removed for brevity

Composite resource conditions

<!-- vale Google.Colons = NO -->

A composite resource has two status conditions: Synced and Ready.

<!-- vale Google.Colons = YES -->

Crossplane sets the Synced status condition to True when it's able to successfully reconcile the composite resource. If Crossplane can't reconcile the composite resource it reports an error in the Synced condition.

Crossplane sets the Ready status condition to True when the composite resource's composition function pipeline reports that all its composed resources are ready. If a composed resource isn't ready Crossplane reports it in the Ready condition.

Composite resource labels

Crossplane adds labels to composed resources to show their relationship to other Crossplane components.

Crossplane adds the {{<hover label="complabel" line="4">}} crossplane.io/composite{{</hover>}} label to all composed resources. The label matches the name of the composite. Crossplane applies the composite label to any resource created by a composite, creating a reference between the resource and owning composite resource.

shell
kubectl describe mydatabase.example.org/my-database-x9rx9
Name:         my-database2-x9rx9
Namespace:    default
Labels:       crossplane.io/composite=my-database-x9rx9