Hatching Specs#
Fandango provides an include()
function that you can use to include existing Fandango content.
This allows you to distribute specifications over multiple files, defining base specs whose definitions can be further refined in specs that use them.
Including Specs with include()
#
Specifically, in a .fan
file, a call to include(FILE)
Finds and loads
FILE
(typically in the same location as the including file)Executes the code in
FILE
Parses and adds the grammar in
FILE
Parses and adds the constraints in
FILE
.
The include()
function allows for incremental refinement of Fandango specifications - you can create some base base.fan
spec, and then have more specialized specifications that alter grammar rules, add more constraints, or refine the code.
Incremental Refinement#
Let us assume you have a base spec for a particular format, say, base.fan
.
Then, in a refined spec (say, refined.fan
) that includes base.fan
, you can
override grammar definitions, by redefining rules;
override function and constant definitions, by redefining them; and
add additional constraints.
As an example, consider our persons.fan
definition of a name database.
We can create a more specialized version persons50.fan
by including persons.fan
and adding a constraint:
include('persons.fan')
where int(<age>) < 50
Likewise, we can create a specialized version persons-faker.fan
that uses fakers by overriding the <first_name>
and <last_name>
definitions:
from faker import Faker
fake = Faker()
include('persons.fan')
<first_name> ::= <name> := fake.first_name()
<last_name> ::= <name> := fake.last_name()
The include mechanism thus allows us to split responsibilities across multiple files:
We can have one spec #1 with basic definitions of individual elements
We can have a spec #2 that uses (includes) these basic definitions from spec #1 to define a syntax
We can have a spec #3 that refines spec #2 to define a specific format for a particular program or device
We can have a spec #4 that refines spec #3 towards a particular testing goal.
These mechanisms are akin to inheritance and specialization in object-oriented programming.
Tip
Generally, Fandango will warn about unused symbols, but not in an included .fan
file.
Crafting a Library#
If you create multiple specifications, you may wonder where best to store them. The rules for where Fandango searches for included files are complex, but they boil down to two simple rules:
Tip
Store your included Fandango specs either
in the directory where the including specs are, or
in
$HOME/.local/share/fandango
(or$HOME/Library/Fandango
on a Mac).
include()
vs. import
#
Python provides its own import mechanism for referring to existing features. In general, you should use
import
whenever you want to make use of Python functions; andinclude()
only if you want to make use of Fandango features.
Warning
Using include
for pure Python code, as in include('code.py')
is not recommended.
Most importantly, the current Fandango implementation will process âincludedâ Python code only after all code in the âincludingâ spec has been run. In contrast, the effects of import
are immediate.