On this page:
Syntax
#%quotes
syntax_  is_  integer
syntax_  is_  boolean
syntax_  is_  symbol
syntax_  is_  string
syntax_  is_  operator
syntax_  is_  keyword
syntax_  to_  integer
syntax_  to_  boolean
syntax_  to_  symbol
syntax_  to_  string
syntax_  operator_  to_  string
syntax_  keyword_  to_  string
integer_  to_  syntax
boolean_  to_  syntax
symbol_  to_  syntax
string_  to_  syntax
string_  to_  operator_  syntax
string_  to_  keyword_  syntax
syntax_  is_  list
syntax_  is_  parens
syntax_  is_  braces
syntax_  is_  quotes
syntax_  is_  block
syntax_  is_  alts
syntax_  to_  list
syntax_  parens_  to_  list
syntax_  braces_  to_  list
syntax_  quotes_  to_  list
syntax_  block_  to_  list
syntax_  alts_  to_  list
syntax_  group_  to_  list
syntax_  groups_  to_  list
list_  to_  syntax
list_  to_  parens_  syntax
list_  to_  braces_  syntax
list_  to_  quotes_  syntax
list_  to_  block_  syntax
list_  to_  alts_  syntax
list_  to_  group_  syntax
list_  to_  groups_  syntax
syntax_  split
syntax_  join
syntax_  read
syntax_  generate_  temporary

8 Syntax Objects, Patterns, and Templates🔗ℹ

See Syntax Objects for an overview of syntax objects.

A syntax object is a representation of source code. It is written as quoted term using single quotes, such as '1 + * 4 f()'. A syntax object is not a string, even though it uses quote marks. Instead, the usual shrubbery rules apply inside quotes, and shrubbery structure is preserved in a syntax object. That is, a string records a sequence of characters, but a syntax object represents structure that might be written in different ways; in particular, the printed form of a syntax object tends to use «» shrubbery notation instead of shrubbery notation’s whitespace-sensitive format.

When a syntax object is written using quotes, the #%quotes form is used implicitly, similar to the way that square brackets implicitly use #%brackets.

> 'apple'

- Syntax

'apple'

> #%quotes 'apple'

- Syntax

'apple'

> '1 + 2'

- Syntax

'1 + 2'

> 'fun (x):

     x + 1'

- Syntax

'fun (x): x + 1'

An expression written with quotes is more generally a template, because it can include $ as an escape and an ellipsis (written as ...) to instantiate a repetition. Syntax patterns in match are also written with quotes, $ as an escape, and an ellipsis to match a repetition.

Rules for $ and ... in patterns:

When $ appears in a pattern, then it must be followed by either (1) an identifier, or (2) parentheses containing an identifier followed by :: Identifier.

In the simple case that the identifier is not annotated with :: Identifier and the pattern has no ellipses (written as ...), then the identifier is bound to a non-empty sequence of terms from the corresponding part of the input syntax object. Sequences are matched greedily, meaning that as many terms are matched to an identifier as possible to find a match for the pattern overall.

> match '1 2 3 4'

  | '1 $x 4': x

- Syntax

'2 3'

> match 'a b c d'

  | '$x $y': [x, y]

- Listof(Syntax)

['a b c', 'd']

An identifier annotated with :: Identifier after $ can only match an individual identifier in the input.

> match 'a b c d'

  | '$(x :: Identifier) $y': [x, y]

- Listof(Syntax)

['a', 'b c d']

> match '1 2'

  | '$(x :: Identifier) $y': [x, y]

  | '$z': [z]

- Listof(Syntax)

['1 2']

When an ellipsis ... appears in a pattern, then it greedily matches 0 or more repetitions of the preceding pattern element. If the preceding element contains a $ escape, then the escaped identifier is not bound to a single syntax object, but it is instead bound as a repetition that holds each matched term. A repetition can only be referenced through a corresponding escape in a template. Ellipses can be nested in a pattern, and a repetition must be used in a template with the same amount of nesting as in its pattern.

> match 'a a a b d'

  | 'a ... b c ... d': "matches"

- String

"matches"

> match 'a a a b'

  | '$x ... b': 'matches as $x ...'

- Syntax

'matches as a a a'

> match '(a 1) (b 2) (c 3)'

  | '($x $y) ...': ['$x ...', '$y ...']

- Listof(Syntax)

['a b c', '1 2 3']

> match '(a: 1) (b: 2 3 4) (c: 5 6)'

  | '($x: $y ...) ...': ['$x ...', '($y ...) ...']

- Listof(Syntax)

['a b c', '(1) (2 3 4) (5 6)']

