Documentation
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:

- 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 RestEndpointis deleted | 
| updateMethod | Put | Updating the objects with which HTTP primitive. Available values: Put,PatchandPost. | 
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 RestEndpointis deleted | 
| updateMethod | Put | Updating the objects with which HTTP primitive. Available values: Put,PatchandPost. | 
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

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: Bearerheader | 
| add_header_basic | username(string), password(string) | Add an Authorization: Basicheader | 
| 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.