Skip to content

Commit f53ad39

Browse files
committedMar 6, 2025
refine and document Racket-Rhombus interoperability
1 parent 0a55db7 commit f53ad39

File tree

18 files changed

+392
-6
lines changed

18 files changed

+392
-6
lines changed
 

‎rhombus-lib/info.rkt

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,4 +17,4 @@
1717
(define license '(Apache-2.0 OR MIT))
1818

1919
;; keep in sync with runtime version at "rhombus/private/amalgam/version.rkt"
20-
(define version "0.33")
20+
(define version "0.34")

‎rhombus-lib/rhombus/dot.rkt

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
#lang racket/base
2+
(require (submod "private/amalgam.rkt" dot))
3+
4+
(provide dynamic-dot-ref
5+
dynamic-dot-set!)
6+
7+
(define (dynamic-dot-ref obj name)
8+
(unless (symbol? name)
9+
(raise-argument-error 'dynamic-dot-ref "Symbol" name))
10+
(dot-lookup-by-name obj name))
11+
12+
(define (dynamic-dot-set! obj name val)
13+
(unless (symbol? name)
14+
(raise-argument-error 'dynamic-dot-set! "Symbol" name))
15+
(dot-assign-by-name obj name val))

‎rhombus-lib/rhombus/parse.rkt

+8-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,11 @@
11
#lang racket/base
2+
(require (rename-in (submod "private/amalgam.rkt" parse)
3+
[rhombus-expression parse:rhombus-expression]))
24

3-
(require (submod "private/amalgam.rkt" parse))
5+
(provide rhombus-expression
6+
rhombus-top)
47

5-
(provide (all-from-out (submod "private/amalgam.rkt" parse)))
8+
(define-syntax-rule (rhombus-expression e)
9+
;; an expression may expand to `(begin (quote-syntax statinfo) expr)`,
10+
;; so make sure that's not spliced in a definition context
11+
(let () (parse:rhombus-expression e)))

‎rhombus-lib/rhombus/private/amalgam-src.rkt

+3
Original file line numberDiff line numberDiff line change
@@ -120,3 +120,6 @@
120120
(module+ evt
121121
(bounce "amalgam/evt.rkt")
122122
(bounce "amalgam/evt.rhm"))
123+
124+
(module+ dot
125+
(bounce (submod "amalgam/dot.rkt" for-external)))

‎rhombus-lib/rhombus/private/amalgam/dot.rkt

+4
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,10 @@
4747
parse-dot-expr
4848
parse-dot-repet)))
4949

50+
(module+ for-external
51+
(provide dot-lookup-by-name
52+
dot-assign-by-name))
53+
5054
(begin-for-syntax
5155
(property dot-provider (handler))
5256

‎rhombus-lib/rhombus/private/amalgam/version.rkt

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,4 +3,4 @@
33
(provide version)
44

55
;; keep in sync with package version at "../../../info.rkt"
6-
(define version "0.33")
6+
(define version "0.34")

‎rhombus-scribble-lib/rhombus/scribble/private/typeset-example.rkt

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,8 @@
33
syntax/parse/pre)
44
racket/string
55
racket/syntax-srcloc
6-
rhombus/parse
6+
(only-in rhombus/parse
7+
rhombus-expression)
78
racket/sandbox
89
file/convertible
910
racket/port

‎rhombus/rhombus/scribblings/info.rkt

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
#lang info
22

33
(define scribblings '(("rhombus.scrbl" (multi-page) (language))
4-
("reference/rhombus-reference.scrbl" (multi-page) (language))))
4+
("reference/rhombus-reference.scrbl" (multi-page) (language))
5+
("rhombus-racket/rhombus-racket.scrbl" (multi-page) (interop))))
56

