A Crossplane composition function that lets you compose Composites using a set of python classes enabling an elegant and terse syntax. Here is what the following example is doing:
- Create an MR named 'vpc' with apiVersion 'ec2.aws.crossplane.io/v1beta1' and kind 'VPC'
- Set the vpc region and cidr from the XR spec values
- Set the XR status.vpcId to the created vpc id
apiVersion: apiextensions.crossplane.io/v1
kind: Composition
metadata:
name: create-vpc
spec:
compositeTypeRef:
apiVersion: example.crossplane.io/v1
kind: XR
mode: Pipeline
pipeline:
- step:
functionRef:
name: function-pythonic
input:
apiVersion: pythonic.fn.crossplane.io/v1beta1
kind: Composite
composite: |
class Composite(BaseComposite):
def compose(self):
vpc = self.resources.vpc('ec2.aws.crossplane.io/v1beta1', 'VPC')
vpc.spec.forProvider.region = self.spec.region
vpc.spec.forProvider.cidrBlock = self.spec.cidr
self.status.vpcId = vpc.status.atProvider.vpcId
In addtion to an inline script, the python implementation can be specified as the complete path to a python class. See Filing system Composites.
In the examples directory are many exemples, including all of the function-go-templating examples implemented using function-pythonic. The eks-cluster example is a good complex example creating the entire vpc structure needed for an EKS cluster.
apiVersion: pkg.crossplane.io/v1
kind: Function
metadata:
name: function-pythonic
spec:
package: ghcr.io/fortra/function-pythonic:v0.0.3
function-pythonic automatically handles dependencies between composed resources.
Just compose everything as if it is immediately created and the framework will delay the creation of any resources which depend on other resources which do not exist yet. In other words, it accomplishes what function-sequencer provides, but it automatically detects the dependencies.
If a resource has been created and a dependency no longer exists due to some unexpected
condition, the composition will be terminated or the observed value for that field will
be used, depending on the unknownsFatal
settings.
Take the following example:
vpc = self.resources.VPC('ec2.aws.crossplane.io/v1beta1', 'VPC')
vpc.spec.forProvider.region = 'us-east-1
vpc.spec.forProvider.cidrBlock = '10.0.0.0/16'
subnet = self.resources.SubnetA('ec2.aws.crossplane.io/v1beta1', 'Subnet')
subnet.spec.forProvider.region = 'us-east-1'
subnet.spec.forProvider.vpcId = vpc.status.atProvider.vpcId
subnet.spec.forProvider.availabilityZone = 'us-east-1a'
subnet.spec.forProvider.cidrBlock = '10.0.0.0/20'
If the Subnet does not yet exist, the framework will detect if the vpcId set in the Subnet is unknown, and will delay the creation of the subnet.
Once the Subnet has been created, if for some unexpected reason the vpcId passed
to the Subnet is unknown, the framework will detect it and either terminate
the Composite composition or use the vpcId in the observed Subnet. The default
action taken is to fast fail by terminating the composition. This can be
overridden for all composed resource by setting the Composite self.unknownsFatal
field
to False, or at the individual composed resource level by setting the
Resource.unknownsFatal
field to False.
All Protobuf messages are wrapped by a set of python classes which enable using both object attribute names and dictionary key names to traverse the Protobuf message contents. For example, the following examples obtain the same value from the RunFunctionRequest message:
region = request.observed.composite.resource.spec.region
region = request['observed']['composite']['resource']['spec']['region']
Getting values from free form map and list values will not throw errors for keys that do not exist, but will return an unknown placeholder which evaluates as False. For example, the following will evaluate as False with a just created RunFunctionResponse message:
vpcId = response.desired.resources.vpc.resource.status.atProvider.vpcId
if vpcId:
# The vpcId is available
Note that maps or lists that do exist but do not have any members will evaluate
as True, contrary to Python dicts and lists. Use the len
function to test
if the map or list exists and has members.
When setting fields, all intermediary unknown placeholders will automatically be created. For example, this will create all items needed to set the region on the desired resource:
response.desired.resources.vpc.resource.spec.forProvider.region = 'us-east-1'
Calling a message or map will clear it and will set any provided key word arguments. For example, this will either create or clear the resource and then set its apiVersion and kind:
response.desired.resources.vpc.resource(apiVersion='ec2.aws.crossplane.io/v1beta1', kind='VPC')
The following functions are provided to create Protobuf structures:
Function | Description |
---|---|
Map | Create a new Protobuf map |
List | Create a new Protobuf list |
Unknown | Create a new Protobuf unknown placeholder |
Yaml | Create a new Protobuf structure from a yaml string |
Json | Create a new Protobuf structure from a json string |
B64Encode | Encode a string into base 64 |
B64Decode | Decode a string from base 64 |
The following items are supported in all the Protobuf Message wrapper classes: bool
,
len
, contains
, iter
, hash
, ==
, str
, format
To convert a Protobuf message to a string value, use either str
or format
.
yaml = str(request) # get the request as yaml
yaml = format(request) # also get the request as yaml
yaml = format(request, 'yaml') # yet another get the request as yaml
json = format(request, 'json') # get the request as json
json = format(request, 'jsonc') # get the request as json compact
proto = format(request, 'protobuf') # get the request as a protobuf string
Composite composition is performed from a Composite orientation. A BaseComposite
class
is subclassed and the compose
method is implemented.
class Composite(BaseComposite):
def compose(self):
# Compose the Composite
The compose method can also declare itself as performing async io:
class Composite(BaseComposite):
async def compose(self):
# Compose the Composite using async io when needed
The BaseComposite class provides the following fields for manipulating the Composite itself:
Field | Type | Description |
---|---|---|
self.observed | Map | Low level direct access to the observed composite |
self.desired | Map | Low level direct access to the desired composite |
self.apiVersion | String | The composite observed apiVersion |
self.kind | String | The composite observed kind |
self.metadata | Map | The composite observed metadata |
self.spec | Map | The composite observed spec |
self.status | Map | The composite desired and observed status, read from observed if not in desired |
self.conditions | Conditions | The composite desired and observed conditions, read from observed if not in desired |
self.connection | Connection | The composite desired and observed connection detials, read from observed if not in desired |
self.results | Results | Returned results on the Composite and optionally on the Claim |
self.ready | Boolean | The composite desired ready state |
The BaseComposite also provides access to the following Crossplane Function level features:
Field | Type | Description |
---|---|---|
self.request | Message | Low level direct access to the RunFunctionRequest message |
self.response | Message | Low level direct access to the RunFunctionResponse message |
self.logger | Logger | Python logger to log messages to the running function stdout |
self.ttl | Integer | Get or set the response TTL, in seconds |
self.credentials | Credentials | The request credentials |
self.context | Map | The response context, initialized from the request context |
self.environment | Map | The response environment, initialized from the request context environment |
self.requireds | Requireds | Request and read additional local Kubernetes resources |
self.resources | Resources | Define and process composed resources |
self.unknownsFatal | Boolean | Terminate the composition if already created resources are assigned unknown values, default True |
self.autoReady | Boolean | Perform auto ready processing on all composed resources, default True |
Creating and accessing composed resources is performed using the BaseComposite.resources
field.
BaseComposite.resources
is a dictionary of the composed resources whose key is the composition
resource name. The value returned when getting a resource from BaseComposite is the following
Resource class:
Field | Type | Description |
---|---|---|
Resource(apiVersion,kind,namespace,name) | Resource | Reset the resource and set the optional parameters |
Resource.name | String | The composition composed resource name |
Resource.observed | Map | Low level direct access to the observed composed resource |
Resource.desired | Map | Low level direct access to the desired composed resource |
Resource.apiVersion | String | The composed resource apiVersion |
Resource.kind | String | The composed resource kind |
Resource.externalName | String | The composed resource external name |
Resource.metadata | Map | The composed resource desired metadata |
Resource.spec | Map | The resource spec |
Resource.data | Map | The resource data |
Resource.status | Map | The resource status |
Resource.conditions | Conditions | The resource conditions |
Resource.connection | Connection | The resource connection details |
Resource.ready | Boolean | The resource ready state |
Resource.unknownsFatal | Boolean | Terminate the composition if this resource has been created and is assigned unknown values, default is Composite.unknownsFatal |
Resource.autoReady | Boolean | Perform auto ready processing on this resource, default is Composite.autoReady |
Creating and accessing required resources is performed using the BaseComposite.requireds
field.
BaseComposite.requireds
is a dictionary of the required resources whose key is the required
resource name. The value returned when getting a required resource from BaseComposite is the
following RequiredResources class:
Field | Type | Description |
---|---|---|
RequiredResource(apiVersion,kind,namespace,name,labels) | RequiredResource | Reset the required resource and set the optional parameters |
RequiredResources.name | String | The required resources name |
RequiredResources.apiVersion | String | The required resources apiVersion |
RequiredResources.kind | String | The required resources kind |
RequiredResources.namespace | String | The namespace to match when returning the required resources, see note below |
RequiredResources.matchName | String | The names to match when returning the required resources |
RequiredResources.matchLabels | Map | The labels to match when returning the required resources |
The current version of crossplane-sdk-python used by function-pythonic does not support namespace selection. For now, use matchLabels and filter the results if required.
RequiredResources acts like a Python list to provide access to the found required resources. Each resource in the list is the following RequiredResource class:
Field | Description |
---|---|
RequiredResource.name | The required resource name |
RequiredResource.observed | Low level direct access to the observed required resource |
RequiredResource.apiVersion | The required resource apiVersion |
RequiredResource.kind | The required resource kind |
RequiredResource.metadata | The required resource metadata |
RequiredResource.spec | The required resource spec |
RequiredResource.data | The required resource data |
RequiredResource.status | The required resource status |
RequiredResource.conditions | The required resource conditions |
The conditions
field is a map of the resource's status conditions array, with
the map key being the condition type.
Field | Description |
---|---|
Condition.type | The condtion type |
Condition.status | RequiredResource.observed |
RequiredResource.apiVersion | The required resource apiVersion |
RequiredResource.kind | The required resource kind |
RequiredResource.metadata | The required resource metadata |
RequiredResource.spec | The required resource spec |
RequiredResource.data | The required resource data |
RequiredResource.status | The required resource status |
RequiredResource.conditions | The required resource conditions |
Tired of creating a CompositeResourceDefinition, a Composition, and a Composite just to run that Composition once in a single use or initialize task?
function-pythonic installs a Composite
CompositeResourceDefinition that enables
creating such tasks using a single Composite resource:
apiVersion: pythonic.fortra.com/v1alpha1
kind: Composite
metadata:
name: composite-example
spec:
composite: |
class Composite(BaseComposite):
def compose(self):
self.status.composite = 'Hello, World!'
Composition Composite implementations can be coded in a stand alone python files by configuring the function-pythonic deployment with the code mounted into the package-runtime container, and then adding the mount point to the python path using the --python-path command line option.
apiVersion: pkg.crossplane.io/v1beta1
kind: DeploymentRuntimeConfig
metadata:
name: function-pythonic
spec:
deploymentTemplate:
spec:
template:
spec:
containers:
- name: package-runtime
args:
- --debug
- --python-path
- /mnt/composites
volumeMounts:
- name: composites
mountPath: /mnt/composites
volumes:
- name: composites
configMap:
name: pythonic-composites
See the filing-system example.
function-pythonic supports a --pip-install
command line option which will run pip install
with the configured pip install command. For example:
apiVersion: pkg.crossplane.io/v1beta1
kind: DeploymentRuntimeConfig
metadata:
name: function-pythonic
spec:
deploymentTemplate:
spec:
template:
spec:
containers:
- name: package-runtime
args:
- --debug
- --pip-install
- --quiet aiobotocore==2.23.2
The Protobuf python package used by function-pythonic limits the depth of yaml
elements and the total size of yaml parsed. This results in a limit of approximately
30 levels of nested yaml fields. This check can be disabled using the --allow-oversize-protos
command line option. For example:
apiVersion: pkg.crossplane.io/v1beta1
kind: DeploymentRuntimeConfig
metadata:
name: function-pythonic
spec:
deploymentTemplate:
spec:
template:
spec:
containers:
- name: package-runtime
args:
- --debug
- --allow-oversize-protos