On this page:
syntax_  class
syntax_  class.together
fields
root_  swap
description
error_  mode
kind
field
match_  def
match_  when
match_  unless
default
description
Term
Identifier
Operator
Name
Identifier  Name
Keyword
String
Int
Number
Boolean
Literal
Sequence
Group
Multi
Block
9.0.0.1

11.2 Syntax Classes🔗ℹ

definition

syntax_class stxclass_id_name maybe_args:

  class_clause

  ...

| pattern_case

| ...

 

maybe_args

 = 

(bind_id_maybe_kw_opt, ...)

 | 

(bind_id_maybe_kw_opt, ..., & rest_id)

 | 

ϵ

 

bind_id_maybe_kw_opt

 = 

a bind_maybe_kw_opt, but with a plain id as bind

 

class_clause

 = 

description desc_rhs

 | 

error_mode error_mode_rhs

 | 

kind kind_rhs

 | 

fields: field_decl

 | 

root_swap: field_to_root_id root_to_field_id

 

pattern_case

 = 

syntax_pattern

 | 

syntax_pattern: pattern_body; ...

 

pattern_body

 = 

field field_decl

 | 

match_def also_decl

 | 

match_when when_rhs

 | 

match_unless unless_rhs

 | 

body

Defines a syntax class stxclass_id_name that can be used in syntax patterns with ::. A syntax class can optionally have arguments; unless all arguments are have devault values, every use of the syntax class with :: must supply arguments. An bind_id_maybe_kw_opt is like a bind_maybe_kw_opt for fun, but each binding must be a plain id (i.e., annotations and general pattern matching are not supported). Identifiers bound as arguments are visible in class_clause bodies. Use syntax_class.together to define a syntax class that refers to itself or a group of mutually referential syntax classes.

Syntax forms matched by the syntax class are described by pattern_case alternatives. Each kind of class_clause alternative can be supplied at most once.

An optional description clause provides a description of the syntax class, which is used to produce clearer error messages when a term is rejected by the syntax class. The result of the block block must be a string or #false, where #false is equivalent to not specifying a description. When error_mode is declared as ~opaque, then parsing error messages will not refer to the interior details of the pattern cases; instead, messages will use the description string.

An optional kind declaration indicates the context within a pattern where a syntax class can be used, and it determines the kind of match that each pattern specifies. See kind for details. The default is inferred from the shapes for pattern_cases as either ~term, ~sequence, or ~multi.

A fields declaration limits the set of pattern variables that are accessible from the class, where variables used in all pattern_cases are otherwise available (as described next). Each identifier in fields must be a field name that would be made available. A fields declaration can also specify the repetition depth and context kind of a field, in which case pattern_cases must be consistent with the declaration; see fields for more information. A root_swap class moves the value of one of the would-be fields to the root while moving the root to a fresh field; see root_swap for more information.

The pattern_case alternatives are the main content of a syntax class. After the class name is defined, then when a variable id is bound through a syntax pattern with $(id :: name), it matches a syntax object that matches any of the pattern_cases in the definition of stx_class_id, where the pattern_cases are tried first to last. A pattern variable that is included in all of the pattern_cases is a field of the syntax class, which is accessed from a binding id using dot notation. For example, if the pattern variable is var, its value is accessed from id using id.var. When a field is a repetition, it can only be accessed statically—that is, when the id in id.var is bound as a pattern variable annotated with stxclass_id_name, or via expr.var when expr has the static information of Syntax.matched_of(stxclass_id_name). When a syntax object with fields for a syntax class is used to produce another system object (e.g., the syntax object is used in a template to make a larger syntax object), fields are not associated with any part of the new syntax object; however, a syntax object with fields preserves its fields when bound itself as a field value in another syntax class.

A pattern_case matches when

Every pattern variable in the initial syntax_pattern of a pattern_case as well as every variable in every match_def is a candidate field name, as long as it is also a candidate in all other syntax_patterns within the syntax class. In addition, names declared with field are also candidates. A field must have the same repetition depth across all pattern cases, unless it is excluded from the syntax class’s result through a fields declaration that does not list the field.

The body of a pattern_case can include other definitions and expressions. Those definitions and expressions can use pattern variables bound in the main syntax_pattern of the case as well as any preceding match_def clause or a field declared by a preceding field. Consecutive definitions and expressions within a pattern_case form a definition context, but separated sets of definitions and expressions can refer only to definitions in earlier sets, and let is allowed only in the last set of definitions and expressions.

