EDABuilder#
edabuilder is a standalone CLI tool for performing EDA Store operations such as building and publishing EDA apps or scaffolding the boilerplate code for new apps. The download instructions are mentioned in Environment setup page.
edabuilder has the following capabilities:
- init: initialise a directory as a project that will hold multiple apps.
- create app: scaffold the boilerplate for a new application.
- create resource: create a new API resource. These resources are the inputs and outputs of your app, i.e. the main way to interact with your app.
- create intent: scaffold boilerplate code for a new config/state intent for one of your API resources, if you hadn't done so already at the time of creating the resource.
- create version: create a new API version.
- build-push: package an app and host it in a registry1.
- publish: publish your app to a catalog1.
- generate: based on your API resources, defined in the Go files under- api/, generate equivalent python models to be used as an SDK in your intents, CRDs, OpenAPI specs, and keep your manifest up-to-date.
- generate appsettings: Scaffold a new- settingsdirectory, which you can further customise to provide documentation for your app's settings.
- generate appsettings-openapi: Generate an OpenAPI spec based on the settings struct in your- settingsdirectory.
- deploy: build, publish and install your application in your EDA cluster, allowing you to test it immediately and iteratively, during development1.
Working directories
edabuilder will run using the current working directory as context. You should run project related commands from the root of your project (edabuilder init for example), and application related commands from the root of your applications (edabuilder create for example).
Initialising a project#
When you run edabuilder init [<project-name>], either the current directory or a new one, named project-name, will be initialised for app development. This entails populating a couple of directories (common/, test/, utils/) with common libraries to be imported in the app scripts you will be writing, as well as creating a PROJECT file that keeps track of some project-wide parameters, such as the publishing authority of your apps, your production registry, etc. These parameters can all be passed via arguments to the command. To see the available options, run edabuilder init -h.
Creating an app#
Running edabuilder create app <app-name> from the root of your project will net you a new directory containing a scaffolded structure for an app with a couple of subdirectories created in it. They all have self-explanatory, categorised, names that suggest where to put app files. Additionally, a manifest.yaml file is created. The edabuilder tool will maintain parts of this manifest along the way. More on that in the other sections.
Creating a resource#
As stated earlier, resources are the main way to interact with your app. Run edabuilder create resource MyCoolKind to scaffold a barebones resource: the "API definition" is the Go file located at api/<apiversion>/mycoolkind_types.go.
From this file, the equivalent python models are generated at api/<apiversion>/pysrc/mycoolkind.py, as well as a CRD under crds/ and the OpenAPI spec under openapiv3/. Your manifest is automatically kept up-to-date with this change, meaning this new pysrc and the CRD are ready to be packaged in your app already.
Creating an intent#
In and of itself, a resource consisting of only a CRD is not very useful. You likely intend to trigger some code when an instance of your resource is created, updated, or deleted. We call the scripts containing the code for these trigger events "intents".
Two types of intents exist: config intents and state intents. The config intent is really meant for direct interaction: A user creates a resource with some desired parameters, and the intent code is subsequently triggered, performing some action in the cluster. A resource bundled with a config intent is called a "config resource".
The state intent will in most cases be more of a watcher: Based on some observed state of the cluster, the code e.g. performs some translations or exports some metrics. A resource bundled with a state intent is called a "state resource".
Since intents and resources are such tightly coupled concepts, intent code of either type can be scaffolded at the time of creating a resource, with the --scaffold-config or --scaffold-state options. If you forgot to scaffold the intent boilerplate at resource creation time, you can add it after the fact by using the edabuilder create intent subcommand.
After creating an intent (or creating a resource with one of the intent scaffolding options for that matter), your manifest is kept up-to-date with the appropriate scripts and trigger events.
Creating a Version#
Through edabuilder create version you can create a new API version, e.g. when you are introducing backwards compatibility breaking changes over the lifetime of your app. Here you have the option to carry over API resources from a previous version. If you choose to do so, a boilerplate migration script will be scaffolded per resource that is present in your old API version. The API version in the manifest specification is automatically updated to the new version, and conversion scripts are tracked in the manifest as well.
Generating your app#
edabuilder generate is an idempotent command designed to keep any generated code in your app up-to-date. It uses your API resources as input to generate a python SDK under api/<apiversion>/pysrc, CRDs and OpenAPI. It also ensures your manifest is up-to-date.
Many of the other subcommands use generate under the hood, so it's likely you won't have to run it explicitly very often.
Adding App Settings#
There are two edabuilder commands which allow you to provide install-time settings for your application. For instance, if your app includes a resource, (e.g. a k8s deployment) for which you want the user to be able to specify configurable parameters like CPU limit, etc.
The first command, edabuilder generate appsettings, will evaluate all CRs that are linked in your app's Manifest and look for settings annotations. A setting is annotated with # app-set: ${<setting name>}. To continue the example of a CPU limit set for a Deployment resource, the Deployment CR linked in the Manifest would look something like:
apiVersion: apps/v1
kind: Deployment
metadata:
  name: my-app-controller
spec:
  # omitted for brevity
  containers:
    - image: ghcr.io/your-container-image-uri
      resources:
        limits:
          cpu: "4" # app-set: ${controllerCpuLimit}
Where the setting JSON name is controllerCpuLimit and its default value is "4".
 After gathering all the settings you have annotated in your app CRs, edabuilder will collect these in a single Go struct that becomes the source of truth for your settings' OpenAPI specification. This struct, AppSettingsSpec, is located at <app-root>/settings/appsettings_types.go. For the CPU limit example above, this is the resulting struct:
// AppSettingsSpec defines the desired state of AppSettings.
type AppSettingsSpec struct {
    // Document the 'ControllerCpuLimit' setting here, so it shows up nicely in your generated OpenAPI spec.
    // +kubebuilder:default="4"
    // +eda:ui:title="ControllerCpuLimit"
    ControllerCpuLimit string `json:"controllerCpuLimit,omitempty"`
}
You should manually review and edit this struct to make sure the documentation comments and field types are correct before continuing. Given that a single structure defines the app settings means that the settings' names should be unique across all CRs of your app.
Iterating on settings
The generate appsettings command is strictly meant for the initial scaffolding of the application' settings structure. Rerunning this command (which requires the -f flag) is a destructive operation, in the sense that it will revert all manual changes you may have done in AppSettingsSpec.
If you have already scaffolded the settings and made updates to the AppSettingsSpec struct and would like to add a new setting, then simply add a field to the struct for your new setting and do not rerun the generate appsettings command.
Once you're done documenting your app settings, you can finalise your app settings for the EDA Store with the second command, edabuilder generate appsettings-openapi. It will generate an OpenAPI spec under <app-root>/appsettings-openapi/. Your Manifest will be updated automatically to include this file.
Your app is now ready to be installed by the EDA Store with install-time settings.
Installing with a non-default value of a setting
App settings can be provided directly in the AppInstaller workflow or at the app install time in the EDA UI.
To provide a custom value for a setting in the workflow, add the <setting-name>: <custom-value> key-value pair to the list entry of the appropriate app in the Workflow's spec.input.apps[*].appSettings map. Here is an example Workflow to install some app with a custom value for the controllerCpuLimit setting:
Install Workflow
Configuring deploy targets#
The edabuilder deploy command is customisable through a configuration file located at ~/.config/edabuilder/config.yaml2. It allows a user to provide the so called deploy targets - the custom OCI registry and application catalog pairs that a user can select from when deploying an app. The configuration file is structured as follows:
# a map of named deploy targets
deploy-targets:
  internal-target: # first deploy target
    # Is authentication required to read from both registry and catalog?
    # When set to false or omitted, indicates that the registry/catalog are public
    read-authentication: false
    registry:
      # URL of your registry, i.e. [<scheme>]<fqdn>[:port]
      url: corporateregistry.internal
      # Whether to skip TLS verification for the registry [default: false]
      skip-tls-verify: true
    catalog:
      # URL of your catalog repo, i.e. [<scheme>]<fqdn>[:port]/<repository>.git
      url: https://gitlab.mycorp.internal/dev/eda-apps.git
      skip-tls-verify: true
  external-target: # second deploy target
    registry:
       url: ghcr.io
    catalog:
      url: https://github.com/someorg/eda-apps.git
# the name of the currently active deploy target
# in-cluster is a special value that indicates the in-cluster deploy target
current-deploy-target: in-cluster
The deployment targets are defined in the config.yaml file in the edabuilder config directory using the following top level keys:
- deploy-targets- a YAML object that is a map of deploy target configurations.
- current-deploy-target- a string value that matches one of the configured deploy-target names. Defaults to in-cluster which is a reserved name for a deploy target being the EDA cluster itself, with the dev registry and dev catalog deployed in it.
in-cluster deploy target#
 The default deploy target is the in-cluster target, which is a reserved name for the OCI registry and Catalog deployed by edabuilder in the EDA cluster itself. The edabuilder deploy command will use this target by default if no other target is specified.
Current deploy target#
After you have added the desired deploy targets to the config.yaml file, you can set the current deploy target by running either by specifying its name in the configuration file, or using the CLI command:
To reset the current deploy target to the default in-cluster target, you can run:
Read authentication#
Both the catalog and registry may require authentication for cloning or fetching. Private Git repositories and private registries always require authentication. This means that, for the app store to pull the corresponding artifacts, a secret containing the necessary authentication or token data must be provided.
However, when using public registries and catalogs, users do not need to provide authentication data for read operations.
To provide this flexibility, the read-authentication boolean setting is available at the deploy-target level.
- When set to false(or when not present), both the catalog and registry are considered public, and reads can be performed without authentication.
- When set to true, authentication is required to read from the registry and catalog. In this case, during thedeploycommand, a prompt will appear asking if you wantedabuilderto configure the associated secrets using authentication data previously provided via theedabuilder login registryoredabuilder login catalogcommands (stored in~/.config/edabuilder/auth.json3).
If you choose "Yes," the matching authentication data is provisioned as a Kubernetes secret and referenced in the corresponding catalog or registry. If you select "No," no secrets are configured, and a prompt will inform you that these secrets should be created manually.
-  For more information on packaging, publishing and iteratively deploying apps, refer to Build and Publish ↩↩↩ 
-  You can provide a custom location for this file by setting the EDABUILDER_CONFIGenvironment variable. ↩
-  You can provide a custom location for this file by setting the EDABUILDER_AUTH_CONFIGenvironment variable. ↩