This document is an outline of the guiding design principles of Fennel. Fennel's community values are covered in the code of conduct.
First and foremost is the notion that Fennel is a compiler with no runtime. This places somewhat severe limits on what we can accomplish, but it also creates a valuable sense of focus. We are of course very fortunate to be building on a language like Lua where the runtime semantics are for the most part excellent, and the areas upon which we improve can be identified at compile time.
This means Fennel (the language) consists entirely of macros and special forms, and no functions. Fennel (the compiler) of course has plenty of functions in it, but they are for the most part not intended for use outside the context of embedding the compiler in another Lua program.
The exception to this rule is fennel.view
which can be
used independently; it addresses a severe problem in Lua's runtime
semantics where tostring
on a table produces nearly-useless
results. But this can be thought of as simply another library which
happens to be included in the compiler. The fennel.view
function is a prerequisite to having a useful repl.
The repl of course is also a function you can call at runtime if you embed the compiler, but this is a special case that blurs the lines between runtime and compile time. After all, what is compile time except that subset of runtime during which the function being run happens to be a compiler?
Well-written Lua programs exhibit an excellent sense of transparency largely due to how Lua leans on lexical scoping so predominantly. When you look at a good Lua program, you can tell exactly where any given identifier comes from just by following the basic rules of lexical scope. Badly-written Lua programs often use globals and do not have this property.
With Fennel we try to take this even further by making globals an
error by default. It's still possible to write programs that use globals
using _G
(indeed for Lua interop this sometimes cannot be
avoided) but it should be very clear when this happens; it's not
something that you would do by accident or due to laziness.
One counter-example here is the deprecated
require-macros
form; it introduced new identifiers into the
scope without making it clear what the names were. That is why it was
replaced by the much clearer import-macros
. The two below
are equivalent, but one has hidden implicit scope changes and the other
exhibits transparency:
require-macros :my.macros) ; what did we introduce here? who knows!
(
import-macros {: transform-bar : skip-element} :my-macros) (
Of course this comes at the cost of a little extra verbosity, but it is well worth it. In Fennel programs, you should never have a hard time answering the question "where did this come from?"
The most obvious legitimate criticism of Lua is that it makes it easy
to set or read globals by accident simply by making a typo in the name
of an identifier. This is easily fixed by requiring global access to be
explicit; it's perhaps the most obvious way that Fennel tries to catch
common mistakes. But there are others; for instance Fennel does not
allow you to shadow the name of a special form with a local. It also
doesn't allow you to omit the body from a let
form like
many other lisps do:
fn abc []
(let [a 1
(2
b ; <- missing body!
c (calculate-c)]) + a b c)) (
This will be flagged as an error because the entire let
form is closed after the call to calculate-c
when the
intent was clearly only to close the binding form.
Another example would be that you can't call set
on a
local unless it is introduced using var
. This means that if
you have code which assumes the locals will remain the same and then go
and mess with that assumption it is an error; you have to explicitly
declare that assumption void first before you are permitted to violate
it.
This touches on a broader theme: it's easier to understand code when you can look at it and immediately know certain things will never happen. By excluding certain capabilities from the language, certain mistakes become impossible.
For example, Fennel code will never use a block of memory after it
has been freed, because malloc
and free
are
not even part of its vocabulary. In languages with immutable data
structures, it's impossible to have bugs which come from one piece of
code making a change to data in a way that another function did not
expect. Fennel does not have immutable data structures, but still we
recognize that removing the ability to do things (or making them opt-in
instead of opt-out) can significantly improve the resulting code.
Other examples include the lack of goto
and the lack of
early returns. Or how if a loop terminates early, it will make this
obvious by using an &until
clause at the top of the
loop; you don't have to read the entire loop body to search for a
break
as you would in Lua.
Older lisps overload parentheses to mean lots of different things;
they are used for data lists, but they are also used to signify function
and macro calls or to group key/value pairs together and around an
entire group of key/value pairs in let
. There are many
other uses.
Fennel overloads delimiters in a few ways, but the distinction should
be visually clearer and much more limited by context. Parentheses almost
always mean a function or macro call; the main exception is inside a
binding form where it can be used to bind multiple values. The other
exception is the now-deprecated ?
notation for pattern
matching guards; it has been replaced by calling where
.
Square brackets usually indicate a sequential table, but in a macro they
can indicate a binding form. Perhaps were Fennel rooted in a language
richer in typographical delimiters than English, this overloading would
not be necessary and every delimiter pair could have exactly one
meaning.
This is something Lua drops the ball on in a few places; it overloads
one notation to mean different things. For instance, for
in
Lua can be used to numerically step from one number to another in a
loop, or it can be used to step thru an iterator. Fennel separates this
out into for
to be used with numeric stepping and
each
which uses iterators. Another example is the table
literal notation: Lua uses {}
for sequential tables as well
as key/value tables, while Fennel uses []
for sequential
tables following more recent programming convention.
Fennel uses notation in other ways to avoid ambiguity; for instance
when &as
was introduced in destructuring forms for
giving access to the entire table, the &
character was
reserved so that it could not be used in identifiers. This also makes it
easier to write macros which do similar things; now we have a way to
indicate that a given symbol must have some meaning assigned to it other
than being an identifier.