This is a spec for a very well-designed declarative language and a Python package to work with it.
Obviously, current solutions for config et al. (JSON, YAML, TOML) are very unsuitable as software development progress is going on, so here I propose a brand new format.
WTFL has all the features of its predecessors, and much more, with a very well-designed friendly syntax.
It is case-insensitive, obvious and feature-rich language.
Here is an example of the format:
...This is a comment
key's "value"
weight of apple is 123
This is equivalent to this JSON:
{
"key": "value",
"apple": {
"weight": 123
}
}
Package's API is similar to built-in json
:
read an object from a string:
def wtfl.loads(
s, # string to load
*,
parse_float, # function for float parsing, `float` used by default
parse_int, # function for integer parsing (only unprefixed decimal), `int` used by default
parse_roman, # function for roman numerals parsing, accepts whole literal as a string (0r...)
parse_numbers, # function for integer literals parsing (0b..., 0o..., and so on)
) ->
read an object from a file (same argument meaning as .loads
):
wtfl.load(file, *, parse_float, parse_int, parse_roman, parse_numbers)
dump an object into a string:
def wtfl.dumps(
obj: object, # object to dump
*,
# if True, non-serializable keys / values are skipped
skipkeys: bool = False,
# if True, encodes non-ASCII characters
ensure_ascii: bool = True,
# if a non-negative integer, used as indent size. if a string, used as indent character. If None, everything is inlined
indent: int | str | None = 2,
# a function that is used to recover from serialization errors.
# if current value is not serializable and skipkeys=False, tries to serialize the result of default(value)
default: Callable[[object], str] | None = None,
# if True, sorts object keys
sort_keys: bool = False,
)
dump an object into a file (same argument meaning as .dumps()
):
wtfl.dump(file, obj, *, skipkeys, ensure_ascii, indent, default, sort_keys)
It is very important to know all the keywords of WTFL, since all those keywords are important.
Here's a list (keywords in the same list entry are equivalent and used interchangeably in the spec):
is
/are
/'s
/'re
/do
/does
/be
isn't
/aren't
/'sn't
/'ren't
of
also
/and also
/but also
that
/this
/these
/those
there
have
/has
/'ve
haven't
/hasn't
/'ven't
can
cannot
/can't
true
/falsen't
false
/truen't
and
return
skip
stay
to
You can use keywords as names (except true / false / haven't), this list is just so you know them all.
Keys are at the center of the language. In WTFL, keys are defined in two ways:
- Same as values. Provide a value and it will be converted to string.
- As bare name. A bare name should start with any char of
[A-Za-z_-]
and consist of chars in[A-Za-z0-9_-]
.
Any key can have optional piece preceding it. Possible choices are: a
, an
, the
, de
, du
, le
, la
, les
, des
, but
and um
.
Here are some examples:
32 is 32
0x45 is 0x45
4.5's 4.5
"key2" be "key2"
the jasdofia913247 is "what?"
a mogu's "sus"
It's equivalent to this JSON:
{
"32": 32,
"69": 69,
"4.5": 4.5,
"key2": "key2",
"jasdofia913247": "what?",
"mogu": "sus"
}
Many formats, such as TOML, don't allow writing everything in one line. That's a shame, since, as we all know, writing everything in one line greatly improves readability and this block is a living proof of that rule. You can easily read everything written here without any discomfort, so why restrict a language to multiple lines?
Since inlining is so good, WTFL introduces it in a very simple manner, just write everything in one line:
apple does have weight is 123 color is "red" that is everything you need to know about apples but also key is 123
(With one exception: array/object definitions have to end in newline or also
)
Is equivalent to this JSON:
{
"apple": {
"weight": 123,
"color": "red"
},
"key": 123
}
Setting values to keys is straightforward using is
(or are
or any other alias):
key is 35
key2 be 0x56
This sets "key" to 35. But there are various other ways.
For example, if you are sure key has to have some value, use has to be
or have to be
:
key has to be 35
key is 36 ...no!!!! this would raise some error.
If you want to say the key can have some value, use can be
:
key can be 35
key is 36 ...well, okay, good luck next time
...or can't be
/ cannot be
for opposite meaning:
key can't be 35
key is 35 ...what? you just said it can't be! error here
You also can just allow and restrict keys:
key can be
key2 can't be
key is 23 ...okay
key2 is 23 ...nope
Be sure to not mess up the order (see Time Travel section for workarounds):
key is 23
key can't be 23 ...produces a warning "Too late, it's already done", but no error (since it's already done)
Setting key several times makes a lot of sense, but only the last value is preserved (see Time Travel section for workarounds)
key is 23
key is 35
...The key is 35 now
There are decimal integer literals, such as 102
, -4
, 0
, and, as expected, some other:
- Binary (2) with
0b
prefix:0b10010
- Octal (8) with
0o
prefix:0o755
- Decimal (10) with
0d
prefix:0d102
- Duodecimal (12) with
0z
prefix:0z9a1
- Vigesimal (20) with
0v
prefix:0vjija
- Roman numerals with
0r
prefix:0rXIMI
Floats are also supported: 0.1
, 2.
, .2
, 2.3e4
, 2.3e4
String literals are enclosed in double quotes, supporting all the usual Python escape sequences:
sus is "amogus"
...is represented by haven't
keyword:
value is haven't
Equivalent to:
{
"value": null
}
That type is very simple: You have true
/ falsen't
, and you have false
/ truen't
Nested structures are often needed, so WTFL has those!
Suppose you have this JSON:
{
"apple": {
"weight": 123,
"color": "red"
}
}
Here's how you define this complex object with WTFL, using have
/ has
keyword:
apple does have
weight is 123
color is "red"
that's all folks!
Indentation is optional, as well as the exclamation mark after "that's all folks".
In fact, you can use any line to end the object, as long as it starts with (that|there)
:
that's all
that is everything, folks
there's nothing more I think
there are no more keys, go home
there it is, the end of the object!
Be aware that you cannot use also
in that line (see Inline section for more info)
If you want something less structured, you can define keys one-by-one:
weight of apple is 123
color of apple is "red"
As any good format, WTFL has arrays. Arrays are defined very straightforwardly:
apple_crate does have
has
weight is 123
color is "red"
that was the object
23 ...what? a number in apple crate? okay...
that was the array
Equivalent JSON:
{
"apple_crate": [
{
"weight": 123,
"color": "red"
},
23
]
}
Be aware that...
has
that's all
...is an empty object. If you want an empty list, use:
has 0
If you create an array and then add a non-integer key, it will turn into an object:
apple_crate does have
has
weight is 123
color is "red"
that was the object
23 ...what? a number in apple crate? okay...
that was the array
frog of apple_crate is 10
Produces:
{
"apple_crate": {
"0": {
"weight": 123,
"color": "red"
},
"1": 23,
"frog": 10
}
}
Keys less than 0 are ignored, keys greater than array length are added, but moved so that there's no holes in the resulting array.
It is very obvious that there has to be a time travel feature in a modern config format.
You can travel to the past, to the future, and (experimental) to the present
To travel to the past, use return
keyword:
key is 24
...1 is offset in statements. Your statement will be transported right before `key is 24`
return 1 and key can't be 24
This would raise an error, because key can't be 24 at line 1
To travel to the future, use skip
keyword:
skip 2 and key is 35
key is 24
This would set key to 35, because statement 0 is transported two statements forward, after key is 35
.
This is highly experimental. Use at your own risk.
You can stay in the present (not travelling anywhere)
stay and key is 35
key is 24
-
If you travel back in time and kill your parents, you will never be born This is sad, but doesn't influence WTFL
-
If you travel back in time and kill my parents, I will never be born Please don't exploit this critical vulnerability
-
If you travel in the same line several times, you will create a chain of events inside a single time point. Maybe that's what you want, maybe all bees are aliens. I can't know everything.
- I've decided to not add dates / time support, because of the time travel (and possibly date travel)