When an ellipsis appears by itself in a shrubbery group, then the pattern matches 0 or more repetitions of the preceding group. If the group is an alternative written with |, then the pattern matches 0 or more alternatives.

> match 'a

         b c

         d'

  | '$x

     ...':

      '{$x, ...}'

- Syntax

'{a, b c, d}'

> match 'cases

         | a

         | b c

         | d'

  | 'cases

     | $x

     | ...':

      '{$x, ...}'

- Syntax

'{a, b c, d}'

Rules for $ and ... in templates:

When $ appears in a template, then it must be followed by an identifier or an expression that is written as a single term (e.g., a parenthesized expression).

If an identifier is provided, then it can refer to a repetition that is bound by a syntax-pattern match, as long as the $-escaped identifier in the template is under a number of ellipses that match the repetition binding. Each element of the repetition is put in the result syntax object in place of the escape. The rules for patterns above show several example templates that use repetitions.

If an escape does not refer to a repetition, then it must have an expression that produces a syntax object, and it must not be under any ellipses. The syntax object replaces the escape in the result syntax object.

> '1 $(if #true | '2' | 'oops') 3'

- Syntax

'1 2 3'

type

Syntax

The type for expressions that produce syntax objects.

expression

#%quotes 'term ...

          ...'

Produces a syntax object, quoting the terms literally instead of treating them as subexpressions. Usually, #%quotes is omitted, since it is implied by using quotes as an expression form.

See above for information about $ escapes within the quotes for a syntax object.

function

fun syntax_is_integer(stx :: Syntax) :: Boolean

 

function

fun syntax_is_boolean(stx :: Syntax) :: Boolean

 

function

fun syntax_is_symbol(stx :: Syntax) :: Boolean

 

function

fun syntax_is_string(stx :: Syntax) :: Boolean

 

function

fun syntax_is_operator(stx :: Syntax) :: Boolean

 

function

fun syntax_is_keyword(stx :: Syntax) :: Boolean

The syntax_is_integer function checks whether a syntax object has a single term representing a number, returning #true if so and #false otherwise. The syntax_is_boolean, syntax_is_symbol, syntax_is_string, and syntax_is_operator functions similarly check for different kinds of primitive values as representations.

> syntax_is_integer('9')

- Boolean

#true

> syntax_is_integer('apple')

- Boolean

#false

> syntax_is_keyword('~apple')

- Boolean

#true

function

fun syntax_to_integer(stx :: Syntax) :: Int

 

function

fun syntax_to_boolean(stx :: Syntax) :: Boolean

 

function

fun syntax_to_symbol(stx :: Syntax) :: Symbol

 

function

fun syntax_to_string(stx :: Syntax) :: String

 

function

fun syntax_operator_to_string(stx :: Syntax) :: String

 

function

fun syntax_keyword_to_string(stx :: Syntax) :: String

The syntax_to_integer function extracts the number that a syntax object represents, but only if syntax_is_integer would return #true; otherwise an exception is raised. The functions syntax_to_boolean, syntax_to_symbol, and syntax_to_string similarly extract values from syntax representations. The syntax_operator_to_string and syntax_keyword_to_string functions encodes an operator or keyword as a string, respectively.

> syntax_to_integer('9')

- Int

9

> syntax_operator_to_string('+')

- String

"+"

function

fun integer_to_syntax(n :: Int) :: Syntax

 

function

fun boolean_to_syntax(bool :: Boolean) :: Syntax

 

function

fun symbol_to_syntax(sym :: Symbol) :: Syntax

 

function

fun string_to_syntax(str :: String) :: Syntax

 

function

fun string_to_operator_syntax(str :: String) :: Syntax

 

function

fun string_to_keyword_syntax(str :: String) :: Syntax

The inverse of syntax_to_integer, etc., converting a value into a syntax representation.

> integer_to_syntax(9)

- Syntax

'9'

function

fun syntax_is_list(stx :: Syntax) :: Boolean

 

function

fun syntax_is_parens(stx :: Syntax) :: Boolean

 

function

fun syntax_is_braces(stx :: Syntax) :: Boolean

 

function

fun syntax_is_quotes(stx :: Syntax) :: Boolean

 

function

fun syntax_is_block(stx :: Syntax) :: Boolean

 

function

fun syntax_is_alts(stx :: Syntax) :: Boolean

The syntax_is_list function checks whether a syntax object would match the pattern '[$elem, ...]', returning #true if so and #false otherwise. Similarly, the syntax_is_parens function checks whether a syntax object would match the pattern '($elem, ...)', the syntax_is_braces function checks whether a syntax object would match the pattern '{$elem, ...}', the syntax_is_quotes function checks whether a syntax object would match the pattern ''$elem; ...'', the syntax_is_block function checks whether a syntax object would match the pattern ': $elem; ...', and the syntax_is_alts function checks whether a syntax object would match the pattern ' | $elem; ... | ...'.

