mirror of https://github.com/adamdruppe/arsd.git
1588 lines
34 KiB
D
1588 lines
34 KiB
D
/+
|
||
== arsd.ini ==
|
||
Copyright Elias Batek (0xEAB) 2025.
|
||
Distributed under the Boost Software License, Version 1.0.
|
||
+/
|
||
/++
|
||
INI configuration file support
|
||
|
||
This module provides a configurable INI parser with support for multiple
|
||
“dialects” of the format.
|
||
|
||
---
|
||
import arsd.ini;
|
||
|
||
IniDocument!string parseIniFile(string filePath) {
|
||
import std.file : readText;
|
||
return parseIniDocument(readText(filePath));
|
||
}
|
||
---
|
||
+/
|
||
module arsd.ini;
|
||
|
||
///
|
||
@safe unittest {
|
||
// INI example data (e.g. from an `autorun.inf` file)
|
||
static immutable string rawIniData =
|
||
"[autorun]\n"
|
||
~ "open=setup.exe\n"
|
||
~ "icon=setup.exe,0\n";
|
||
|
||
// Parse the document into an associative array:
|
||
string[string][string] data = parseIniAA(rawIniData);
|
||
|
||
string open = data["autorun"]["open"];
|
||
string icon = data["autorun"]["icon"];
|
||
|
||
assert(open == "setup.exe");
|
||
assert(icon == "setup.exe,0");
|
||
}
|
||
|
||
/++
|
||
Determines whether a type `T` is a string type compatible with this library.
|
||
+/
|
||
enum isCompatibleString(T) = (is(T == string) || is(T == const(char)[]) || is(T == char[]));
|
||
|
||
//dfmt off
|
||
/++
|
||
Feature set to be understood by the parser.
|
||
|
||
---
|
||
enum myDialect = (IniDialect.defaults | IniDialect.inlineComments);
|
||
---
|
||
+/
|
||
enum IniDialect : ulong {
|
||
/++
|
||
Minimum feature set.
|
||
|
||
No comments, no extras, no nothing.
|
||
Only sections, keys and values.
|
||
Everything fits into these categories from a certain point of view.
|
||
+/
|
||
lite = 0,
|
||
|
||
/++
|
||
Parse line comments (starting with `;`).
|
||
|
||
```ini
|
||
; This is a line comment.
|
||
;This one too.
|
||
|
||
key = value ;But this isn't one.
|
||
```
|
||
+/
|
||
lineComments = 0b_0000_0000_0000_0001,
|
||
|
||
/++
|
||
Parse inline comments (starting with `;`).
|
||
|
||
```ini
|
||
key1 = value2 ; Inline comment.
|
||
key2 = value2 ;Inline comment.
|
||
key3 = value3; Inline comment.
|
||
;Not a true inline comment (but technically equivalent).
|
||
```
|
||
+/
|
||
inlineComments = 0b_0000_0000_0000_0011,
|
||
|
||
/++
|
||
Parse line comments starting with `#`.
|
||
|
||
```ini
|
||
# This is a comment.
|
||
#Too.
|
||
key = value # Not a line comment.
|
||
```
|
||
+/
|
||
hashLineComments = 0b_0000_0000_0000_0100,
|
||
|
||
/++
|
||
Parse inline comments starting with `#`.
|
||
|
||
```ini
|
||
key1 = value2 # Inline comment.
|
||
key2 = value2 #Inline comment.
|
||
key3 = value3# Inline comment.
|
||
#Not a true inline comment (but technically equivalent).
|
||
```
|
||
+/
|
||
hashInlineComments = 0b_0000_0000_0000_1100,
|
||
|
||
/++
|
||
Parse quoted strings.
|
||
|
||
```ini
|
||
key1 = non-quoted value
|
||
key2 = "quoted value"
|
||
|
||
"quoted key" = value
|
||
non-quoted key = value
|
||
|
||
"another key" = "another value"
|
||
|
||
multi line = "line 1
|
||
line 2"
|
||
```
|
||
+/
|
||
quotedStrings = 0b_0000_0000_0001_0000,
|
||
|
||
/++
|
||
Parse quoted strings using single-quotes.
|
||
|
||
```ini
|
||
key1 = non-quoted value
|
||
key2 = 'quoted value'
|
||
|
||
'quoted key' = value
|
||
non-quoted key = value
|
||
|
||
'another key' = 'another value'
|
||
|
||
multi line = 'line 1
|
||
line 2'
|
||
```
|
||
+/
|
||
singleQuoteQuotedStrings = 0b_0000_0000_0010_0000,
|
||
|
||
/++
|
||
Parse key/value pairs separated with a colon (`:`).
|
||
|
||
```ini
|
||
key: value
|
||
key= value
|
||
```
|
||
+/
|
||
colonKeys = 0b_0000_0000_0100_0000,
|
||
|
||
/++
|
||
Concats substrings and emits them as a single token.
|
||
|
||
$(LIST
|
||
* For a mutable `char[]` input,
|
||
this will rewrite the data in the input array.
|
||
* For a non-mutable `immutable(char)[]` (=`string`) or `const(char)[]` input,
|
||
this will allocate a new array with the GC.
|
||
)
|
||
|
||
```ini
|
||
key = "Value1" "Value2"
|
||
; → Value1Value2
|
||
```
|
||
+/
|
||
concatSubstrings = 0b_0000_0001_0000_0000,
|
||
|
||
/++
|
||
Evaluates escape sequences in the input string.
|
||
|
||
$(LIST
|
||
* For a mutable `char[]` input,
|
||
this will rewrite the data in the input array.
|
||
* For a non-mutable `immutable(char)[]` (=`string`) or `const(char)[]` input,
|
||
this will allocate a new array with the GC.
|
||
)
|
||
|
||
$(SMALL_TABLE
|
||
Special escape sequences
|
||
`\\` | Backslash
|
||
`\0` | Null character
|
||
`\n` | Line feed
|
||
`\r` | Carriage return
|
||
`\t` | Tabulator
|
||
)
|
||
|
||
```ini
|
||
key1 = Line 1\nLine 2
|
||
; → Line 1
|
||
; Line 2
|
||
|
||
key2 = One \\ and one \;
|
||
; → One \ and one ;
|
||
```
|
||
+/
|
||
escapeSequences = 0b_0000_0010_0000_0000,
|
||
|
||
/++
|
||
Folds lines on escaped linebreaks.
|
||
|
||
$(LIST
|
||
* For a mutable `char[]` input,
|
||
this will rewrite the data in the input array.
|
||
* For a non-mutable `immutable(char)[]` (=`string`) or `const(char)[]` input,
|
||
this will allocate a new array with the GC.
|
||
)
|
||
|
||
```ini
|
||
key1 = word1\
|
||
word2
|
||
; → word1word2
|
||
|
||
key2 = foo \
|
||
bar
|
||
; → foo bar
|
||
```
|
||
+/
|
||
lineFolding = 0b_0000_0100_0000_0000,
|
||
|
||
/++
|
||
Imitates the behavior of the INI parser implementation found in PHP.
|
||
|
||
$(WARNING
|
||
This preset may be adjusted without further notice in the future
|
||
in cases where it increases alignment with PHP’s implementation.
|
||
)
|
||
+/
|
||
presetPhp = (
|
||
lineComments
|
||
| inlineComments
|
||
| hashLineComments
|
||
| hashInlineComments
|
||
| quotedStrings
|
||
| singleQuoteQuotedStrings
|
||
| concatSubstrings
|
||
),
|
||
|
||
///
|
||
presetDefaults = (
|
||
lineComments
|
||
| quotedStrings
|
||
| singleQuoteQuotedStrings
|
||
),
|
||
|
||
///
|
||
defaults = presetDefaults,
|
||
}
|
||
//dfmt on
|
||
|
||
private bool hasFeature(ulong dialect, ulong feature) @safe pure nothrow @nogc {
|
||
return ((dialect & feature) > 0);
|
||
}
|
||
|
||
/++
|
||
Type of a token (as output by the parser)
|
||
+/
|
||
public enum IniTokenType {
|
||
/// indicates an error
|
||
invalid = 0,
|
||
|
||
/// insignificant whitespace
|
||
whitespace,
|
||
/// section header opening bracket
|
||
bracketOpen,
|
||
/// section header closing bracket
|
||
bracketClose,
|
||
/// key/value separator, e.g. '='
|
||
keyValueSeparator,
|
||
/// line break, i.e. LF, CRLF or CR
|
||
lineBreak,
|
||
|
||
/// text comment
|
||
comment,
|
||
|
||
/// item key data
|
||
key,
|
||
/// item value data
|
||
value,
|
||
/// section name data
|
||
sectionHeader,
|
||
}
|
||
|
||
/++
|
||
Token of INI data (as output by the parser)
|
||
+/
|
||
struct IniToken(string) if (isCompatibleString!string) {
|
||
///
|
||
IniTokenType type;
|
||
|
||
/++
|
||
Content
|
||
+/
|
||
string data;
|
||
}
|
||
|
||
private alias TokenType = IniTokenType;
|
||
private alias Dialect = IniDialect;
|
||
|
||
private enum LocationState {
|
||
newLine,
|
||
key,
|
||
value,
|
||
sectionHeader,
|
||
}
|
||
|
||
/++
|
||
Low-level INI parser
|
||
|
||
See_also:
|
||
$(LIST
|
||
* [IniFilteredParser]
|
||
* [parseIniDocument]
|
||
* [parseIniAA]
|
||
)
|
||
+/
|
||
struct IniParser(
|
||
IniDialect dialect = IniDialect.defaults,
|
||
string = immutable(char)[],
|
||
) if (isCompatibleString!string) {
|
||
|
||
public {
|
||
///
|
||
alias Token = IniToken!string;
|
||
}
|
||
|
||
private {
|
||
string _source;
|
||
Token _front;
|
||
bool _empty = true;
|
||
|
||
LocationState _locationState = LocationState.newLine;
|
||
}
|
||
|
||
@safe pure nothrow @nogc:
|
||
|
||
///
|
||
public this(string rawIni) {
|
||
_source = rawIni;
|
||
_empty = false;
|
||
|
||
this.popFront();
|
||
}
|
||
|
||
// Range API
|
||
public {
|
||
|
||
///
|
||
bool empty() const {
|
||
return _empty;
|
||
}
|
||
|
||
///
|
||
inout(Token) front() inout {
|
||
return _front;
|
||
}
|
||
|
||
///
|
||
void popFront() {
|
||
if (_source.length == 0) {
|
||
_empty = true;
|
||
return;
|
||
}
|
||
|
||
_front = this.fetchFront();
|
||
}
|
||
|
||
///
|
||
inout(typeof(this)) save() inout {
|
||
return this;
|
||
}
|
||
}
|
||
|
||
// extras
|
||
public {
|
||
|
||
/++
|
||
Skips tokens that are irrelevant for further processing
|
||
|
||
Returns:
|
||
true = if there are no further tokens,
|
||
i.e. whether the range is empty now
|
||
+/
|
||
bool skipIrrelevant(bool skipComments = true) {
|
||
static bool isIrrelevant(const TokenType type, const bool skipComments) {
|
||
pragma(inline, true);
|
||
|
||
final switch (type) with (TokenType) {
|
||
case invalid:
|
||
return false;
|
||
|
||
case whitespace:
|
||
case bracketOpen:
|
||
case bracketClose:
|
||
case keyValueSeparator:
|
||
case lineBreak:
|
||
return true;
|
||
|
||
case comment:
|
||
return skipComments;
|
||
|
||
case sectionHeader:
|
||
case key:
|
||
case value:
|
||
return false;
|
||
}
|
||
}
|
||
|
||
while (!this.empty) {
|
||
const irrelevant = isIrrelevant(_front.type, skipComments);
|
||
|
||
if (!irrelevant) {
|
||
return false;
|
||
}
|
||
|
||
this.popFront();
|
||
}
|
||
|
||
return true;
|
||
}
|
||
}
|
||
|
||
private {
|
||
|
||
bool isOnFinalChar() const @nogc {
|
||
pragma(inline, true);
|
||
return (_source.length == 1);
|
||
}
|
||
|
||
bool isAtStartOfLineOrEquivalent() @nogc {
|
||
return (_locationState == LocationState.newLine);
|
||
}
|
||
|
||
Token makeToken(TokenType type, size_t length) @nogc {
|
||
auto token = Token(type, _source[0 .. length]);
|
||
_source = _source[length .. $];
|
||
return token;
|
||
}
|
||
|
||
Token makeToken(TokenType type, size_t length, size_t skip) @nogc {
|
||
_source = _source[skip .. $];
|
||
return this.makeToken(type, length);
|
||
}
|
||
|
||
Token lexWhitespace() @nogc {
|
||
foreach (immutable idxM1, const c; _source[1 .. $]) {
|
||
switch (c) {
|
||
case '\x09':
|
||
case '\x0B':
|
||
case '\x0C':
|
||
case ' ':
|
||
break;
|
||
|
||
default:
|
||
return this.makeToken(TokenType.whitespace, (idxM1 + 1));
|
||
}
|
||
}
|
||
|
||
// all whitespace
|
||
return this.makeToken(TokenType.whitespace, _source.length);
|
||
}
|
||
|
||
Token lexComment() @nogc {
|
||
foreach (immutable idxM1, const c; _source[1 .. $]) {
|
||
switch (c) {
|
||
default:
|
||
break;
|
||
|
||
case '\x0A':
|
||
case '\x0D':
|
||
return this.makeToken(TokenType.comment, idxM1, 1);
|
||
}
|
||
}
|
||
|
||
return this.makeToken(TokenType.comment, (-1 + _source.length), 1);
|
||
}
|
||
|
||
Token lexTextImpl(TokenType tokenType)() {
|
||
|
||
enum Result {
|
||
end,
|
||
regular,
|
||
whitespace,
|
||
}
|
||
|
||
enum QuotedString : ubyte {
|
||
none = 0,
|
||
regular,
|
||
single,
|
||
}
|
||
|
||
// dfmt off
|
||
enum hasAnyQuotedString = (
|
||
dialect.hasFeature(Dialect.quotedStrings) ||
|
||
dialect.hasFeature(Dialect.singleQuoteQuotedStrings)
|
||
);
|
||
// dfmt on
|
||
|
||
static if (hasAnyQuotedString) {
|
||
auto inQuotedString = QuotedString.none;
|
||
}
|
||
static if (dialect.hasFeature(Dialect.quotedStrings)) {
|
||
if (_source[0] == '"') {
|
||
inQuotedString = QuotedString.regular;
|
||
|
||
// chomp quote initiator
|
||
_source = _source[1 .. $];
|
||
}
|
||
}
|
||
static if (dialect.hasFeature(Dialect.singleQuoteQuotedStrings)) {
|
||
if (_source[0] == '\'') {
|
||
inQuotedString = QuotedString.single;
|
||
|
||
// chomp quote initiator
|
||
_source = _source[1 .. $];
|
||
}
|
||
}
|
||
static if (!hasAnyQuotedString) {
|
||
enum inQuotedString = QuotedString.none;
|
||
}
|
||
|
||
Result nextChar(const char c) @safe pure nothrow @nogc {
|
||
pragma(inline, true);
|
||
|
||
switch (c) {
|
||
default:
|
||
return Result.regular;
|
||
|
||
case '\x09':
|
||
case '\x0B':
|
||
case '\x0C':
|
||
case ' ':
|
||
return (inQuotedString != QuotedString.none) ? Result.regular : Result.whitespace;
|
||
|
||
case '\x0A':
|
||
case '\x0D':
|
||
return (inQuotedString != QuotedString.none)
|
||
? Result.regular : Result.end;
|
||
|
||
case '"':
|
||
static if (dialect.hasFeature(Dialect.quotedStrings)) {
|
||
return (inQuotedString == QuotedString.regular)
|
||
? Result.end : Result.regular;
|
||
} else {
|
||
return Result.regular;
|
||
}
|
||
|
||
case '\'':
|
||
static if (dialect.hasFeature(Dialect.singleQuoteQuotedStrings)) {
|
||
return (inQuotedString == QuotedString.single)
|
||
? Result.end : Result.regular;
|
||
} else {
|
||
return Result.regular;
|
||
}
|
||
|
||
case '#':
|
||
if (dialect.hasFeature(Dialect.hashInlineComments)) {
|
||
return (inQuotedString != QuotedString.none)
|
||
? Result.regular : Result.end;
|
||
} else {
|
||
return Result.regular;
|
||
}
|
||
|
||
case ';':
|
||
if (dialect.hasFeature(Dialect.inlineComments)) {
|
||
return (inQuotedString != QuotedString.none)
|
||
? Result.regular : Result.end;
|
||
} else {
|
||
return Result.regular;
|
||
}
|
||
|
||
case ':':
|
||
static if (dialect.hasFeature(Dialect.colonKeys)) {
|
||
goto case '=';
|
||
} else {
|
||
return Result.regular;
|
||
}
|
||
|
||
case '=':
|
||
static if (tokenType == TokenType.key) {
|
||
return (inQuotedString != QuotedString.none)
|
||
? Result.regular : Result.end;
|
||
} else {
|
||
return Result.regular;
|
||
}
|
||
|
||
case ']':
|
||
static if (tokenType == TokenType.sectionHeader) {
|
||
return (inQuotedString != QuotedString.none)
|
||
? Result.regular : Result.end;
|
||
} else {
|
||
return Result.regular;
|
||
}
|
||
}
|
||
|
||
assert(false, "Bug: This should have been unreachable.");
|
||
}
|
||
|
||
size_t idxLastText = 0;
|
||
foreach (immutable idx, const c; _source) {
|
||
const status = nextChar(c);
|
||
|
||
if (status == Result.end) {
|
||
break;
|
||
} else if (status == Result.whitespace) {
|
||
continue;
|
||
}
|
||
|
||
idxLastText = idx;
|
||
}
|
||
|
||
const idxEOT = (idxLastText + 1);
|
||
auto token = Token(tokenType, _source[0 .. idxEOT]);
|
||
_source = _source[idxEOT .. $];
|
||
|
||
if (inQuotedString != QuotedString.none) {
|
||
if (_source.length > 0) {
|
||
// chomp quote terminator
|
||
_source = _source[1 .. $];
|
||
}
|
||
}
|
||
|
||
return token;
|
||
}
|
||
|
||
Token lexText() {
|
||
final switch (_locationState) {
|
||
case LocationState.newLine:
|
||
case LocationState.key:
|
||
return this.lexTextImpl!(TokenType.key);
|
||
|
||
case LocationState.value:
|
||
return this.lexTextImpl!(TokenType.value);
|
||
|
||
case LocationState.sectionHeader:
|
||
return this.lexTextImpl!(TokenType.sectionHeader);
|
||
}
|
||
}
|
||
|
||
Token fetchFront() {
|
||
switch (_source[0]) {
|
||
|
||
default:
|
||
return this.lexText();
|
||
|
||
case '\x0A': {
|
||
_locationState = LocationState.newLine;
|
||
return this.makeToken(TokenType.lineBreak, 1);
|
||
}
|
||
|
||
case '\x0D': {
|
||
_locationState = LocationState.newLine;
|
||
|
||
// CR<EOF>?
|
||
if (this.isOnFinalChar) {
|
||
return this.makeToken(TokenType.lineBreak, 1);
|
||
}
|
||
|
||
// CRLF?
|
||
if (_source[1] == '\x0A') {
|
||
return this.makeToken(TokenType.lineBreak, 2);
|
||
}
|
||
|
||
// CR
|
||
return this.makeToken(TokenType.lineBreak, 1);
|
||
}
|
||
|
||
case '\x09':
|
||
case '\x0B':
|
||
case '\x0C':
|
||
case ' ':
|
||
return this.lexWhitespace();
|
||
|
||
case ':':
|
||
static if (dialect.hasFeature(Dialect.colonKeys)) {
|
||
goto case '=';
|
||
}
|
||
return this.lexText();
|
||
|
||
case '=':
|
||
_locationState = LocationState.value;
|
||
return this.makeToken(TokenType.keyValueSeparator, 1);
|
||
|
||
case '[':
|
||
_locationState = LocationState.sectionHeader;
|
||
return this.makeToken(TokenType.bracketOpen, 1);
|
||
|
||
case ']':
|
||
_locationState = LocationState.key;
|
||
return this.makeToken(TokenType.bracketClose, 1);
|
||
|
||
case ';': {
|
||
static if (dialect.hasFeature(Dialect.inlineComments)) {
|
||
return this.lexComment();
|
||
} else static if (dialect.hasFeature(Dialect.lineComments)) {
|
||
if (this.isAtStartOfLineOrEquivalent) {
|
||
return this.lexComment();
|
||
}
|
||
return this.lexText();
|
||
} else {
|
||
return this.lexText();
|
||
}
|
||
}
|
||
|
||
case '#': {
|
||
static if (dialect.hasFeature(Dialect.hashInlineComments)) {
|
||
return this.lexComment();
|
||
} else static if (dialect.hasFeature(Dialect.hashLineComments)) {
|
||
if (this.isAtStartOfLineOrEquivalent) {
|
||
return this.lexComment();
|
||
}
|
||
return this.lexText();
|
||
} else {
|
||
return this.lexText();
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/++
|
||
Low-level INI parser with filtered output
|
||
|
||
This wrapper will only supply tokens of these types:
|
||
|
||
$(LIST
|
||
* IniTokenType.key
|
||
* IniTokenType.value
|
||
* IniTokenType.sectionHeader
|
||
* IniTokenType.invalid
|
||
)
|
||
|
||
See_also:
|
||
$(LIST
|
||
* [IniParser]
|
||
* [parseIniDocument]
|
||
* [parseIniAA]
|
||
)
|
||
+/
|
||
struct IniFilteredParser(
|
||
IniDialect dialect = IniDialect.defaults,
|
||
string = immutable(char)[],
|
||
) {
|
||
///
|
||
public alias Token = IniToken!string;
|
||
|
||
private IniParser!(dialect, string) _parser;
|
||
|
||
public @safe pure nothrow @nogc:
|
||
|
||
///
|
||
public this(IniParser!(dialect, string) parser) {
|
||
_parser = parser;
|
||
}
|
||
|
||
///
|
||
public this(string rawIni) {
|
||
auto parser = IniParser!(dialect, string)(rawIni);
|
||
this(parser);
|
||
}
|
||
|
||
///
|
||
bool empty() => _parser.skipIrrelevant(true);
|
||
|
||
///
|
||
inout(Token) front() inout => _parser.front;
|
||
|
||
///
|
||
void popFront() => _parser.popFront();
|
||
|
||
///
|
||
inout(typeof(this)) save() inout {
|
||
return this;
|
||
}
|
||
}
|
||
|
||
///
|
||
@safe unittest {
|
||
// INI document (demo data)
|
||
static immutable string rawIniDocument = `; This is a comment.
|
||
[section1]
|
||
foo = bar ;another comment
|
||
oachkatzl = schwoaf ;try pronouncing that
|
||
`;
|
||
|
||
// Combine feature flags to build the required dialect.
|
||
const myDialect = (IniDialect.defaults | IniDialect.inlineComments);
|
||
|
||
// Instantiate a new parser and supply our document string.
|
||
auto parser = IniParser!(myDialect)(rawIniDocument);
|
||
|
||
int comments = 0;
|
||
int sections = 0;
|
||
int keys = 0;
|
||
int values = 0;
|
||
|
||
// Process token by token.
|
||
foreach (const parser.Token token; parser) {
|
||
if (token.type == IniTokenType.comment) {
|
||
++comments;
|
||
}
|
||
if (token.type == IniTokenType.sectionHeader) {
|
||
++sections;
|
||
}
|
||
if (token.type == IniTokenType.key) {
|
||
++keys;
|
||
}
|
||
if (token.type == IniTokenType.value) {
|
||
++values;
|
||
}
|
||
}
|
||
|
||
assert(comments == 3);
|
||
assert(sections == 1);
|
||
assert(keys == 2);
|
||
assert(values == 2);
|
||
}
|
||
|
||
@safe unittest {
|
||
static immutable string rawIniDocument = `; This is a comment.
|
||
[section1]
|
||
s1key1 = value1
|
||
s1key2 = value2
|
||
|
||
; Another comment
|
||
|
||
[section no.2]
|
||
s2key1 = "value3"
|
||
s2key2 = value no.4
|
||
`;
|
||
|
||
auto parser = IniParser!()(rawIniDocument);
|
||
alias Token = typeof(parser).Token;
|
||
|
||
{
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.comment, " This is a comment."));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front.type == TokenType.lineBreak);
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.bracketOpen, "["));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.sectionHeader, "section1"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.bracketClose, "]"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front.type == TokenType.lineBreak);
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.key, "s1key1"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.whitespace, " "));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.keyValueSeparator, "="));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.whitespace, " "));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.value, "value1"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front.type == TokenType.lineBreak);
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.key, "s1key2"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.value, "value2"), parser.front.data);
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front.type == TokenType.lineBreak);
|
||
}
|
||
|
||
{
|
||
assert(!parser.skipIrrelevant());
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.sectionHeader, "section no.2"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.key, "s2key1"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.value, "value3"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.key, "s2key2"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(!parser.empty);
|
||
assert(parser.front == Token(TokenType.value, "value no.4"));
|
||
}
|
||
|
||
parser.popFront();
|
||
assert(parser.skipIrrelevant());
|
||
assert(parser.empty());
|
||
}
|
||
|
||
@safe unittest {
|
||
static immutable rawIni = "#not-a = comment";
|
||
auto parser = makeIniParser(rawIni);
|
||
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, "#not-a"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "comment"));
|
||
|
||
parser.popFront();
|
||
assert(parser.empty);
|
||
}
|
||
|
||
@safe unittest {
|
||
static immutable rawIni = "#actually_a = comment\r\n\t#another one\r\n\t\t ; oh, and a third one";
|
||
enum dialect = (Dialect.hashLineComments | Dialect.lineComments);
|
||
auto parser = makeIniParser!dialect(rawIni);
|
||
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.comment, "actually_a = comment"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
assert(parser.front == parser.Token(TokenType.comment, "another one"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
assert(parser.front == parser.Token(TokenType.comment, " oh, and a third one"));
|
||
|
||
parser.popFront();
|
||
assert(parser.empty);
|
||
}
|
||
|
||
@safe unittest {
|
||
static immutable rawIni = ";not a = line comment\nkey = value ;not-a-comment \nfoo = bar # not a comment\t";
|
||
enum dialect = Dialect.lite;
|
||
auto parser = makeIniParser!dialect(rawIni);
|
||
|
||
{
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, ";not a"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "line comment"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front.type == TokenType.key);
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "value ;not-a-comment"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front.type == TokenType.key);
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "bar # not a comment"));
|
||
}
|
||
}
|
||
|
||
@safe unittest {
|
||
static immutable rawIni = "; line comment 0\t\n\nkey = value ; comment-1\nfoo = bar #comment 2\n";
|
||
enum dialect = (Dialect.inlineComments | Dialect.hashInlineComments);
|
||
auto parser = makeIniParser!dialect(rawIni);
|
||
|
||
{
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.comment, " line comment 0\t"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
assert(parser.front.type == TokenType.key);
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
assert(parser.front == parser.Token(TokenType.value, "value"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
assert(parser.front == parser.Token(TokenType.comment, " comment-1"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
assert(parser.front.type == TokenType.key);
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
assert(parser.front == parser.Token(TokenType.value, "bar"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
assert(parser.front == parser.Token(TokenType.comment, "comment 2"));
|
||
}
|
||
|
||
parser.popFront();
|
||
assert(parser.skipIrrelevant(false));
|
||
}
|
||
|
||
@safe unittest {
|
||
static immutable rawIni = "key = value;inline";
|
||
enum dialect = Dialect.inlineComments;
|
||
auto parser = makeIniParser!dialect(rawIni);
|
||
|
||
assert(!parser.empty);
|
||
parser.front == parser.Token(TokenType.key, "key");
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
parser.front == parser.Token(TokenType.value, "value");
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant(false));
|
||
parser.front == parser.Token(TokenType.comment, "inline");
|
||
|
||
parser.popFront();
|
||
assert(parser.empty);
|
||
}
|
||
|
||
@safe unittest {
|
||
static immutable rawIni = "key: value\n"
|
||
~ "foo= bar\n"
|
||
~ "lol :rofl\n"
|
||
~ "Oachkatzl : -Schwoaf\n"
|
||
~ `"Schüler:innen": 10`;
|
||
enum dialect = (Dialect.colonKeys | Dialect.quotedStrings);
|
||
auto parser = makeIniParser!dialect(rawIni);
|
||
|
||
{
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, "key"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "value"));
|
||
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.key, "foo"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "bar"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.key, "lol"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "rofl"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.key, "Oachkatzl"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "-Schwoaf"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.key, "Schüler:innen"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.skipIrrelevant());
|
||
assert(parser.front == parser.Token(TokenType.value, "10"));
|
||
}
|
||
|
||
parser.popFront();
|
||
assert(parser.skipIrrelevant());
|
||
}
|
||
|
||
@safe unittest {
|
||
static immutable rawIni =
|
||
"\"foo=bar\"=foobar\n"
|
||
~ "'foo = bar' = foo_bar\n"
|
||
~ "foo = \"bar\"\n"
|
||
~ "foo = 'bar'\n"
|
||
~ "multi_line = 'line1\nline2'\n"
|
||
~ "syntax = \"error";
|
||
enum dialect = (Dialect.quotedStrings | Dialect.singleQuoteQuotedStrings);
|
||
auto parser = makeIniFilteredParser!dialect(rawIni);
|
||
|
||
{
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, "foo=bar"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.value, "foobar"));
|
||
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, "foo = bar"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.value, "foo_bar"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, "foo"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.value, "bar"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, "foo"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.value, "bar"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, "multi_line"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.value, "line1\nline2"));
|
||
}
|
||
|
||
{
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.key, "syntax"));
|
||
|
||
parser.popFront();
|
||
assert(!parser.empty);
|
||
assert(parser.front == parser.Token(TokenType.value, "error"));
|
||
}
|
||
|
||
parser.popFront();
|
||
assert(parser.empty);
|
||
}
|
||
|
||
/++
|
||
Convenience function to create a low-level parser
|
||
|
||
$(TIP
|
||
Unlike with the constructor of [IniParser],
|
||
the compiler is able to infer the `string` template parameter.
|
||
)
|
||
|
||
See_also:
|
||
[makeIniFilteredParser]
|
||
+/
|
||
IniParser!(dialect, string) makeIniParser(
|
||
IniDialect dialect = IniDialect.defaults,
|
||
string = immutable(char)[],
|
||
)(
|
||
string rawIni,
|
||
) @safe pure nothrow @nogc if (isCompatibleString!string) {
|
||
return IniParser!(dialect, string)(rawIni);
|
||
}
|
||
|
||
///
|
||
@safe unittest {
|
||
string regular;
|
||
auto parser1 = makeIniParser(regular);
|
||
assert(parser1.empty); // exclude from docs
|
||
|
||
char[] mutable;
|
||
auto parser2 = makeIniParser(mutable);
|
||
assert(parser2.empty); // exclude from docs
|
||
|
||
const(char)[] constChars;
|
||
auto parser3 = makeIniParser(constChars);
|
||
assert(parser3.empty); // exclude from docs
|
||
}
|
||
|
||
/++
|
||
Convenience function to create a low-level filtered parser
|
||
|
||
$(TIP
|
||
Unlike with the constructor of [IniFilteredParser],
|
||
the compiler is able to infer the `string` template parameter.
|
||
)
|
||
|
||
See_also:
|
||
[makeIniParser]
|
||
+/
|
||
IniFilteredParser!(dialect, string) makeIniFilteredParser(
|
||
IniDialect dialect = IniDialect.defaults,
|
||
string = immutable(char)[],
|
||
)(
|
||
string rawIni,
|
||
) @safe pure nothrow @nogc if (isCompatibleString!string) {
|
||
return IniFilteredParser!(dialect, string)(rawIni);
|
||
}
|
||
|
||
///
|
||
@safe unittest {
|
||
string regular;
|
||
auto parser1 = makeIniFilteredParser(regular);
|
||
assert(parser1.empty); // exclude from docs
|
||
|
||
char[] mutable;
|
||
auto parser2 = makeIniFilteredParser(mutable);
|
||
assert(parser2.empty); // exclude from docs
|
||
|
||
const(char)[] constChars;
|
||
auto parser3 = makeIniFilteredParser(constChars);
|
||
assert(parser3.empty); // exclude from docs
|
||
}
|
||
|
||
// undocumented
|
||
debug {
|
||
void writelnTokens(IniDialect dialect, string)(IniParser!(dialect, string) parser) @safe {
|
||
import std.stdio : writeln;
|
||
|
||
foreach (token; parser) {
|
||
writeln(token);
|
||
}
|
||
}
|
||
|
||
void writelnTokens(IniDialect dialect, string)(IniFilteredParser!(dialect, string) parser) @safe {
|
||
import std.stdio : writeln;
|
||
|
||
foreach (token; parser) {
|
||
writeln(token);
|
||
}
|
||
}
|
||
}
|
||
|
||
/++
|
||
Data entry of an INI document
|
||
+/
|
||
struct IniKeyValuePair(string) if (isCompatibleString!string) {
|
||
///
|
||
string key;
|
||
|
||
///
|
||
string value;
|
||
}
|
||
|
||
/++
|
||
Section of an INI document
|
||
|
||
$(NOTE
|
||
Data entries from the document’s root – i.e. those with no designated section –
|
||
are stored in a section with its `name` set to `null`.
|
||
)
|
||
+/
|
||
struct IniSection(string) if (isCompatibleString!string) {
|
||
///
|
||
alias KeyValuePair = IniKeyValuePair!string;
|
||
|
||
/++
|
||
Name of the section
|
||
|
||
Also known as “key”.
|
||
+/
|
||
string name;
|
||
|
||
/++
|
||
Data entries of the section
|
||
+/
|
||
KeyValuePair[] items;
|
||
}
|
||
|
||
/++
|
||
DOM representation of an INI document
|
||
+/
|
||
struct IniDocument(string) if (isCompatibleString!string) {
|
||
///
|
||
alias Section = IniSection!string;
|
||
|
||
/++
|
||
Sections of the document
|
||
|
||
$(NOTE
|
||
Data entries from the document’s root – i.e. those with no designated section –
|
||
are stored in a section with its `name` set to `null`.
|
||
|
||
If there are no named sections in a document, there will be only a single section with no name (`null`).
|
||
)
|
||
+/
|
||
Section[] sections;
|
||
}
|
||
|
||
/++
|
||
Parses an INI string into a document ("DOM").
|
||
|
||
See_also:
|
||
[parseIniAA]
|
||
+/
|
||
IniDocument!string parseIniDocument(IniDialect dialect = IniDialect.defaults, string)(string rawIni) @safe pure nothrow
|
||
if (isCompatibleString!string) {
|
||
alias Document = IniDocument!string;
|
||
alias Section = IniSection!string;
|
||
alias KeyValuePair = IniKeyValuePair!string;
|
||
|
||
auto parser = IniParser!(dialect, string)(rawIni);
|
||
|
||
auto document = Document(null);
|
||
auto section = Section(null, null);
|
||
auto kvp = KeyValuePair(null, null);
|
||
|
||
void commitKeyValuePair(string nextKey = null) {
|
||
if (kvp.key !is null) {
|
||
section.items ~= kvp;
|
||
}
|
||
kvp = KeyValuePair(nextKey, null);
|
||
}
|
||
|
||
void commitSection(string nextSectionName) {
|
||
commitKeyValuePair(null);
|
||
|
||
const isNamelessAndEmpty = (
|
||
(section.name is null)
|
||
&& (section.items.length == 0)
|
||
);
|
||
|
||
if (!isNamelessAndEmpty) {
|
||
document.sections ~= section;
|
||
}
|
||
|
||
if (nextSectionName !is null) {
|
||
section = Section(nextSectionName, null);
|
||
}
|
||
}
|
||
|
||
while (!parser.skipIrrelevant()) {
|
||
switch (parser.front.type) with (TokenType) {
|
||
|
||
case key:
|
||
commitKeyValuePair(parser.front.data);
|
||
break;
|
||
|
||
case value:
|
||
kvp.value = parser.front.data;
|
||
break;
|
||
|
||
case sectionHeader:
|
||
commitSection(parser.front.data);
|
||
break;
|
||
|
||
default:
|
||
assert(false, "Unexpected parsing error."); // TODO
|
||
}
|
||
|
||
parser.popFront();
|
||
}
|
||
|
||
commitSection(null);
|
||
|
||
return document;
|
||
}
|
||
|
||
///
|
||
@safe unittest {
|
||
// INI document (demo data)
|
||
static immutable string iniString = `; This is a comment.
|
||
|
||
Oachkatzlschwoaf = Seriously, try pronouncing this :P
|
||
|
||
[Section #1]
|
||
foo = bar
|
||
d = rocks
|
||
|
||
; Another comment
|
||
|
||
[Section No.2]
|
||
name = Walter Bright
|
||
company = "Digital Mars"
|
||
`;
|
||
|
||
// Parse the document.
|
||
auto doc = parseIniDocument(iniString);
|
||
|
||
version (none) // exclude from docs
|
||
// …is equivalent to:
|
||
auto doc = parseIniDocument!(IniDialect.defaults)(iniString);
|
||
|
||
assert(doc.sections.length == 3);
|
||
|
||
// "Root" section (no name):
|
||
assert(doc.sections[0].name is null);
|
||
assert(doc.sections[0].items == [
|
||
IniKeyValuePair!string("Oachkatzlschwoaf", "Seriously, try pronouncing this :P"),
|
||
]);
|
||
|
||
// A section with a name:
|
||
assert(doc.sections[1].name == "Section #1");
|
||
assert(doc.sections[1].items.length == 2);
|
||
assert(doc.sections[1].items[0] == IniKeyValuePair!string("foo", "bar"));
|
||
assert(doc.sections[1].items[1] == IniKeyValuePair!string("d", "rocks"));
|
||
|
||
// Another section:
|
||
assert(doc.sections[2].name == "Section No.2");
|
||
assert(doc.sections[2].items == [
|
||
IniKeyValuePair!string("name", "Walter Bright"),
|
||
IniKeyValuePair!string("company", "Digital Mars"),
|
||
]);
|
||
}
|
||
|
||
@safe unittest {
|
||
auto doc = parseIniDocument("");
|
||
assert(doc.sections == []);
|
||
|
||
doc = parseIniDocument(";Comment\n;Comment2\n");
|
||
assert(doc.sections == []);
|
||
}
|
||
|
||
@safe unittest {
|
||
char[] mutable = ['f', 'o', 'o', '=', 'b', 'a', 'r', '\n'];
|
||
|
||
auto doc = parseIniDocument(mutable);
|
||
assert(doc.sections[0].items[0].key == "foo");
|
||
assert(doc.sections[0].items[0].value == "bar");
|
||
|
||
// is mutable
|
||
static assert(is(typeof(doc.sections[0].items[0].value) == char[]));
|
||
}
|
||
|
||
/++
|
||
Parses an INI string into an associate array.
|
||
|
||
See_also:
|
||
[parseIniDocument]
|
||
+/
|
||
string[string][string] parseIniAA(IniDialect dialect = IniDialect.defaults, string)(string rawIni) @safe pure nothrow {
|
||
// TODO: duplicate handling
|
||
auto parser = IniParser!(dialect, string)(rawIni);
|
||
|
||
string[string][string] document;
|
||
string[string] section;
|
||
|
||
string sectionName = null;
|
||
string keyName = null;
|
||
|
||
void commitSection() {
|
||
sectionName = null;
|
||
}
|
||
|
||
while (!parser.skipIrrelevant()) {
|
||
switch (parser.front.type) with (TokenType) {
|
||
|
||
case key:
|
||
keyName = parser.front.data;
|
||
break;
|
||
|
||
case value:
|
||
section[keyName] = parser.front.data;
|
||
break;
|
||
|
||
case sectionHeader:
|
||
if ((sectionName !is null) || (section.length > 0)) {
|
||
document[sectionName] = section;
|
||
section = null;
|
||
}
|
||
sectionName = parser.front.data;
|
||
break;
|
||
|
||
default:
|
||
assert(false, "Unexpected parsing error."); // TODO
|
||
}
|
||
|
||
parser.popFront();
|
||
}
|
||
|
||
if ((sectionName !is null) || (section.length > 0)) {
|
||
document[sectionName] = section;
|
||
}
|
||
|
||
return document;
|
||
}
|
||
|
||
///
|
||
@safe unittest {
|
||
// INI document
|
||
static immutable string demoData = `; This is a comment.
|
||
|
||
Oachkatzlschwoaf = Seriously, try pronouncing this :P
|
||
|
||
[Section #1]
|
||
foo = bar
|
||
d = rocks
|
||
|
||
; Another comment
|
||
|
||
[Section No.2]
|
||
name = Walter Bright
|
||
company = "Digital Mars"
|
||
website = <https://digitalmars.com/>
|
||
;email = "noreply@example.org"
|
||
`;
|
||
|
||
// Parse the document into an associative array.
|
||
auto aa = parseIniAA(demoData);
|
||
|
||
assert(aa.length == 3);
|
||
|
||
assert(aa[null].length == 1);
|
||
assert(aa[null]["Oachkatzlschwoaf"] == "Seriously, try pronouncing this :P");
|
||
|
||
assert(aa["Section #1"].length == 2);
|
||
assert(aa["Section #1"]["foo"] == "bar");
|
||
assert(aa["Section #1"]["d"] == "rocks");
|
||
|
||
string[string] section2 = aa["Section No.2"];
|
||
assert(section2.length == 3);
|
||
assert(section2["name"] == "Walter Bright");
|
||
assert(section2["company"] == "Digital Mars");
|
||
assert(section2["website"] == "<https://digitalmars.com/>");
|
||
|
||
// "email" is commented out
|
||
assert(!("email" in section2));
|
||
}
|