In miniKanren, succeed
can be defined as (define succeed (== #t #t))
, and fail
can be defined as (define fail (=== #t #f))
. But what about #s
and #u
as short forms of succeed
and fail
, as they appear in The Reasoned Schemer?
(define #s succeed)
produces an error in Racket:
Welcome to Racket v7.2.
> (require Racket-miniKanren/miniKanren/mk)
> (define #s succeed)
; readline-input:2:8: read-syntax: expected `(`, `[`, or `{` after `#s` [,bt
; for context]
#<procedure:...iniKanren/mk.rkt:337:4>
; readline-input:2:18: read-syntax: unexpected `)` [,bt for context]
I have the feeling that this has something to do with reader macros.
How can I define #s
for succeed
and #u
for fail
in Scheme and also in Racket?
I am using the canonical miniKanren implementation for Scheme and the canonical miniKanren implementation for Racket.
Identifiers in Racket can not begin with
#
. It is simple to bind the identifierss
andu
. Redefining the meaning of#s
and#u
is not as simple, since it needs to happen in the reader. Normally#something
signals to reader that something special is to be read. The input(foo bar)
will be read as a list,#(foo bar)
will be read as a vector, and#s(foo bar)
will be read as a structure. You can read about the standard syntax here:https://docs.racket-lang.org/reference/reader.html?q=%23s#%28mod-path._reader%29
Now if you want to change the meaning of
#s
and#u
you need to look at readtables. Each time the reader sees an#
it consults a readtable to see how to handle the following characters. Since reading happens before parsing/expansion and evaluation, you can't change the reader simply by calling a function in your program. You will need to either use the#reader
extension mechanism or create your own language.For more on readtables: https://docs.racket-lang.org/reference/readtables.html?q=reader-macro
The Guide has an example of how to use reader extensions: https://docs.racket-lang.org/guide/hash-reader.html