A variable bound with a syntax class (within a syntax pattern) can be used without dot notation. The variable is bound to a syntax object corresponding to the entire match of a syntax_pattern.

> meta syntax_class Arithmetic

  | '$x + $y'

  | '$x - $y'

> expr.macro 'doubled_operands $(a :: Arithmetic)':

    '$a.x * 2 + $a.y * 2'

> doubled_operands 3 + 5

16

> expr.macro 'add_one_to_expression $(a :: Arithmetic)':

    '$a + 1'

> add_one_to_expression 2 + 2

5

> meta syntax_class NTerms

  | '~one $a':

      field b = '0'

      field average = '$(a.unwrap() / 2)'

  | '~two $a $b':

      def sum = a.unwrap() + b.unwrap()

      field average = '$(sum / 2)'

> expr.macro 'second_term $(e :: NTerms)':

    e.b

> second_term ~two 1 2

2

> second_term ~one 3

0

> expr.macro 'average $(e :: NTerms)':

    e.average

> average ~two 24 42

33

definition

syntax_class.together:

  syntax_class stxclass_decl

  ...

Declares syntax classes that can refer to themselves and each other. Unlike a syntax_class form in other contexts, each syntax_class form within syntax_class.together is required to declare its fields with fields, otherwise no fields are available from the syntax class.

syntax_class.together:

  syntax_class ModPath:

    fields: [elem, ...]

  | '$head':

      field [elem, ...]: [head]

  | '$head / $(mp :: ModPath)':

      field [elem, ...] = [head, mp.elem, ...]

> match 'a / b / c'

  | '$(mp :: ModPath)':

      [mp.elem, ...]

['a', 'b', 'c']

syntax class clause

fields spec ...

 

syntax class clause

fields: spec ...; ...

 

spec

 = 

id_maybe_rep

 | 

id_maybe_rep: kind kind_decl

 

id_maybe_rep

 = 

id_maybe_annot

 | 

[id_maybe_rep, ellipsis]

 

id_maybe_annot

 = 

id

 | 

id :: annot

 | 

id :~ annot

 

ellipsis

 = 

...

Limits the set of fields that are provided by a syntax class to the listed ids. See syntax_class. A field’s repetition depth is declared by wrapping it within [] with ... like a repetition binding. If no spec is present at all, no fields will be provided.

If an annot is supplied for a id_maybe_annot, then the field has the static information of annot, and the annotation is checked or converted when :: is used. If a field with the same name has an annotation in a pattern clause, then the one supplied in fields applies afterward, and static information is combined as with statinfo_meta.and. When not within [], a field with an annot must be in its own group (i.e., on its own line, usually).

If a kind is not declared for a field identifier, then the context kind is inferred from patterns within the syntax class. If kind is declared, then it must match the context kind that would be inferred for the field from all patterns, and an annot cannot be provided.

Adjusts the match value and fields of a syntax class. The field_to_root_id names a field that the syntax class would otherwise provide, but the field is removed, and its value instead becomes the main value of an identifier that is bound with the syntax class. The field for this first field_to_root_id must not be a repetition. Meanwhile, the matching terms that would otherwise be the variable’s value are associated instead with a fresh field named by the second root_to_field_id.

A :: Syntax annotation is implicitly added to field_to_root_id to ensure that its value is a syntax object.

syntax_class Parenthesized:

  root_swap: content group

| '($content)'

> match '(1 2 3)'

  | '$(p :: Parenthesized)':

      [p, p.group]

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

syntax class clause

description:

  body;

  ...

 

syntax class clause

description expr

Configures a syntax class’s description for error reporting. See syntax_class.

syntax class clause

error_mode: error_mode_keyword

 

syntax class clause

error_mode error_mode_keyword

 

error_mode_keyword

 = 

~opaque

 | 

~transparent

Configures the way that failures to match a syntax class are reported. See syntax_class.

syntax class clause

kind: kind_keyword

 

syntax class clause

kind kind_keyword

 

kind_keyword

 = 

~term

 | 

~sequence

 | 

~group

 | 

~multi

 | 

~block

Provided both normally and as meta.

Determines the contexts where a syntax class can be used and the kinds of matches that it produces:

  • ~term: each pattern case represents a single term, and the syntax class can be used in the same way at the Term syntax class.

  • ~sequence: each pattern case represents a sequence of terms that is spliced within a group. This is the default mode of a syntax class when no kind is specified.

  • ~group: each pattern case represents a group, and the syntax class can be used in the same places as Group (i.e., at the end of an enclosing group).

  • ~multi: each pattern case represents multiple groups, and the syntax class can be used in the same way at the Multi syntax class.

  • ~block: each pattern case represents a block, and the syntax class can be used in the same way at the Block syntax class

With ~term, each pattern case must match only a single term, and with ~block, each pattern case must be a block pattern.

See also syntax_class.

pattern clause

field id_maybe_rep:

  body

  ...

 

pattern clause

field id_maybe_rep = expr

 

id_maybe_rep

 = 

id_maybe_annot

 | 

[id_maybe_rep, ellipsis]

 

id_maybe_annot

 = 

id

 | 

id :: annot

 | 

id :~ annot

 

ellipsis

 = 

...

Similar to def, but restricted to defining a plain identifier or a simple list repetition within a syntax_class pattern case, and adds a field (or, at least, a candidate field) to the pattern case.

The result of the right-hand body sequence or expr is not required to be a syntax object or have syntax objects in nested lists. If the field is referenced so that it’s value is included in a syntax template, a non-syntax value is converted to syntax at that point. Otherwise, the field can be used directly to access non-syntax values.

If an annot is supplied in id_maybe_annot, then the field has the static information of annot, and the annotation is checked or converted when :: is used. Annotations from different pattern clauses are combined using statinfo_meta.or to determine static information for accesses of the field.

See also syntax_class.

pattern clause

match_def 'pattern':

  body

  ...

 

pattern clause

match_def 'pattern' = expr

Constrains a pattern case to match only when the right-hand body or expr matches pattern, and adds the pattern variables of pattern to the set of (candidate) syntax class fields.

See syntax_class.

pattern clause

match_when:

  body

  ...

 

pattern clause

match_when expr

 

pattern clause

match_unless:

  body

  ...

 

pattern clause

match_unless expr

Constrains a pattern case to match only when a body or expr produces a true value in the case of match_when or a false value in the case of match_unless.

See syntax_class.

pattern clause

default id_maybe_rep:

  body

  ...

 

pattern clause

default id_maybe_rep = expr

 

pattern clause

description string

 

pattern clause

description: string

Pattern clauses that are allowed only within group_option_sequence and term_option_sequence.

syntax class

syntax_class Term: kind: ~term

 

syntax class

syntax_class Identifier: kind: ~term

 

syntax class

syntax_class Operator: kind: ~term

 

syntax class

syntax_class Name: kind: ~sequence

 

syntax class

syntax_class IdentifierName: kind: ~sequence

 

syntax class

syntax_class Keyword: kind: ~term

 

syntax class

syntax_class String: kind: ~term

 

syntax class

syntax_class Int: kind: ~term

 

syntax class

syntax_class Number: kind: ~term

 

syntax class

syntax_class Boolean: kind: ~term

 

syntax class

syntax_class Literal: kind: ~term

 

syntax class

syntax_class Sequence: kind: ~sequence

 

syntax class

syntax_class Group: kind: ~group

 

syntax class

syntax_class Multi: kind: ~multi

 

syntax class

syntax_class Block: kind: ~block

Provided both normally and as meta.

Syntax classes, all of which imply a single-term match except for Sequence, Group, Multi, Block, and IdentifierName.

The Sequence syntax class matches any number of terms, including zero terms. That is, it matches a group or a multi-group sequence that has zero groups.

The Group syntax class can be used only for a pattern identifier that has no other pattern escapes after it in the same group. The identifier is bound to a match for a non-empty sequence of term syntax objects.

The Multi syntax class can be used only for a pattern identifier that is alone within its group. The identifier is bound to a match for a sequence of groups.

The Block syntax class can be used only for a pattern escape that is along within a block. The identifier is bound to a match for the entire block as a single term (i.e., as a single-term syntax object that has a block term, and not as a multi-group syntax object).

The Name syntax class is an extension of the union of Identifier and Operator that matches dotted sequences like an op_or_id_name form.

The IdentifierName syntax class is an extension of Identifier that matches dotted sequences like an id_name form.

The Literal syntax class matches a term that is not an identifier, operator, block, or alternatives.