Defining a RestEndpoint

The following section will document the spec of a RestEndpoint.

For more completes examples please see the one in the repo

Availability of the values for both handlebars templates and rhai scripts

The following values are populated in the order of the documented flow: workflow diagram

  • input
  • init
  • pre
  • read
  • write
  • post

Which mean write is empty when computing the reads instructions. But read is fully populated when running the writes.

All these keys will have sub-keys based on the names of their definition.

input from Secret and ConfigMap

The following definition:

spec:
  inputs:
  - name: mySecret
    secretRef:
      name: secret-name

Will populate the value input.mySecret with the whole definition of the secret secret-name. Which mean input.mySecret.metadata.name is a string with the value secret-name if the said secret do exist.

Please note that the object is provided as-is. So the data from a secret will be base64 encoded. The base64_decode helper for handlebars and the same rhai method are there to accomodate you.

If your definition can accomodate the absence of the Secret (or ConfigMap), use the optional: true key as bellow :

spec:
  inputs:
  - name: myOptConfig
    configMapRef:
      optional: true
      name: secret-name

Any referenced Secret or ConfigMap missing without that optional: true key will stop the process. A kubernetes event will be produced, and a conditions in the status will explain the missing piece(s).

notes: kuberest do not watch changes on theses Secret/ConfigMap. So change will be reflected on the next verification loop. Hence, using kuberest as a secret-copier is a bad abuse: the dedicated operators doing this use watch, so the changes are reflected immediately.

input from rendered handlebars templates

This is an helper to help you save a value to be reused multiple time later in the RestEndpoint definition and be sure the value will be the same.

spec:
  inputs:
  - name: uuid
    handleBarsRender: "{{ uuid_new_v7 }}"

will put a generated uuid in input.uuid.

Going further, JSON object are also supported:

spec:
  inputs:
  - name: config
    handleBarsRender: |-
      {{#to_json format="yaml"}}
      merge_method: ff
      issues_enabled: false
      {{/to_json}}

Set input.config.merge_method to "ff".

input from a Password Generator

:warn: While the kuberest do provide the feature, it is probably advised to use a dedicated operator like kubernetes-secret-generator if you need a lot of them and maintain them. Use with caution.

This feature is mainly provided to set strong default password for technical users within the manipulated endpoint. Do not overuse this ;)

spec:
  inputs:
  - name: password
    passwordGenerator:
      length: 32
      weightAlphas:  60
      weightNumbers: 20
      weightSymbols: 20

Will set a generated password with default configuration for input.password. The following would have defined the same thing.

spec:
  inputs:
  - name: password
    passwordGenerator: {}

The default configuration create a 32 charaters long password with 60% of letters (capitalized or otherwise) 20% of numbers and 20% of symbols.

Configuring the REST client

spec:
  client:
    baseurl: "https://gitlab.com/api/v4"

Is a strict minimum for any RestEndpoint definition and set the rest client base url to use Gitlab REST endpoints.

Authentication

Since no REST endpoint come without any form of authentication, a more complete definition would be:

spec:
  inputs:
  - name: token
    secretRef:
      name: my-personal-gitlab-token
  client:
    baseurl: "https://gitlab.com/api/v4"
    headers:
      Authorization: "Bearer {{ base64_decode input.token.data.gitlab_token }}"

Basic authentication is also possible:

spec:
  inputs:
  - name: admin
    secretRef:
      name: gitea-admin-user
  client:
    baseurl: "https://gitea.your-company.com/api/v1"
    headers:
      Authorization: "{{ header_basic (base64_decode input.admin.data.username) (base64_decode input.admin.data.password) }}"

Others headers

You can define any other needed headers as you please. The following headers are already set for you, but you can overwrite them (with all the implied consequenses: kuberest will still expect json data and will still send the http body as json encoded)

spec:
  client:
    headers:
      Content-Type: "application/json; charset=utf-8"
      Accept: "application/json"

Further client configuration

Here are the other availables options for the client and their default value:

spec:
  client:
    keyName: id
    teardown: true
    updateMethod: Put
key default description
keyName id the key in the returned object that identify the object
teardown true Should the created objects (writes section) be deleted when then RestEndpoint is deleted
updateMethod Put Updating the objects with which HTTP primitive. Available values: Put, Patch and Post.

All theses 3 keys can be overwritten per “write group” if needed.

Defines read from HTTP Get

All reads will use the client definition

spec:
  client:
    baseurl: "https://gitlab.com/api/v4"
  reads:
  - name: projects
    path: "users/123456"
    items:
    - name: list
      key: "projects"

Would define read.projects.list from the result of the following command: curl https://gitlab.com/api/v4/users/123456/projects. Which mean, the URL used is defined by <client.baseurl>/<read_group.name>/<item.key>.

You can defined as many read groups and item as needed.

Hint: item key can be empty, so you can make this a little less awkward:

spec:
  reads:
  - name: seb
    path: "users/123456/projects"
    items: [{"name": "list", "key": ""}]
  - name: paul
    path: "users/654321/projects"
    items: [{"name": "list", "key": ""}]

Which would set read.seb.list and read.paul.list. Alternatively, you can:

spec:
  reads:
  - name: projects
    path: "users"
    items:
    - name: seb
      key: 123456/projects
    - name: paul
      key: 654321/projects

Which would set read.projects.seb and read.projects.paul with the same value as in the previous example.

Create REST objects and get their definitions set in write

The URL construction for writes work exactly the same as for reads. The actual key for each created object will be stored and kept to reuse on updates.

spec:
  client:
    baseurl: "https://gitea.your-company.com/api/v1"
  writes:
  - name: application
    path: user/applications/oauth2
    items:
    - name: woodpecker
      values: |-
        confidential_client: true
        name: woodpecker
        redirect_uris:
        - "https://woodpecker.your-company.com/authorize"

So the initial call here would make a POST to “https://gitea.your-company.com/api/v1/user/applications/oauth2” while the later scheduled updates will do a PUT to “https://gitea.your-company.com/api/v1/user/applications/oauth2/<id>” where <id> is the obtained “id” on the initial POST response.

values is an HandleBars template that return a YAML formated object.

Further writes configuration

You can overwrite the behaviour of the controller for a single REST endpoint using:

spec:
  writes:
  - name: application
    keyName: id
    teardown: true
    updateMethod: Put
key default description
keyName id the key in the returned object that identify the object
teardown true Should the created objects (writes section) be deleted when then RestEndpoint is deleted
updateMethod Put Updating the objects with which HTTP primitive. Available values: Put, Patch and Post.

Saving the results with outputs

Saving the results is straightforward:

spec:
  outputs:
  - kind: ConfigMap
    metadata:
      name: set-all-projects-merge-method
    data:
      result.yaml: |-
        ---
        {{ json_to_str post.results format="yaml" }}

Availables kind: ConfigMap and Secret.

Availables metedata fields: name, labels, annotations and namespace (if multi-tenancy is disabled).

Data values are also handlebars templates.

note: Secret data is passed to the cluster as stringData, so the cluster will base64 encode the secret values for us. No need to mess with base64_encode here.

spec:
  outputs:
  - kind: Secret
    metadata:
      name: woodpecker-gitea-openid
    data:
      WOODPECKER_GITEA_CLIENT: "{{ write.application.woodpecker.client_id }}"
      WOODPECKER_GITEA_SECRET: "{{ write.application.woodpecker.client_secret }}"

Available helpers for handlebars templates

From dependencies

kuberest use the handlebars_misc_helpers crate which provide a lot of others helpers see the documentation on that page for the complete list.

From this package

Helper Arguments Description Example
base64_decode base64_encoded_string Decode a base64 encode string (usefull for Secret values) base64_decode input.secret.data.some_key
base64_encode string_to_encode Encode a string/buffer with the base64 encoding base64_encode "username:password"
gen_password length Generate a password if specified length gen_password 32
gen_password_alphanum length Generate a password if specified length without any specials characters gen_password_alphanum 8
header_basic username, password Generate a “Basic encoded_auth” value to set with a “Authorization” header header_basic (base64_decode input.secret.data.username) (base64_decode input.secret.data.password)
argon_hash password Hash a password using the Argon2 algorithm argon_hash (base64_decode input.secret.data.password)
bcrypt_hash password Hash a password using the BCrypt algorithm bcrypt_hash (base64_decode input.secret.data.password)

rhai scripts

Rhai is a memory-safe scripting engine. It is a very pleasant language to play with. Have a look at their examples to learn the language capabilities.

The only reason why this ability exist is: many of the apps we love have configuration endpoints that do not comply with REST (Grafana, SonarQube to name a few). Provoding this ability come with risk of abuses. Dont, for your own cluster safety ;)

