zconfig 1.1.3

A simple config file generator and parser


To use this package, run the following command in your project's root directory:

Manual usage
Put the following dependency into your project's dependences section:

zconfig

zconfig is a library that easily serializes and deserializes config files or directly provide them through the command line based on a given annotated struct.

The library provides annotations which are used on plain old data structs which give them information such as descriptions or other shortnames.

The actual parsing is not done by this library but is directly feeded into D's getopt function. This library just parses a config file and generates arguments for getopt out of them.

The benefit by using getopt is that you essentially synchronize the command line arguments and the config file options with just one struct.

Annotations

AnnotationDescription
@SectionDescribes a section. When a struct member is annotated with it then the member will be written underneath [section].
@DescAdds a description to a struct member which will be printed on the command line help and on the config file.
@ShortProvide an alternative (short-)name to a struct member. Allows the same setting to be changed via alternative (short-)names.
@OnlyCLIDo not (de-)serialize the struct member but only read it from the command line.
@ConfigFileMark a struct member to be the filename of the to-be-read config file. This is a special annotation and only be provided to one struct member. It also can only be provided through the command line.
@PassThroughSame as getopt passThrough.
@RequiredSame as getopt required. If the option is not provided an error is thrown.
@HandlerCall a custom handler function for the annotated struct member which is responsible for setting its value.

API

FunctionDescription
initializeConfigAutomatically generates the getopt arguments and calls said function with it.
getConfigArgumentsParses the config file and creates an args array ouf of them excluding options provided through the command line.
writeExampleConfigFileCreates an example config based on an annotated struct.

Additional information is written inside the source. Run dub build docs to view them in html format.

Usage

Basic example

struct MyConfig
{
    @ConfigFile @Short("c") @Desc("Read this config file instead of the default.")
    string configFile;
    @Desc("My number.")
    int number;
    @Desc("My bool.")
    bool toggle;

    @Section("service")
    {
        @Required @Short("host") @Desc("Hostname of the service.")
        string hostname = "localhost";
        @Required @Short("p") @Desc("Port of the service.")
        int port = 1234;
    }
}

enum usage = "My program version 1.0 does things.";

int main(string[] args)
{
    string[] configArgs = getConfigArguments!MyConfig("myconf.conf", args);

    if (configArgs.length > 0)
    {
        import std.array : insertInPlace;

        // Prepend them into the command line args
        args.insertInPlace(1, configArgs);
    }

    MyConfig conf;
    bool helpWanted = false;

    import std.getopt : GetOptException;
    try
    {
        conf = initializeConfig!(MyConfig, usage)(args, helpWanted);
    }
    catch (GetOptException e)
    {
        import std.stdio : stderr;
        stderr.writefln("Invalid argument: %s", e.msg);
        return 1;
    }

    if (helpWanted)
    {
        return 0;
    }
}

Generate example config file

int main(string[] args)
{
    writeExampleConfigFile!MyConfig("myconf.conf");
    return 0;
}

myconf.conf will have the following content:

; My number.
; Default value: 0
;number=0

; My bool.
; Default value: false
;toggle=false

[service]
; Hostname of the service.
; Default value: localhost
;hostname=localhost

; Port of the service.
; Default value: 1234
;port=1234

Handler/Callback example

It is possible to define a custom handler to set a config struct member. The following restrictions apply on the handler function:

  • Have two parameters.
  • The first parameter is the string value as given through args and must therefore have type string.
  • The second parameter must be declared ref or out and must have the same type as the config member. This is the config struct's variable the handler is supposed to set.

Here is an example that parses a string (provided through a config file or the CLI arguments) and extracts the x and y coordinate to set for the config struct.

struct Point
{
    int x;
    int y;
}

void pointHandler(string value, out Point confValue)
{
    import std.string : split;
    import std.conv : to;
    auto segments = value.split("x");

    confValue.x = segments[0].to!int;
    confValue.y = segments[1].to!int;
}

struct MyConfig
{
    @Desc("Defines a point") @Handler!pointHandler
    Point coordinate;
}

Calling the program with e.g. ./myapp --coordinate=20x62 will call the pointHandler with value="20x62". It will then get split and the respective fields are set with the resulting config file having MyConfig.coordinate.x = 20 and MyConfig.coordinate.y = 62.

Authors:
  • zhad3
Dependencies:
none
Versions:
1.1.3 2022-Aug-26
1.1.2 2022-Jun-15
1.1.1 2022-Jun-14
1.1.0 2022-Jun-13
1.0.1 2022-Jun-06
Show all 7 versions
Download Stats:
  • 1 downloads today

  • 3 downloads this week

  • 24 downloads this month

  • 350 downloads total

Score:
0.8
Short URL:
zconfig.dub.pm