fea-rs 0.19.0

Tools for working with Adobe OpenType Feature files.
Documentation
# fea-rs

[![Crates.io](https://img.shields.io/crates/v/fea-rs?style=flat-square)](https://crates.io/crates/fea-rs)
[![docs.rs](https://img.shields.io/docsrs/fea-rs?style=flat-square)](https://docs.rs/fea-rs)
[![License](https://img.shields.io/badge/license-Apache%202.0-blue?style=flat-square)](LICENSE-APACHE)
[![License](https://img.shields.io/badge/license-MIT-blue?style=flat-square)](LICENSE-MIT)

Parsing and compiling [Adobe OpenType feature][spec] files.

**status**: We should be able to compile most inputs. Some generated tables
may not be optimal, but should be correct. Obscure syntax may not be supported.
Please report bugs.

## quickstart

To use this tool to compile OpenType features from a UFO file:

```sh
$ cargo run PATH_TO_FONT.ufo -o my_generated_font.ttf
```

Alternatively, you can provide a path to a FEA file directly, in which case you
will also need to pass a path to a file containing the glyph order: one glyph
per line, in utf-8.

```sh
$ cargo run features.fea --glyph-order glyph_order.txt -o my_font.ttf
```

## testing

This crate uses a number of testing strategies, although all the tests can be
run with `cargo test`.

In addition to unit tests, we have a custom property-testing system for testing
parsing and compilation. This involves data that is stored in the `test-data`
directory. These tests work by taking some input, producing some output, and
then comparing that output to an expected output that is included in the test
data. There are three different formats for this:

- *parse tests*: These ensure that we generate the expected AST or errors for
  various inputs. See [`test-data/parse-tests/README.md`][parse readme] for an
  overview.
- *compile tests*: These ensure that we generate the expected [TTX output][ttx]
  or errors for a various inputs. See [`test-data/compile-tests/README.md`][compile readme] for an overview.
- *fonttools tests*: Very similar to the compile tests, but instead of being cases
  that we define ourselves, we reuse the property tests from [fonttools
  feaLib][feaLib tests]. This ensures that we generate equivalent output to
  feaLib.

## architecture sketch

The overall design of this crate is heavily inspired by the design of [rust
analyzer]. At a very high level, given some source, we:
- *lex* the source file into raw tokens ([src/parse/lexer.rs][lexer-src])
- *parse* these tokens into an abstract syntax tree ([src/parse/parser.rs][parse-src])
- *validate* this tree, ensuring that it conforms with the specification
  ([src/compile/validate.rs][validate-src])
- *compile* the validated tree into OpenType tables
  ([src/compile/compile_ctx.rs][compile-src])

### Parsing

Parsing is broken up into a lexing and then a parsing step. Lexing identifies
tokens (numbers, strings, identifiers, symbols, keywords) but has no knowledge
of the syntax of the FEA language. Parsing takes a stream of tokens, and builds
a syntax tree out of them.

The parser is "error recovering": when parsing fails, the parser skips tokens
until it finds something that might begin a valid statement in the current
context, and then tries again. Errors are collected, and reported at the end.

### AST

The AST design is adapted from the [AST in rowan][rowan ast], part of rust
analyzer. The basic idea is that when constructing an AST node, we ensure that
certain things are true about that node's contents. We can then use the type of
that node (assigned by us) to cast that node into a concrete type which knows
how to interpret that node's contents.

### Validation

After building the AST, we perform a validation pass. This checks that
statements in the tree comply with the spec: for instance, it checks if a
referenced name exists, or if a given statement is allowed in a particular table
block. Validation is also 'error recovering'; if something fails to validate we
will continue to check the rest of the tree, and report all errors at the end.

If validation succeeds, then compilation should always succeed.

### Compilation

After validation, we do a final compilation pass, which walks the tree and
assembles the various tables and lookups. This uses [fontations][] to generate
tables, which can then be added to a font.

Some general design concepts:
- in a given 'stage', collect errors as they are encountered and report them at
  the end. For instance during parsing we will continue parsing after an error
  has occurred, and report all parse errors once parsing is complete.


[spec]: http://adobe-type-tools.github.io/afdko/OpenTypeFeatureFileSpecification.html
[rust analyzer]: https://github.com/rust-analyzer/rust-analyzer/
[rowan ast]: https://github.com/rust-analyzer/rust-analyzer/blob/master/docs/dev/syntax.md#ast
[fontations]: https://github.com/googlefonts/fontations
[ttx]: https://fonttools.readthedocs.io/en/latest/ttx.html
[feaLib tests]: https://github.com/fonttools/fonttools/tree/main/Tests/feaLib/data
[parse readme]: ./fea-rs/test-data/parse-tests/README.md
[compile readme]: ./fea-rs/test-data/compile-tests/README.md
[lexer-src]: ./fea-rs/src/parse/lexer.rs
[parse-src]: ./fea-rs/src/parse/parser.rs
[validate-src]: ./fea-rs/src/compile/validate.rs
[compile-src]: ./fea-rs/src/compile/compile_ctx.rs.rs