Discussion:
What should the constructor for a record look like?
HiPhish
2018-08-26 13:04:44 UTC
Permalink
Hello, it's me again, the guy who wants to implement MessagePack [1] in Guile.
The specification defines a type of "extension" [2], a pair of an 8-bit
integer and a byte array for data. Implementing this type as a record is
obvious, but what should be the name of the constructor?

(define-record-type ext
(make-ext type data)
ext?
(type ext-type)
(data ext-data))

Either `make-ext` or just `ext` seem appropriate. I have seen both types of
constructors, what do you guys say? And while I'm on the topic, what about
types? An extension is only valid if the first field (type) is an integer from
-128 to 127 and the data a vector of bytes (integer in the range from 0 to
255). How do I enforce this invariant without static typing?

[1] https://msgpack.org/
[2] https://github.com/msgpack/msgpack/blob/master/spec.md#ext-format-family
Matt Wette
2018-08-26 16:11:42 UTC
Permalink
Post by HiPhish
Hello, it's me again, the guy who wants to implement MessagePack [1] in Guile.
The specification defines a type of "extension" [2], a pair of an 8-bit
integer and a byte array for data. Implementing this type as a record is
obvious, but what should be the name of the constructor?
(define-record-type ext
(make-ext type data)
ext?
(type ext-type)
(data ext-data))
Either `make-ext` or just `ext` seem appropriate. I have seen both types of
constructors, what do you guys say? And while I'm on the topic, what about
types? An extension is only valid if the first field (type) is an integer from
-128 to 127 and the data a vector of bytes (integer in the range from 0 to
255). How do I enforce this invariant without static typing?
[1] https://msgpack.org/
[2] https://github.com/msgpack/msgpack/blob/master/spec.md#ext-format-family
I like make-ext vs ext.

I assume you are using bytevectors for the data. You could "enforce" that type in
the record by using a separate definition of `ext?'.

(define-record-type ext
(make-ext type data)
ext-record?
(type ext-type)
(data ext-data))

(define (ext? obj)
(and (ext-record? obj)
(integer? (ext-type obj))
(bytevector? (ext-data obj)))
Ludovic Courtès
2018-08-27 12:22:38 UTC
Permalink
Hi,
Post by HiPhish
Hello, it's me again, the guy who wants to implement MessagePack [1] in Guile.
The specification defines a type of "extension" [2], a pair of an 8-bit
integer and a byte array for data. Implementing this type as a record is
obvious, but what should be the name of the constructor?
(define-record-type ext
(make-ext type data)
ext?
(type ext-type)
(data ext-data))
Either `make-ext` or just `ext` seem appropriate.
For immutable records (which should be the norm :-)), I often leave out
‘make-’ nowadays. (The analogy is ‘string’ vs. ‘make-string’, for
instance.)
Post by HiPhish
I have seen both types of constructors, what do you guys say? And
guys and gals :-)
Post by HiPhish
while I'm on the topic, what about types? An extension is only valid
if the first field (type) is an integer from -128 to 127 and the data
a vector of bytes (integer in the range from 0 to 255). How do I
enforce this invariant without static typing?
You could enforce it by not exporting the raw record constructor, and
instead exporting a procedure that performs all the necessary checks:

(define-record-type <ext>
(%ext type data) ;private
ext?
…)

(define (ext type data) ;public
(assert-valid-arguments type data)
(%ext type data))

HTH!

Ludo’.

Loading...