Lightweight Configuration for Services and Applications

The :std/config library module provides basic procedures for managing configuration for services and applications using plists.

Usage

(import :std/config)

Overview

Undoubtedly there is a need for configuration certain aspects of an application or service. There are numerous ad hoc configuration languages, ranging from atrocities like toml to json and so on.

In Gerbil we take a much simpler approach that takes advantage of LISP homoiconicity. Specifically, we don't need to invent yet another configuration language; we can just use the reader/writer and allow any primitive object to be part of the configuration.

As such, we settle for the use of plists (property lists), where keywords are the keys for configuration aspects and any primitive object can be the value, which also naturally leads to arbitrary nesting and so on.

The only restriction we place is that the configuration must start with a config: key followed by a value indicating the schema of the configuration, with application specific semantics.

In terms of representation on disk, the plist is stored flat (without the enclosing list delimiter) which makes for an easier editing experience.

Note

The :std/config module provides functionality for reading, editing, and writing configurations. At the moment we only provide primitive functionality, but moving forward we will also add schema support (using the type system) and macro generation of accessors and mutators. This is all planned for v0.19.

Procedures

config-get

(config-get cfg key (default #f))

Retrieves the value associated with key in the configuration cfg; if the key is not present, the the default value is returned.

config-get!

(config-get! cfg key)

Like config-get but raises an error if the value is not present or false.

config-set!

(config-set! cfg key val)

Sets the config value val for key, mutating the congfiguration if needed. Returns the new configuration.

config-push!

(defrule (config-push! cfg key val)
  (set! cfg (config-set! cfg key val)))

Sets the config value val for key and updates the variable cfg for with the resulting configuration.

config-check!

(config-check! cfg type)

checks whether the configuration object cfg is a configuration with schema type.

write-config

(write-config cfg (output (current-output-port)) pretty: (pretty? #f))

Flat writes the configuration object cfg to output; pretty prints if pretty? is true.

save-config!

(save-config! cfg path)

Saves the configuration object cfg on disk to path.

read-config

(read-config (input (current-input-port)))

Reads a configuration object from input

load-config

(load-config path type)

Loads a configuration from disk path path and verifies the configuration schema. Returns the loaded configuration.

string->object

(string->object str)

Converts a string to a primitive object; complentary to the object->string builtin.

string->integer

(string->integer str)

Converts a string to an integer, raising an error if the resulting object is a not an int4eger.

Example

Here is an example from ensemble server configuration in the gerbil ensemble tool.

The schema of server configuration is like this:

config: ensemble-server-v0

domain: <symbol>
identifier: <server-identifier>
supervisor: <server-identifier>
registry:   <server-identifier>
cookie:     <path>
admin:      <path>

;;; execution
role:    <symbol>
secondary-roles: (<symbol> ...)
exe:     <string>
args:    (<string> ...)
policy:  <symbol>
env:     <string>
envvars: (<string> ...)

;;; logging
log-level: <symbol>
log-file:  <path>
log-dir:   <path>

;;; bindings
addresses: (<address> ...)
auth: ((<server-identifier> <capability>) ...)
known-servers: ((<server-identifier> <address> ...) ...)

;;; application specific configuration
application: ((<symbol> config ...) ...)

And here is some code that updates the server configuration according to command line arguments:

(defrule (config-server! opt cfg)
  (let-hash opt
    (cond (.?secondary-roles => (cut config-push! cfg secondary-roles: <>)))
    (cond (.?env             => (cut config-push! cfg env: <>)))
    (cond (.?envvars         => (cut config-push! cfg envvars: <>)))
    (cond (.?log-level       => (cut config-push! cfg log-level: <>)))
    (cond (.?addresses       => (cut config-push! cfg addresses: <>)))
    (cond (.?auth-servers    => (cut config-push! cfg auth: <>)))
    (cond (.?known-servers   => (cut config-push! cfg known-servers: <>)))
    (when .?application
      (let (default-app-config-path
             (path-expand (symbol->string .application)
                          (path-expand "config" (gerbil-path))))
        (unless (or .?config (file-exists? default-app-config-path))
          (error "no application configuration"))
        (let* ((app-config-path (or .?config default-app-config-path))
               (app-config (call-with-input-file app-config-path read-config))
               (app-alist  (config-get cfg application: [])))
          (cond
           ((assq .application app-alist)
            => (lambda (p) (set-cdr! p app-config)))
           (else
            (set! app-alist [[.application :: app-config] :: app-alist])))
          (config-push! cfg application: app-alist))))))