jsinf
v1.1.2
Published
A JSINF reference implementation
Downloads
13
Readme
jsinf
This comprises the reference decoder for jsinf format.
Are you looking for something other than YAML or JSON for your configuration files? Perhaps JSINF is an answer. If you know the standard .INI files of windows, then this copes with those... but it also does a lot more:
- Returns the config as a standard Javascript object, that you can easily convert to JSON if you wish.
- Supports nested objects (unlike a standard .INI file). eg, can read .gitignore.
- Supports values that extend over multiple lines!
- Supports 'default' blocks. That is, you don't even need to start with
some_property =
!!!
The API, is similar to JSON with parse
and stringify
:
var jsinf = require('jsinf');
...
var obj = jsinf.parse(some_data);
var text = jsinf.stringify(obj);
Note:
- Of course, comments and whitespace are not retained when using
jsinf.stringify(jsinf.parse(some_data))
, but the data integrity will remain intact. (See the tests to verify this).
A Quick Look at the Format
a_property = some value
[section1]
key1 = value1
key2 = value2
[section2]
# this is a comment.
key1 = value3
[section2.nested]
#A comment for a nested object
nested_key = A
value that
runs over many lines,
but can't have a blank line.
---
A block of text that gets assigned
to the property called 'value'
You can put pretty much anything in
here. You don't need the final --- if the
end of the file is reached.
#this is not a comment. It stays in
#with this text block unmolested.
---
Installation
Easiest is with npm, in an existing npm project:
npm install jsinf --save
Usage
const jsinf = require('jsinf');
...
var inf_file = `
[section]
value1 = a value
value2 = another value
[another section]
# a comment!
value2 = some 3rd value
`;
var inf_data = jfinf.decode(inf_file);
console.log(inf_data.section.value1); // prints 'a value';
The Format
The format is extremely forgiving and designed for non-technical people.
Nested Objects
[section.sub section]
name=John D. Smith
becomes the equivalent JSON:
{
"section" : {
"sub section" : {
"name": "John D. Smith"
}
}
}
The default seperator is a dot '.', however you can change it easily:
// use # as a sub section divider.
var inf_data = jsinf.decode(inf_file, { subsection_divider: '#'});
Note:
- Valid chars for subsections are any word or number, and '-' , '.', '/', '#', '$' and '\'
Multi-line support
[section]
address=32 Fallows Way
Timbuctoo
Somewhere in Africa
Note that blank lines are not allowed. You may wish to consider 'Default Value' and 'Block Default Value' options instead.
Default Values
[section]
I am some
text that
runs over several
lines!
And can have single blank lines in it.
becomes:
{
"section": {
"value": "I am some\ntext that..."
}
}
Block Default Value
If you have strong requirements for a large block with lots of 'noise' in it, then you may wish to consider a block:
[section]
name=John D. Smith
---
I am a block
that runs until --- is found on it's own line
[this is not a section]
# this is not a comment
---
Note:
You can change the property name that is used:
// use 'text' as the default.
var inf_data = jsinf.decode(inf_file, { default_key: 'text'});
console.log(inf_data.section.text); // prints 'I am a block\n\n...'
The data would look like:
{
"section": {
"name" : "John D. Smith"
"text": "I am a block\n\n..."
}
}
Note:
- There can only be one block per section. If multiple are detected, the last one is used.
You can change the separator sequence with:
// Change the divider to 5 (or more) consecutive asterisks.
var inf_data = jsinf.decode(inf_file, { block_divider: '\\*{5,}'});
Parse Options
The following options are available when parsing:
default_key: 'value' // This specifies where 'default values' are placed. Default is 'value'
, subsection_divider: '.' // This specifies how we split a [section.sub section.subsub section]
// If Empty, does not split at all, and everything is just a 'section'
// Can be a function:
// .eg for a .gitconfig styled: [diff "difftool"]:
, subsection_divider: function(section) {
return section.split(' ').map(function(name) { // splits on a space
return name.replace(/^\"(.*?)\"$/, '$1') }); // trims out sub-section surrounding quotes
}
, subsection_nameclean: // This allows the 'clean up' of a sub-section name.
// By default the string is trimmed of whitespace and (matching) surrounding quotes
// eg.
function(name) {
return name.replace( /^\"(.*?)\"$/, '$1');
}
/*
key, comment & section_name charsets
allows (some) control of what is ok and what is not
if a RegExp is defined, then full control is given (Unless careful, problems will ensue)
*/
, valid_key_chars: "-./\\#$'\"" // Word Characters A-Za-z90-9_ are included by default.
// Use a RegExp for full control (TAKE CARE! don't use an equals!):
, valid_key_chars: /[\w\-\.\/\#\$\\\'\"]*[\w]/
, valid_section_chars: " -./\\#$'\"" // <== has a 'space' in there
, valid_section_chars: /[\w\'\"]*?[\w\-\.\/ \#\$\\\'\"]*[\w\'\"]/ // TAKE CARE! don't use an equals!
, valid_comment_chars: "#;"
, valid_comment_chars: /^[#;](.*)(?:\n|$)/
, block_divider: '\\-{3,}' // also '\\-\\-\\-+'
, block_divider: /\-{3,}/
, allow_code: true // Allow values with code blocks. eg { (123+4).toString() + (new Date()).toString() + " ok!"}
// If concerned with security/ config is from an untrusted source: SET THIS TO FALSE!
, allow_arrays: true // allow values as an array: eg [ value1, value2, value3 ].
// Limited usage. Only for simple Array types (without commas). Is equal to:
// "value1, value2, value3".split(',').map(function(item) { return item.replace(/^\s*(.*?)\s*$/,'$1')})
, log: true // Enable logging of internals. Good for debugging, but that's about it.
, log: function(str) { console.log(str); }
##Stringify Options
The following options are available when stringifying:
default_key: 'value' // This specifies where 'default values' are placed. Default is 'value'
, subsection_divider: '.' // This specifies how we split a [section.sub section.subsub section]
// (If Empty, does not split at all, and everything is just a 'section', NOT RECOMMENDED)
, block_divider: '---' // This MUST match the 'parse' regular expression, but must be a fixed string here
, log: true // Enable logging of internals. Good for debugging, but that's about it.
, log: function(str) { console.log(str); }