All rhai scripts (init/pre/post/teardown) are expected to return a Map, so the most minimal script is:

#{}

This is an empty Map in the rhai language. Failing to return a Map will stop the process. Errors are documented in the conditions of the RestEndpoint.

pre script usage

Templating/Preparing data for later stage(s):

fn genValues(name) {
  #{
    name: name,
    other: "properties",
    to: #{
      configure: "your",
      way: name+"-test"
    }
  }
}
#{
  pierre: genValues("pierre"),
  paul: genValues("paul"),
  jacques: genValues("jacques"),
}

post script usage

Some api endpoints are not REST friendly, use your scripting skillz for these.

warning You’re on your own. Using this mean kuberest wont handle the teardown process for you. Before using this, write your teardown script. Or you’ll leave garbage behind you.

teardown script usage

workflow diagram

It’s meant to undo your changes in the post script.

If your writes use the pre variables in the templates, then the teardown needs to provides the sames values. YAML allow to duplicate strings, use it:

spec:
  pre: &genValues |-
    #{
      some: "values"
    }
  teardown: *genValues

Available methods for rhai scripts

Method Arguments Return Description Example
gen_password length (number) string Generate a password of requested length let passwd = gen_password(32);
gen_password_alphanum length (number) string Generate a password of requested length without any special characters let weak_passwd = gen_password_alphanum(8);
base64_decode base64_encoded_data (string) string Decode a base64 encode string (usefull for Secret values) letdecoded = base64_decode(input.secret.data.some_key);
base64_encode string_to_encode string Encode a string/buffer with the base64 encoding let encoded = base64_encode("username:password");
json_encode data (any) string Convert any data to their JSON representation string let encoded = json_encode(#{test: "value"});
json_decode encoded_data (string) any convert any json formated data to their rhai object/array/… conterpart let data = json_decode("{\\"name\\":\\"paul\\"}");
yaml_encode data (any) string Convert any data to their YAML representation string let encoded = yaml_encode(#{test: "value"});
yaml_decode encoded_data (string) any convert any YAML formated data to their rhai object/array/… conterpart let data = yaml_decode("name: paul");
yaml_decode_multi encoded_data (string) array convert a multi-document yaml string into an array of the corresponding object/array/… let data = yaml_decode_multi("name: paul");
bcrypt_hash password string hash a clear-text password using the bcrypt algorithm let hash = bcrypt_hash("my_secret_password");
new_argon().hash password string hash a clear-text password using the Argon2 algorithm let hasher = new_argon();let hash = hasher.hash("my_secret_password");

The hbs object

From rhai you have a complete access to the HandleBars templating environement using the hbs object :

Method Arguments Return Description Example
render_from template(string), values (object) string Generate a password of requested length let passwd = hbs.render_from("", values);

The client object

From rhai you have a complete access to the REST client used in kuberest using the client object:

Method Arguments Return Description Example
head path(string) object do an HTTP HEAD on path let res = client.head("projects");
get path(string) object do an HTTP GET on path let res = client.get("projects");
delete path(string) object do an HTTP DELETE on path let res = client.delete("projects/1345/groups/43");
patch path(string), values (object) object do an HTTP PATCH on path let res = client.patch("projects/1345/groups/43", #{name: "test"});
post path(string), values (object) object do an HTTP POST on path let res = client.post("projects/1345/groups/43", #{name: "test"});
put path(string), values (object) object do an HTTP PUT on path let res = client.put("projects/1345/groups/43", #{name: "test"});

In the context of duplicating data to an other service, you can create an other client:

let target = new_client("https://gitlab.com/api/v4");
target.add_header("add_header", "Bearer "+base64_decode(input.gitlab.data.token));
target.add_header_json();
let prjs = target.get("projects");
...
Method Arguments Description
new_client baseurl(string) Create a new http client
set_baseurl baseurl(string) Change the basepath of the client
set_server_ca PEM_certificate (string) Configure the serverCA certificate (for self-signed target)
set_mtls_cert_key PEM_certificate (string),PEM_key (string) Configure Client certificate and key for mTLS authentification
add_header key (string), value (string) Set a header for the client
add_header_bearer token(string) Add an Authorization: Bearer header
add_header_basic username(string), password(string) Add an Authorization: Basic header
add_header_json null Set the Accept and Content-Type headers to json
headers_reset null Remove all stored headers on the client

warning do not forget to write the teardown script.