> syntax_is_list('[w, x y, z]')

- Boolean

#true

> syntax_is_list('a b c')

- Boolean

#false

> syntax_is_parens('(w, x y, z)')

- Boolean

#true

> syntax_is_block(':

                     w

                     x y')

- Boolean

#true

function

fun syntax_to_list(stx :: Syntax) :: Listof(Syntax)

 

function

fun syntax_parens_to_list(stx :: Syntax) :: Listof(Syntax)

 

function

fun syntax_braces_to_list(stx :: Syntax) :: Listof(Syntax)

 

function

fun syntax_quotes_to_list(stx :: Syntax) :: Listof(Syntax)

 

function

fun syntax_block_to_list(stx :: Syntax) :: Listof(Syntax)

 

function

fun syntax_alts_to_list(stx :: Syntax) :: Listof(Syntax)

 

function

fun syntax_group_to_list(stx :: Syntax) :: Listof(Syntax)

 

function

fun syntax_groups_to_list(stx :: Syntax) :: Listof(Syntax)

The syntax_to_list function works on a syntax object that would match '[$elem, ...]', and it returns the list of elem matches. The syntax_parens_to_list, syntax_braces_to_list, syntax_quotes_to_list, syntax_block_to_list, and syntax_alts_to_list functions are similar. The result of syntax_alts_to_list is a list of block syntax.

The syntax_group_to_list function works on a single-group syntax object, and it returns a list of single-term syntax objects. The syntax_groups_to_list function works on any syntax object, and it returns a list of single-group syntax objects.

> syntax_to_list('[w, x y, z]')

- Listof(Syntax)

['w', 'x y', 'z']

> syntax_parens_to_list('(w, x y, z)')

- Listof(Syntax)

['w', 'x y', 'z']

> syntax_alts_to_list('| w | x; y | z')

- Listof(Syntax)

[

  ': w',

  ':

     x

     y',

  ': z'

]

> syntax_groups_to_list('w; x y; z')

- Listof(Syntax)

['w', 'x y', 'z']

function

fun list_to_syntax(lst :: Listof(Syntax)) :: Syntax

 

function

fun list_to_parens_syntax(lst :: Listof(Syntax)) :: Syntax

 

function

fun list_to_braces_syntax(lst :: Listof(Syntax)) :: Syntax

 

function

fun list_to_quotes_syntax(lst :: Listof(Syntax)) :: Syntax

 

function

fun list_to_block_syntax(lst :: Listof(Syntax)) :: Syntax

 

function

fun list_to_alts_syntax(lst :: Listof(Syntax)) :: Syntax

 

function

fun list_to_group_syntax(stx :: Syntax) :: Listof(Syntax)

 

function

fun list_to_groups_syntax(stx :: Syntax) :: Listof(Syntax)

The inverse of syntax_to_list, etc., converting a list of syntax objects to a syntax object.

> list_to_syntax(['w', 'x y', 'z'])

- Syntax

'[w, x y, z]'

> list_to_parens_syntax(['w', 'x y', 'z'])

- Syntax

'(w, x y, z)'

function

fun syntax_split(stx :: Syntax) :: Listof(Syntax)

 

function

fun syntax_join(stx :: Syntax) :: Listof(Syntax)

Functions for simple splitting and joining tasks. For more general and precise tasks, use match and template construction with #%quotes, or use functions like syntax_group_to_list and syntax_groups_to_list.

The syntax_split function takes a single-group syntax object and splits it into a list of term syntax objects, or it takes a multi-group syntax object and splits it into a list of group syntax objects.

The syntax_join function combines a list of term syntax objects into one group syntax object, and it splices a list of multi-group syntax objects into a multi-group syntax object.

> syntax_split('1 2 3')

- Listof(Syntax)

['1', '2', '3']

> syntax_split('1 2 3

                4 5')

- Listof(Syntax)

['1 2 3', '4 5']

> syntax_join(['1', '2', '3'])

- Syntax

'1 2 3'

> syntax_join(['1', '2 3', '4'])

- Syntax

'1

 2 3

 4'

> syntax_join(['1', '2 3', '4

                            5'])

- Syntax

'1

 2 3

 4

 5'

function

fun syntax_read() :: Syntax

Reads shrubbery input from the program’s current input and produces a syntax object. Assuming no syntax errors, input is read until terminated by an end-of-file.

Creates an identifier that is not equal to any other identifier.