#configuration-language #serde #serialization #bindings #conl #human-centric

serde_conl

CONL is a human-centric configuration language. This crate has the serde bindings.

3 releases (breaking)

0.3.0 Nov 4, 2024
0.2.0 Oct 31, 2024
0.1.0 Oct 21, 2024

#548 in Encoding

Download history 127/week @ 2024-10-16 32/week @ 2024-10-23 230/week @ 2024-10-30 27/week @ 2024-11-06 1/week @ 2024-11-13 2/week @ 2024-11-20

122 downloads per month

MIT license

51KB
1.5K SLoC

CONL is a post-minimalist, human-centric configuration language. It is a replacement for JSON/YAML/TOML, etc... that supports a JSON-like data model of values, maps and lists; but is designed to be much easier to work with.

For more information, see the README.

serde_conl provides integration with Serde to serialize and deserialize CONL documents.

Getting started

Add this crate to your project

cargo add serde_conl

Define a type to represent your configuration. Use #[serde(default)] to ensure that any missing fields are set to the default values, and #[serde(deny_unknown_fields)] to provide helpful error messages if someone misspells a field name.

#[derive(Debug, Default, serde::Deserialize, serde::Serialize)]
#[serde(default, deny_unknown_fields)]
struct Config {
    hostname: String,
    port: u32,
    master: bool,
    ssh_args: Vec<String>,
}

Parse the file:

let file = std::fs::File::read("config.conl").unwrap();

let config: Config = serde_conl::from_slice(&file).unwrap();

Notes

Any value that serde supports can be used as a value in a CONL document. Keys can be any scalar value (strings, numbers, bools, etc...) but cannot be lists or maps.

That said, CONL is for configuration... so try to use values that make sense to type in a text file, and stay away from fancy programmer-centric type shenanigans. If you need them, serde_humanize_rs provides a bunch of types like durations and file sizes.

Options are typically represented by omitting the field from the map, but as CONL has no nullability, the empty string is used in map keys or list items. A unit () or struct X; is represented by the empty string.

Enums are represented as they are in JSON (but without the syntax noise). For example:

enum Example {
    A,
    B(u32),
    C { a: u32, b: u32 },
    D(u32, u32)
}
example_a = a
example_b
  b = 1
example_c
  c
    a = 1
    b = 2
example_d
  = 1
  = 2

Dependencies

~350–600KB
~12K SLoC