mirror of https://github.com/adamdruppe/arsd.git
Add convenient INI DOM parser
This commit is contained in:
parent
fcc46ff41b
commit
2e12f1a8f5
221
ini.d
221
ini.d
|
@ -73,6 +73,11 @@ private enum LocationState {
|
||||||
|
|
||||||
/++
|
/++
|
||||||
Low-level INI parser
|
Low-level INI parser
|
||||||
|
|
||||||
|
See_also:
|
||||||
|
$(LIST
|
||||||
|
* [parseIniDocument]
|
||||||
|
)
|
||||||
+/
|
+/
|
||||||
struct IniParser(
|
struct IniParser(
|
||||||
IniDialect dialect = IniDialect.defaults,
|
IniDialect dialect = IniDialect.defaults,
|
||||||
|
@ -126,7 +131,7 @@ struct IniParser(
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
typeof(this) save() inout {
|
inout(typeof(this)) save() inout {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -442,9 +447,50 @@ struct IniParser(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
@safe unittest {
|
@safe unittest {
|
||||||
|
// INI document (demo data)
|
||||||
|
static immutable string rawIniDocument = `; This is a comment.
|
||||||
|
[section1]
|
||||||
|
foo = bar ;another comment
|
||||||
|
oachkatzl = schwoaf ;try pronouncing that
|
||||||
|
`;
|
||||||
|
|
||||||
static immutable document = `; This is a comment.
|
// Combine feature flags to build the required dialect.
|
||||||
|
const myDialect = (Dialect.defaults | Dialect.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]
|
[section1]
|
||||||
s1key1 = value1
|
s1key1 = value1
|
||||||
s1key2 = value2
|
s1key2 = value2
|
||||||
|
@ -456,7 +502,7 @@ s2key1 = "value3"
|
||||||
s2key2 = value no.4
|
s2key2 = value no.4
|
||||||
`;
|
`;
|
||||||
|
|
||||||
auto parser = IniParser!()(document);
|
auto parser = IniParser!()(rawIniDocument);
|
||||||
alias Token = typeof(parser).Token;
|
alias Token = typeof(parser).Token;
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -561,3 +607,172 @@ s2key2 = value no.4
|
||||||
assert(parser.skipIrrelevant());
|
assert(parser.skipIrrelevant());
|
||||||
assert(parser.empty());
|
assert(parser.empty());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
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").
|
||||||
|
+/
|
||||||
|
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)(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.");
|
||||||
|
}
|
||||||
|
|
||||||
|
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"),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue