6.2 @ Reader Internals
6.2.1 Using the @ Reader
You can use the reader via Racket’s #reader form:
#reader scribble/reader @foo{This is free-form text!}
or use the at-exp meta-language as described in Adding @-expressions to a Language.
Note that the Scribble reader reads @-forms as S-expressions. This means that it is up to you to give meanings for these expressions in the usual way: use Racket functions, define your functions, or require functions. For example, typing the above into racket is likely going to produce a “reference to undefined identifier” error, unless foo is defined. You can use string-append instead, or you can define foo as a function (with variable arity).
A common use of the Scribble @-reader is when using Scribble as a documentation system for producing manuals. In this case, the manual text is likely to start with
which installs the @ reader starting in “text mode,” wraps the file content afterward into a Racket module where many useful Racket and documentation related functions are available, and parses the body into a document using scribble/decode. See Document Reader for more information.
Another way to use the reader is to use the use-at-readtable function to switch the current readtable to a readtable that parses @-forms. You can do this in a single command line:
racket -ile scribble/reader "(use-at-readtable)"
6.2.2 Syntax Properties
The Scribble reader attaches properties to syntax objects. These properties might be useful in some rare situations.
Forms that Scribble reads are marked with a 'scribble property, and a value of a list of three elements: the first is 'form, the second is the number of items that were read from the datum part, and the third is the number of items in the body part (strings, sub-forms, and escapes). In both cases, a 0 means an empty datum/body part, and #f means that the corresponding part was omitted. If the form has neither parts, the property is not attached to the result. This property can be used to give different meanings to expressions from the datum and the body parts, for example, implicitly quoted keywords:
(define-syntax (foo stx) (let ([p (syntax-property stx 'scribble)]) (printf ">>> ~s\n" (syntax->datum stx)) (syntax-case stx () [(_ x ...) (and (pair? p) (eq? (car p) 'form) (even? (cadr p))) (let loop ([n (/ (cadr p) 2)] [as '()] [xs (syntax->list #'(x ...))]) (if (zero? n) (with-syntax ([attrs (reverse as)] [(x ...) xs]) #'(list 'foo `attrs x ...)) (loop (sub1 n) (cons (with-syntax ([key (car xs)] [val (cadr xs)]) #'(key ,val)) as) (cddr xs))))])))
> @foo[x 1 y (* 2 3)]{blah} >>> (foo x 1 y (* 2 3) "blah")
'(foo ((x 1) (y 6)) "blah")
In addition, the Scribble parser uses syntax properties to mark syntax
items that are not physically in the original source —
(define-syntax (verb stx) (syntax-case stx () [(_ cmd item ...) #`(cmd #,@(let loop ([items (syntax->list #'(item ...))]) (if (null? items) '() (let* ([fst (car items)] [prop (syntax-property fst 'scribble)] [rst (loop (cdr items))]) (cond [(eq? prop 'indentation) rst] [(not (and (pair? prop) (eq? (car prop) 'newline))) (cons fst rst)] [else (cons (datum->syntax-object fst (cadr prop) fst) rst)])))))]))
> @verb[string-append]{ foo bar } "foo\n bar"
6.2.3 Adding @-expressions to a Language
#lang at-exp | package: at-exp-lib |
For example, #lang at-exp racket/base adds @-reader support to racket/base, so that
#lang at-exp racket/base (define (greet who) @string-append{Hello, @|who|.}) (greet "friend")
reports "Hello, friend.".
In addition to configuring the reader for a module body, at-exp attaches a run-time configuration annotation to the module, so that it if it used as the main module, the current-read-interaction parameter is adjusted to use the @-reader readtable extension.
Changed in version 1.2 of package at-exp-lib: Added current-read-interaction run-time configuration.
6.2.4 Interface
(require scribble/reader) | package: at-exp-lib |
procedure
in : input-port? = (current-input-port)
procedure
(read-syntax [source-name in]) → (or/c syntax? eof-object?)
source-name : any/c = (object-name in) in : input-port? = (current-input-port)
(make-at-readtable #:command-readtable 'dynamic #:datum-readtable 'dynamic)
Changed in version 1.1 of package at-exp-lib: Changed to use 'dynamic for the command and datum readtables.
procedure
(read-inside [in]) → any
in : input-port? = (current-input-port)
procedure
(read-syntax-inside [ source-name in #:command-char command-char]) → (or/c syntax? eof-object?) source-name : any/c = (object-name in) in : input-port? = (current-input-port) command-char : char? = #\@
The given command-char is used to customize the readtable used by the reader, effectively passing it along to make-at-readtable.
Changed in version 1.1 of package at-exp-lib: Changed to use 'dynamic for the command and datum readtables.
procedure
(make-at-readtable [ #:readtable readtable #:command-char command-char #:command-readtable command-readtable #:datum-readtable datum-readtable #:syntax-post-processor syntax-post-proc]) → readtable? readtable : readtable? = (current-readtable) command-char : char? = #\@ command-readtable : (or/c readtable? 'dynamic) = readtable
datum-readtable :
(or/c readtable? boolean? (readtable? . -> . readtable?) 'dynamic) = #t syntax-post-proc : (syntax? . -> . syntax?) = values
readtable —
a readtable to base the @-readtable on. command-char —
the character used for @-forms. command-readtable —
determines the readtable that is extended for reading the command part of an @-form: a readtable —
extended to make | a delimiter instead of a symbol-quoting character 'dynamic —
extends (current-readtable) at the point where a command is parsed to make | a delimiter
datum-readtable —
the readtable used for reading the datum part of an @-form: #t —
uses the constructed @-readtable itself a readtable —
uses the given readtable a readtable-to-readtable function —
called to construct a readtable from the generated @-readtable 'dynamic —
uses (current-readtable) at the point where the datum part is parsed
The idea is that you may want to have completely different uses for the datum part, for example, introducing a convenient key=val syntax for attributes.
syntax-post-proc —
function that is applied on each resulting syntax value after it has been parsed (but before it is wrapped quoting punctuations). You can use this to further control uses of @-forms, for example, making the command be the head of a list: (use-at-readtable #:syntax-post-processor (lambda (stx) (syntax-case stx () [(cmd rest ...) #'(list 'cmd rest ...)] [else (error "@ forms must have a body")])))
Changed in version 1.1 of package at-exp-lib: Added #:command-readtable and the 'dynamic option for #:datum-readtable.
procedure
(make-at-reader #:syntax? syntax? #:inside? inside? ...) → procedure? syntax? : #t inside? : #f
The resulting function has a different contract and action based on these inputs. The expected inputs are as in read or read-syntax depending on syntax?; the function will read a single expression or, if inside? is true, the whole input; it will return a syntactic list of expressions rather than a single one in this case.
Note that syntax? defaults to #t, as this is the more expected common case when you’re dealing with concrete-syntax reading.
procedure
(use-at-readtable ...) → void?
This is mostly useful for playing with the Scribble syntax on the REPL.