67
(define test-omit-paths 'all)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
#lang rhombus/scribble/manual
2+
@(import:
3+
"racket_names.rkt" open
4+
meta_label:
5+
rhombus open)
6+
7+
@(def rhombus_doc = ModulePath'lib("rhombus/scribblings/reference/rhombus-reference.scrbl")')
8+
@(def racket_doc = ModulePath'lib("scribblings/reference/reference.scrbl")')
9+
10+
@title(~tag: "data"){Common Run-Time Representations}
11+
12+
Rhombus and Racket share the same or related representations for many
13+
basic forms of data:
14+
15+
@itemlist(
16+
17+
@item{Booleans are the same.}
18+
19+
@item{Numbers are the same, including @tech(~doc: rhombus_doc){fixnums}
20+
and @tech(~doc: rhombus_doc){flonums}.}
21+
22+
@item{Characters, strings, and byte strings are the same. Beware,
23+
however, that Racket strings are mutable by default, which satisfies
24+
Rhombus's @rhombus(ReadableString, ~annot) but not
25+
@rhombus(String, ~annot).}
26+
27+
@item{Symbols and keywords. A keyword is written in shrubbery notation
28+
for Rhombus with a @litchar{~} prefix, and it is written in S-expression
29+
notation with a @litchar{#:} prefix, but the representation does not
30+
include that prefix.}
31+
32+
@item{Functions, including functions with keyword arguments are the
33+
same. Note that a Racket keyword with non-alphabetic characters can be
34+
written in rhombus using @litchar("#{")…@litchar("}") notation, as in
35+
@rhombus(work(#{#:fast?}: #true)).}
36+
37+
@item{Pairs are the same. Racket @tech(~doc: racket_doc){lists} are
38+
Rhombus @tech(~doc: rhombus_doc){pair lists}, Rhombus
39+
@tech(~doc: rhombus_doc){lists} correspond to
40+
@racketmod_racket_treelist, and Rhombus @tech(~doc: rhombus_doc){mutable
41+
lists} correspond to @racketmod_racket_mutable_treelist.}
42+
43+
@item{Racket @tech(~doc: racket_doc){vectors} and Rhombus
44+
@tech(~doc: rhombus_doc){arrays} are the same, and boxes are the same in
45+
both.}
46+
47+
@item{Racket @tech(~doc: racket_doc){hash tables} and Rhombus
48+
@tech(~doc: rhombus_doc){maps} are the same. Rhombus
49+
@tech(~doc: rhombus_doc){sets} are unrelated to any Racket data
50+
structure, although the implementation internally uses Racket hash
51+
tables.}
52+
53+
@item{Racket @tech(~doc: racket_doc){sequences} and Rhombus
54+
@tech(~doc: rhombus_doc){sequences} are the same.}
55+
56+
@item{The void value and end-of-file object are the same.}
57+
58+
@item{Rhombus and Racket syntax objects are the same. A Rhombus syntax
59+
object encodes shrubbery structure within an S-expression syntax object
60+
as described in
61+
@secref(~doc: ModulePath'lib("shrubbery/scribblings/shrubbery.scrbl")', "parsed-rep").}
62+
63+
)
64+
65+
In many other cases, the Rhombus representation of an entity is a
66+
wrapper on a Racket representation. For example, a Rhombus
67+
@rhombus(Thread, ~class) object wraps a Racket thread object. In those
68+
case, a common convention within Rhombus is to provide access to the
69+
Racket representation through a @rhombus(handle) property, and sometimes
70+
a @rhombus(from_handle) function is provided to construct a Rhombus
71+
object wrapper for a suitable Racket value.
72+
73+
Rhombus wrappers enable a dynamic @rhombus(.) operation to find methods
74+
and properties. Although dynamic @rhombus(.) works on some kinds of
75+
Racket values without a wrapper, such as strings and lists, it relies on
76+
built-in support by the @rhombus(.) operator. Rhombus has no safe and
77+
composable way to extend that built-in set.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
#lang scribble/manual
2+
@(require (for-label rhombus/dot
3+
racket/base
4+
racket/contract/base
5+
rhombus/parse))
6+
7+
@title{Dynamic Dot from Racket}
8+
9+
@defmodule[rhombus/dot]
10+
11+
The Rhombus dot operator can be used via @racket[rhombus-expression],
12+
but the @racketmodname[rhombus/dot] module provides a simpler and more
13+
convenient interface for dynamic access of object fields, methods, and
14+
properties.
15+
16+
@defproc[(dynamic-dot-ref [obj any/c] [name symbol?]) any/c]{
17+
18+
Accesses a field, method, or property @racket[name] from @racket[obj].
19+
An exception is raised if no such member exists in @racket[obj].
20+
21+
}
22+
23+
@defproc[(dynamic-dot-set! [obj any/c] [name symbol?] [val any/c]) any/c]{
24+
25+
Modifies @racket[obj] to set the field or property @racket[name] in
26+
@racket[obj] to @racket[val]. An exception is raised if no such mutable
27+
member exists in @racket[obj].
28+
29+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
#lang rhombus/scribble/manual
2+
@(import:
3+
"racket_names.rkt" open
4+
meta_label:
5+
rhombus open
6+
rhombus/meta:
7+
expose:
8+
expr_meta
9+
defn_meta
10+
decl_meta)
11+
12+
@title(~tag: "module"){Modules and Exports}
13+
14+
Rhombus and Racket modules exist in the same module system with the same
15+
filesystem- and collection-based paths, so a Rhombus module can
16+
@rhombus(import) a Racket module, and a Racket module can
17+
@racket_require a Rhombus module.
18+
19+
@itemlist(
20+
21+
@item{Relative-path module references and Rhombus @rhombus(file, ~impo)
22+
or Racket @racket_file paths work the same, where a @filepath{.rhm} or
23+
@filepath{.rkt} file suffix can be used in either language.}
24+
25+
@item{A module name in Rhombus without quotes, such as
26+
@rhombusmodname(rhombus/random), implicitly gets a @filepath{.rhm}
27+
suffix or refers to a @filepath{main.rhm} file (when there is no
28+
@litchar{/}), while a similar Racket module name like
29+
@racketmod_racket_math implicitly gets a @filepath{.rkt} suffix or
30+
refers to a @filepath{main.rkt} file.
31+
32+
To refer to a collection-based @filepath{.rkt} file from Rhombus, use
33+
the @rhombus(lib, ~impo) form, as in @rhombus(lib("racket/math.rkt")).
34+
To refer to a collection-based @filepath{.rhm} file from Racket, use the
35+
@racket_lib form as in @racketmod_rhombus_random.}
36+
37+
)
38+
39+
While importing bindings from one language into the other is relatively
40+
straightforward, not all imported bindings can be used directly:
41+
42+
@itemlist(
43+
44+
@item{A Racket binding can be used in a Rhombus expression if it
45+
corresponds to a non-macro definition or if it is an identifier macro
46+
that can be used alone as an expression.}
47+
48+
@item{A Racket binding for a syntactic form (i.e., a macro) cannot be
49+
used directly in a Rhombus expression. Even though a use of the
50+
syntactic form in Racket looks like a function call, the Rhombus
51+
function call form does not (necessarily) expand to a Racket
52+
form with a function-call shape.
53+
54+
Using a Racket syntactic form from Rhombus requires a Rhombus macro
55+
that expands to the Racket form. See @secref("racket-expr") for more
56+
information.}
57+
58+
@item{A Rhombus binding can be used in a Racket expression if it
59+
corresponds to a @rhombus(fun, ~defn) definition or a
60+
@rhombus(def, ~defn) form where the binding is just an identifier or
61+
parenthesized identifiers @rhombus(values, ~bind). Other @rhombus(def)
62+
binding forms may work, but there is no guarantee, so do not rely on
63+
current behavior. Note that many Rhombus binding forms define
64+
identifiers that can be used as expressions, but they are implemented as
65+
syntactic forms; notably, the constructor bound by a
66+
@rhombus(class, ~defn) definition is not like a @rhombus(fun) binding.
67+
68+
To use a Rhombus binding form Racket in general, use
69+
@racket_rhombus_expression from @racketmod_rhombus_parse.}
70+
71+
72+
)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
#lang scribble/manual
2+
@(require (for-label rhombus/parse
3+
racket/base))
4+
5+
@title{Rhombus Expressions in Racket}
6+
7+
@defmodule[rhombus/parse]
8+
9+
Rhombus shrubbery forms can be written in S-expressions using the
10+
encoding described in @secref[#:doc '(lib "shrubbery/scribblings/shrubbery.scrbl") "parsed-rep"].
11+
The @racket[rhombus-expression] and @racket[rhombus-top] forms treat an encoding
12+
as a Rhombus expression or a module-level Rhombus sequence, respectively.
13+
14+
The forms use Rhombus bindings that are @racket[require]d into the
15+
enclosing context. Binding does not matter for structural tokens like
16+
@racket[group] or @racket[op], but they matter for S-expression
17+
identifiers that represent shrubbery identifiers and operations.
18+
19+
@defform[(rhombus-expression shrubbery)]{
20+
21+
Parses @racket[shrubbery] as a Rhombus expression for a Racket
22+
expression context. The @racket[shrubbery] form should represent a
23+
group, meaning that it should be a list-shaped S-expression that starts
24+
with the identifier @racket[group].
25+
26+
}
27+
28+
@defform[(rhombus-top shrubbery)]{
29+
30+
Parses @racket[shrubbery] as a Rhombus module-body sequence for a
31+
Racket module context. The @racket[shrubbery] form should be a
32+
list-shaped S-expression that starts with the identifier @racket[multi].
33+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
#lang rhombus/scribble/manual
2+
@(import:
3+
"racket_names.rkt" open
4+
meta_label:
5+
rhombus open
6+
rhombus/meta:
7+
expose:
8+
expr_meta
9+
defn_meta
10+
decl_meta
11+
expr)
12+
13+
@title(~tag: "racket-expr"){Racket Expressions from Rhombus}
14+
15+
Using a Racket syntactic form from Rhombus requires a Rhombus macro that
16+
expands to the Racket form. A Rhombus macro can build a use of a Racket
17+
syntactic form using @rhombus(expr_meta.pack_s_exp),
18+
@rhombus(defn_meta.pack_s_exp), or @rhombus(decl_meta.pack_s_exp). Use
19+
@rhombus(expr_meta.pack_expr) to pack a Rhombus expression for use
20+
within a packed S-expression form.
21+
22+
For example, the following @rhombus(shared) macro expands to a use of the
23+
Racket @racket_shared form, which specifically recognizes uses of the
24+
Racket @racket_cons function.
25+
26+
@examples(
27+
~defn:
28+
import:
29+
rhombus/meta open
30+
lib("racket/base.rkt").cons
31+
lib("racket/shared.rkt").shared as rkt_shared
32+
~defn:
33+
expr.macro 'shared ($(id :: Identifier) = $rhs ...,
34+
...):
35+
$body
36+
...':
37+
expr_meta.pack_s_exp(['rkt_shared',
38+
[[id, expr_meta.pack_expr('$rhs ...')],
39+
...],
40+
expr_meta.pack_expr('block:
41+
$body
42+
...')])
43+
~repl:
44+
shared (x = cons(1, y),
45+
y = cons(2, x)):
46+
x.rest.rest.rest.first
47+
)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
#lang at-exp racket/base
2+
(require scribble/manual
3+
(for-label racket/base
4+
rhombus/parse
5+
racket/shared))
6+
7+
(provide racket_require
8+
racket_lib
9+
racket_file
10+
racketmod_racket_math
11+
racketmod_rhombus_random
12+
racketmod_rhombus_parse
13+
racketmod_racket_treelist
14+
racketmod_racket_mutable_treelist
15+
racket_rhombus_expression
16+
racket_shared
17+
racket_cons)
18+
19+
(define racket_require @racket[require])
20+
(define racket_lib @racket[lib])
21+
(define racket_file @racket[file])
22+
23+
(define racketmod_racket_math @racketmodname[racket/math])
24+
(define racketmod_rhombus_random @racketmodname[(lib "rhombus/random.rhm")])
25+
(define racketmod_rhombus_parse @racketmodname[rhombus/parse])
26+
(define racketmod_racket_treelist @racketmodname[racket/treelist])
27+
(define racketmod_racket_mutable_treelist @racketmodname[racket/mutable-treelist])
28+
29+
(define racket_rhombus_expression @racket[rhombus-expression])
30+
(define racket_shared @racket[shared])
31+
(define racket_cons @racket[cons])

0 commit comments

Comments
 (0)
Please sign in to comment.