2013-05-12 09:46:16 -04:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2024-04-08 15:47:29 -04:00
|
|
|
|
;;; Copyright © 2012-2024 Ludovic Courtès <ludo@gnu.org>
|
2018-04-19 12:33:25 -04:00
|
|
|
|
;;; Copyright © 2018 Mark H Weaver <mhw@netris.org>
|
2013-05-12 09:46:16 -04:00
|
|
|
|
;;;
|
|
|
|
|
;;; This file is part of GNU Guix.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix is free software; you can redistribute it and/or modify it
|
|
|
|
|
;;; under the terms of the GNU General Public License as published by
|
|
|
|
|
;;; the Free Software Foundation; either version 3 of the License, or (at
|
|
|
|
|
;;; your option) any later version.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix is distributed in the hope that it will be useful, but
|
|
|
|
|
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
;;; GNU General Public License for more details.
|
|
|
|
|
;;;
|
|
|
|
|
;;; You should have received a copy of the GNU General Public License
|
|
|
|
|
;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
(define-module (guix records)
|
|
|
|
|
#:use-module (srfi srfi-1)
|
|
|
|
|
#:use-module (srfi srfi-9)
|
2023-06-06 03:06:59 -04:00
|
|
|
|
#:use-module (srfi srfi-11)
|
2013-05-12 09:46:16 -04:00
|
|
|
|
#:use-module (srfi srfi-26)
|
|
|
|
|
#:use-module (ice-9 match)
|
2013-07-10 10:54:17 -04:00
|
|
|
|
#:use-module (ice-9 rdelim)
|
2020-04-07 17:55:14 -04:00
|
|
|
|
#:autoload (system base target) (target-most-positive-fixnum)
|
2013-05-12 09:46:16 -04:00
|
|
|
|
#:export (define-record-type*
|
2019-03-22 09:02:00 -04:00
|
|
|
|
this-record
|
|
|
|
|
|
2013-05-12 09:46:16 -04:00
|
|
|
|
alist->record
|
2013-07-10 10:54:17 -04:00
|
|
|
|
object->fields
|
2017-10-25 14:44:54 -04:00
|
|
|
|
recutils->alist
|
2023-04-28 15:19:05 -04:00
|
|
|
|
match-record
|
|
|
|
|
match-record-lambda))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
;;;
|
|
|
|
|
;;; Utilities for dealing with Scheme records.
|
|
|
|
|
;;;
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
2014-07-17 10:42:19 -04:00
|
|
|
|
(define-syntax record-error
|
|
|
|
|
(syntax-rules ()
|
|
|
|
|
"Report a syntactic error in use of CONSTRUCTOR."
|
|
|
|
|
((_ constructor form fmt args ...)
|
|
|
|
|
(syntax-violation constructor
|
|
|
|
|
(format #f fmt args ...)
|
|
|
|
|
form))))
|
|
|
|
|
|
2018-05-31 11:32:22 -04:00
|
|
|
|
(eval-when (expand load eval)
|
|
|
|
|
;; The procedures below are needed both at run time and at expansion time.
|
|
|
|
|
|
|
|
|
|
(define (current-abi-identifier type)
|
|
|
|
|
"Return an identifier unhygienically derived from TYPE for use as its
|
2018-05-16 04:05:24 -04:00
|
|
|
|
\"current ABI\" variable."
|
2018-05-31 11:32:22 -04:00
|
|
|
|
(let ((type-name (syntax->datum type)))
|
|
|
|
|
(datum->syntax
|
|
|
|
|
type
|
|
|
|
|
(string->symbol
|
|
|
|
|
(string-append "% " (symbol->string type-name)
|
|
|
|
|
" abi-cookie")))))
|
2018-05-16 04:05:24 -04:00
|
|
|
|
|
2024-04-08 15:47:29 -04:00
|
|
|
|
(define (record-abi-mismatch-error type)
|
|
|
|
|
(throw 'record-abi-mismatch-error 'abi-check
|
|
|
|
|
"~a: record ABI mismatch; recompilation needed"
|
|
|
|
|
(list type) '()))
|
|
|
|
|
|
2018-05-31 11:32:22 -04:00
|
|
|
|
(define (abi-check type cookie)
|
|
|
|
|
"Return syntax that checks that the current \"application binary
|
2018-05-16 04:05:24 -04:00
|
|
|
|
interface\" (ABI) for TYPE is equal to COOKIE."
|
2018-05-31 11:32:22 -04:00
|
|
|
|
(with-syntax ((current-abi (current-abi-identifier type)))
|
|
|
|
|
#`(unless (eq? current-abi #,cookie)
|
2018-08-29 16:28:06 -04:00
|
|
|
|
;; The source file where this exception is thrown must be
|
|
|
|
|
;; recompiled.
|
2024-04-08 15:47:29 -04:00
|
|
|
|
(record-abi-mismatch-error #,type))))
|
2019-01-24 14:48:14 -05:00
|
|
|
|
|
2020-01-16 09:00:18 -05:00
|
|
|
|
(define* (report-invalid-field-specifier name bindings
|
|
|
|
|
#:optional parent-form)
|
|
|
|
|
"Report the first invalid binding among BINDINGS. PARENT-FORM is used for
|
|
|
|
|
error-reporting purposes."
|
2019-01-24 14:48:14 -05:00
|
|
|
|
(let loop ((bindings bindings))
|
|
|
|
|
(syntax-case bindings ()
|
|
|
|
|
(((field value) rest ...) ;good
|
|
|
|
|
(loop #'(rest ...)))
|
|
|
|
|
((weird _ ...) ;weird!
|
2020-01-16 09:00:18 -05:00
|
|
|
|
;; WEIRD may be an identifier, thus lacking source location info, and
|
|
|
|
|
;; BINDINGS is a list, also lacking source location info. Hopefully
|
|
|
|
|
;; PARENT-FORM provides source location info.
|
|
|
|
|
(apply syntax-violation name "invalid field specifier"
|
|
|
|
|
(if parent-form
|
|
|
|
|
(list parent-form #'weird)
|
|
|
|
|
(list #'weird)))))))
|
2019-01-24 14:48:14 -05:00
|
|
|
|
|
|
|
|
|
(define (report-duplicate-field-specifier name ctor)
|
|
|
|
|
"Report the first duplicate identifier among the bindings in CTOR."
|
|
|
|
|
(syntax-case ctor ()
|
|
|
|
|
((_ bindings ...)
|
|
|
|
|
(let loop ((bindings #'(bindings ...))
|
|
|
|
|
(seen '()))
|
|
|
|
|
(syntax-case bindings ()
|
|
|
|
|
(((field value) rest ...)
|
|
|
|
|
(not (memq (syntax->datum #'field) seen))
|
|
|
|
|
(loop #'(rest ...) (cons (syntax->datum #'field) seen)))
|
|
|
|
|
((duplicate rest ...)
|
|
|
|
|
(syntax-violation name "duplicate field initializer"
|
|
|
|
|
#'duplicate))
|
|
|
|
|
(()
|
|
|
|
|
#t)))))))
|
2018-05-16 04:05:24 -04:00
|
|
|
|
|
2022-11-19 11:23:04 -05:00
|
|
|
|
(define-syntax map-fields
|
|
|
|
|
(lambda (x)
|
2023-04-28 15:19:01 -04:00
|
|
|
|
(syntax-case x ()
|
|
|
|
|
((_ type within)
|
|
|
|
|
(syntax-violation (syntax->datum #'within)
|
2023-06-09 08:00:51 -04:00
|
|
|
|
"undefined record type"
|
2023-04-28 15:19:01 -04:00
|
|
|
|
#'type))
|
|
|
|
|
(_ (syntax-violation 'map-fields "bad use of syntactic keyword" x x)))))
|
2022-11-19 11:23:04 -05:00
|
|
|
|
|
2019-03-22 09:02:00 -04:00
|
|
|
|
(define-syntax-parameter this-record
|
|
|
|
|
(lambda (s)
|
|
|
|
|
"Return the record being defined. This macro may only be used in the
|
|
|
|
|
context of the definition of a thunked field."
|
|
|
|
|
(syntax-case s ()
|
|
|
|
|
(id
|
|
|
|
|
(identifier? #'id)
|
|
|
|
|
(syntax-violation 'this-record
|
|
|
|
|
"cannot be used outside of a record instantiation"
|
|
|
|
|
#'id)))))
|
|
|
|
|
|
2015-06-11 15:37:49 -04:00
|
|
|
|
(define-syntax make-syntactic-constructor
|
|
|
|
|
(syntax-rules ()
|
|
|
|
|
"Make the syntactic constructor NAME for TYPE, that calls CTOR, and
|
|
|
|
|
expects all of EXPECTED fields to be initialized. DEFAULTS is the list of
|
|
|
|
|
FIELD/DEFAULT-VALUE tuples, THUNKED is the list of identifiers of thunked
|
2021-05-20 09:40:55 -04:00
|
|
|
|
fields, DELAYED is the list of identifiers of delayed fields, and SANITIZERS
|
|
|
|
|
is the list of FIELD/SANITIZER tuples.
|
2018-05-16 04:05:24 -04:00
|
|
|
|
|
|
|
|
|
ABI-COOKIE is the cookie (an integer) against which to check the run-time ABI
|
|
|
|
|
of TYPE matches the expansion-time ABI."
|
2015-06-11 15:37:49 -04:00
|
|
|
|
((_ type name ctor (expected ...)
|
2018-05-16 04:05:24 -04:00
|
|
|
|
#:abi-cookie abi-cookie
|
2015-06-11 15:37:49 -04:00
|
|
|
|
#:thunked thunked
|
2019-03-29 17:40:55 -04:00
|
|
|
|
#:this-identifier this-identifier
|
2015-06-11 15:37:49 -04:00
|
|
|
|
#:delayed delayed
|
2015-06-11 16:57:33 -04:00
|
|
|
|
#:innate innate
|
2021-05-20 09:40:55 -04:00
|
|
|
|
#:sanitizers sanitizers
|
2015-06-11 15:37:49 -04:00
|
|
|
|
#:defaults defaults)
|
|
|
|
|
(define-syntax name
|
|
|
|
|
(lambda (s)
|
|
|
|
|
(define (record-inheritance orig-record field+value)
|
|
|
|
|
;; Produce code that returns a record identical to ORIG-RECORD,
|
|
|
|
|
;; except that values for the FIELD+VALUE alist prevail.
|
|
|
|
|
(define (field-inherited-value f)
|
|
|
|
|
(and=> (find (lambda (x)
|
|
|
|
|
(eq? f (car (syntax->datum x))))
|
|
|
|
|
field+value)
|
|
|
|
|
car))
|
|
|
|
|
|
|
|
|
|
;; Make sure there are no unknown field names.
|
|
|
|
|
(let* ((fields (map (compose car syntax->datum) field+value))
|
|
|
|
|
(unexpected (lset-difference eq? fields '(expected ...))))
|
|
|
|
|
(when (pair? unexpected)
|
|
|
|
|
(record-error 'name s "extraneous field initializers ~a"
|
|
|
|
|
unexpected)))
|
|
|
|
|
|
2017-12-01 12:09:22 -05:00
|
|
|
|
#`(make-struct/no-tail type
|
2015-06-11 15:37:49 -04:00
|
|
|
|
#,@(map (lambda (field index)
|
|
|
|
|
(or (field-inherited-value field)
|
2015-06-11 16:57:33 -04:00
|
|
|
|
(if (innate-field? field)
|
|
|
|
|
(wrap-field-value
|
|
|
|
|
field (field-default-value field))
|
|
|
|
|
#`(struct-ref #,orig-record
|
|
|
|
|
#,index))))
|
2015-06-11 15:37:49 -04:00
|
|
|
|
'(expected ...)
|
|
|
|
|
(iota (length '(expected ...))))))
|
|
|
|
|
|
|
|
|
|
(define (thunked-field? f)
|
|
|
|
|
(memq (syntax->datum f) 'thunked))
|
|
|
|
|
|
|
|
|
|
(define (delayed-field? f)
|
|
|
|
|
(memq (syntax->datum f) 'delayed))
|
|
|
|
|
|
2015-06-11 16:57:33 -04:00
|
|
|
|
(define (innate-field? f)
|
|
|
|
|
(memq (syntax->datum f) 'innate))
|
|
|
|
|
|
2021-05-20 09:40:55 -04:00
|
|
|
|
(define field-sanitizer
|
|
|
|
|
(let ((lst (map (match-lambda
|
|
|
|
|
((f p)
|
|
|
|
|
(list (syntax->datum f) p)))
|
|
|
|
|
#'sanitizers)))
|
|
|
|
|
(lambda (f)
|
|
|
|
|
(or (and=> (assoc-ref lst (syntax->datum f)) car)
|
|
|
|
|
#'(lambda (x) x)))))
|
|
|
|
|
|
2015-06-11 15:37:49 -04:00
|
|
|
|
(define (wrap-field-value f value)
|
2021-05-20 09:40:55 -04:00
|
|
|
|
(let* ((sanitizer (field-sanitizer f))
|
|
|
|
|
(value #`(#,sanitizer #,value)))
|
|
|
|
|
(cond ((thunked-field? f)
|
|
|
|
|
#`(lambda (x)
|
|
|
|
|
(syntax-parameterize ((#,this-identifier
|
|
|
|
|
(lambda (s)
|
|
|
|
|
(syntax-case s ()
|
|
|
|
|
(id
|
|
|
|
|
(identifier? #'id)
|
|
|
|
|
#'x)))))
|
|
|
|
|
#,value)))
|
|
|
|
|
((delayed-field? f)
|
|
|
|
|
#`(delay #,value))
|
|
|
|
|
(else value))))
|
2015-06-11 15:37:49 -04:00
|
|
|
|
|
2015-06-11 15:49:02 -04:00
|
|
|
|
(define default-values
|
|
|
|
|
;; List of symbol/value tuples.
|
|
|
|
|
(map (match-lambda
|
|
|
|
|
((f v)
|
|
|
|
|
(list (syntax->datum f) v)))
|
|
|
|
|
#'defaults))
|
|
|
|
|
|
|
|
|
|
(define (field-default-value f)
|
|
|
|
|
(car (assoc-ref default-values (syntax->datum f))))
|
|
|
|
|
|
2015-06-11 15:37:49 -04:00
|
|
|
|
(define (field-bindings field+value)
|
|
|
|
|
;; Return field to value bindings, for use in 'let*' below.
|
|
|
|
|
(map (lambda (field+value)
|
|
|
|
|
(syntax-case field+value ()
|
|
|
|
|
((field value)
|
|
|
|
|
#`(field
|
|
|
|
|
#,(wrap-field-value #'field #'value)))))
|
|
|
|
|
field+value))
|
|
|
|
|
|
|
|
|
|
(syntax-case s (inherit expected ...)
|
|
|
|
|
((_ (inherit orig-record) (field value) (... ...))
|
|
|
|
|
#`(let* #,(field-bindings #'((field value) (... ...)))
|
2018-05-16 04:05:24 -04:00
|
|
|
|
#,(abi-check #'type abi-cookie)
|
2015-06-11 15:37:49 -04:00
|
|
|
|
#,(record-inheritance #'orig-record
|
|
|
|
|
#'((field value) (... ...)))))
|
|
|
|
|
((_ (field value) (... ...))
|
2015-06-11 15:49:02 -04:00
|
|
|
|
(let ((fields (map syntax->datum #'(field (... ...)))))
|
2015-06-11 15:37:49 -04:00
|
|
|
|
(define (field-value f)
|
2017-04-16 11:25:11 -04:00
|
|
|
|
(or (find (lambda (x)
|
|
|
|
|
(eq? f (syntax->datum x)))
|
|
|
|
|
#'(field (... ...)))
|
2015-06-11 15:49:02 -04:00
|
|
|
|
(wrap-field-value f (field-default-value f))))
|
2015-06-11 15:37:49 -04:00
|
|
|
|
|
2018-04-19 12:33:25 -04:00
|
|
|
|
;; Pass S to make sure source location info is preserved.
|
|
|
|
|
(report-duplicate-field-specifier 'name s)
|
|
|
|
|
|
2015-06-11 15:49:02 -04:00
|
|
|
|
(let ((fields (append fields (map car default-values))))
|
2015-06-11 15:37:49 -04:00
|
|
|
|
(cond ((lset= eq? fields '(expected ...))
|
|
|
|
|
#`(let* #,(field-bindings
|
|
|
|
|
#'((field value) (... ...)))
|
2018-05-16 04:05:24 -04:00
|
|
|
|
#,(abi-check #'type abi-cookie)
|
2015-06-11 15:37:49 -04:00
|
|
|
|
(ctor #,@(map field-value '(expected ...)))))
|
|
|
|
|
((pair? (lset-difference eq? fields
|
|
|
|
|
'(expected ...)))
|
|
|
|
|
(record-error 'name s
|
|
|
|
|
"extraneous field initializers ~a"
|
|
|
|
|
(lset-difference eq? fields
|
|
|
|
|
'(expected ...))))
|
|
|
|
|
(else
|
|
|
|
|
(record-error 'name s
|
|
|
|
|
"missing field initializers ~a"
|
|
|
|
|
(lset-difference eq?
|
|
|
|
|
'(expected ...)
|
2016-07-14 12:58:36 -04:00
|
|
|
|
fields)))))))
|
|
|
|
|
((_ bindings (... ...))
|
|
|
|
|
;; One of BINDINGS doesn't match the (field value) pattern.
|
|
|
|
|
;; Report precisely which one is faulty, instead of letting the
|
|
|
|
|
;; "source expression failed to match any pattern" error.
|
|
|
|
|
(report-invalid-field-specifier 'name
|
2020-01-16 09:00:18 -05:00
|
|
|
|
#'(bindings (... ...))
|
|
|
|
|
s))))))))
|
2015-01-19 16:30:55 -05:00
|
|
|
|
|
2015-06-11 16:22:05 -04:00
|
|
|
|
(define-syntax-rule (define-field-property-predicate predicate property)
|
|
|
|
|
"Define PREDICATE as a procedure that takes a syntax object and, when passed
|
|
|
|
|
a field specification, returns the field name if it has the given PROPERTY."
|
|
|
|
|
(define (predicate s)
|
|
|
|
|
(syntax-case s (property)
|
|
|
|
|
((field (property values (... ...)) _ (... ...))
|
|
|
|
|
#'field)
|
|
|
|
|
((field _ properties (... ...))
|
|
|
|
|
(predicate #'(field properties (... ...))))
|
|
|
|
|
(_ #f))))
|
|
|
|
|
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(define-syntax define-record-type*
|
|
|
|
|
(lambda (s)
|
|
|
|
|
"Define the given record type such that an additional \"syntactic
|
|
|
|
|
constructor\" is defined, which allows instances to be constructed with named
|
2014-10-14 08:44:48 -04:00
|
|
|
|
field initializers, à la SRFI-35, as well as default values. An example use
|
|
|
|
|
may look like this:
|
|
|
|
|
|
|
|
|
|
(define-record-type* <thing> thing make-thing
|
|
|
|
|
thing?
|
2019-03-29 17:40:55 -04:00
|
|
|
|
this-thing
|
2014-10-14 08:44:48 -04:00
|
|
|
|
(name thing-name (default \"chbouib\"))
|
|
|
|
|
(port thing-port
|
2015-06-11 16:57:33 -04:00
|
|
|
|
(default (current-output-port)) (thunked))
|
|
|
|
|
(loc thing-location (innate) (default (current-source-location))))
|
2014-10-14 08:44:48 -04:00
|
|
|
|
|
|
|
|
|
This example defines a macro 'thing' that can be used to instantiate records
|
|
|
|
|
of this type:
|
|
|
|
|
|
|
|
|
|
(thing
|
|
|
|
|
(name \"foo\")
|
|
|
|
|
(port (current-error-port)))
|
|
|
|
|
|
|
|
|
|
The value of 'name' or 'port' could as well be omitted, in which case the
|
|
|
|
|
default value specified in the 'define-record-type*' form is used:
|
|
|
|
|
|
|
|
|
|
(thing)
|
|
|
|
|
|
|
|
|
|
The 'port' field is \"thunked\", meaning that calls like '(thing-port x)' will
|
|
|
|
|
actually compute the field's value in the current dynamic extent, which is
|
2019-03-29 17:40:55 -04:00
|
|
|
|
useful when referring to fluids in a field's value. Furthermore, that thunk
|
|
|
|
|
can access the record it belongs to via the 'this-thing' identifier.
|
2014-10-14 08:44:48 -04:00
|
|
|
|
|
2015-01-19 17:21:47 -05:00
|
|
|
|
A field can also be marked as \"delayed\" instead of \"thunked\", in which
|
|
|
|
|
case its value is effectively wrapped in a (delay …) form.
|
|
|
|
|
|
2021-05-20 09:40:55 -04:00
|
|
|
|
A field can also have an associated \"sanitizer\", which is a procedure that
|
|
|
|
|
takes a user-supplied field value and returns a \"sanitized\" value for the
|
|
|
|
|
field:
|
|
|
|
|
|
|
|
|
|
(define-record-type* <thing> thing make-thing
|
|
|
|
|
thing?
|
|
|
|
|
this-thing
|
|
|
|
|
(name thing-name
|
|
|
|
|
(sanitize (lambda (value)
|
|
|
|
|
(cond ((string? value) value)
|
|
|
|
|
((symbol? value) (symbol->string value))
|
|
|
|
|
(else (throw 'bad! value)))))))
|
|
|
|
|
|
2014-10-14 08:44:48 -04:00
|
|
|
|
It is possible to copy an object 'x' created with 'thing' like this:
|
|
|
|
|
|
|
|
|
|
(thing (inherit x) (name \"bar\"))
|
|
|
|
|
|
|
|
|
|
This expression returns a new object equal to 'x' except for its 'name'
|
2015-06-11 16:57:33 -04:00
|
|
|
|
field and its 'loc' field---the latter is marked as \"innate\", so it is not
|
|
|
|
|
inherited."
|
2014-10-14 08:44:48 -04:00
|
|
|
|
|
2022-11-19 11:23:04 -05:00
|
|
|
|
(define (rtd-identifier type)
|
|
|
|
|
;; Return an identifier derived from TYPE to name its record type
|
|
|
|
|
;; descriptor (RTD).
|
|
|
|
|
(let ((type-name (syntax->datum type)))
|
|
|
|
|
(datum->syntax
|
|
|
|
|
type
|
|
|
|
|
(string->symbol
|
|
|
|
|
(string-append "% " (symbol->string type-name) " rtd")))))
|
|
|
|
|
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(define (field-default-value s)
|
|
|
|
|
(syntax-case s (default)
|
|
|
|
|
((field (default val) _ ...)
|
|
|
|
|
(list #'field #'val))
|
2015-06-11 16:26:18 -04:00
|
|
|
|
((field _ properties ...)
|
|
|
|
|
(field-default-value #'(field properties ...)))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(_ #f)))
|
|
|
|
|
|
2021-05-20 09:40:55 -04:00
|
|
|
|
(define (field-sanitizer s)
|
|
|
|
|
(syntax-case s (sanitize)
|
|
|
|
|
((field (sanitize proc) _ ...)
|
|
|
|
|
(list #'field #'proc))
|
|
|
|
|
((field _ properties ...)
|
|
|
|
|
(field-sanitizer #'(field properties ...)))
|
|
|
|
|
(_ #f)))
|
|
|
|
|
|
2015-06-11 16:22:05 -04:00
|
|
|
|
(define-field-property-predicate delayed-field? delayed)
|
|
|
|
|
(define-field-property-predicate thunked-field? thunked)
|
2015-06-11 16:57:33 -04:00
|
|
|
|
(define-field-property-predicate innate-field? innate)
|
2013-05-12 09:46:16 -04:00
|
|
|
|
|
2015-01-19 17:21:47 -05:00
|
|
|
|
(define (wrapped-field? s)
|
|
|
|
|
(or (thunked-field? s) (delayed-field? s)))
|
|
|
|
|
|
|
|
|
|
(define (wrapped-field-accessor-name field)
|
2013-05-12 09:46:16 -04:00
|
|
|
|
;; Return the name (an unhygienic syntax object) of the "real"
|
2015-01-19 17:21:47 -05:00
|
|
|
|
;; getter for field, which is assumed to be a wrapped field.
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(syntax-case field ()
|
2015-06-11 16:26:18 -04:00
|
|
|
|
((field get properties ...)
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(let* ((getter (syntax->datum #'get))
|
|
|
|
|
(real-getter (symbol-append '% getter '-real)))
|
|
|
|
|
(datum->syntax #'get real-getter)))))
|
|
|
|
|
|
|
|
|
|
(define (field-spec->srfi-9 field)
|
|
|
|
|
;; Convert a field spec of our style to a SRFI-9 field spec of the
|
|
|
|
|
;; form (field get).
|
|
|
|
|
(syntax-case field ()
|
2015-06-11 16:26:18 -04:00
|
|
|
|
((name get properties ...)
|
2013-05-12 09:46:16 -04:00
|
|
|
|
#`(name
|
2015-01-19 17:21:47 -05:00
|
|
|
|
#,(if (wrapped-field? field)
|
|
|
|
|
(wrapped-field-accessor-name field)
|
2013-05-12 09:46:16 -04:00
|
|
|
|
#'get)))))
|
|
|
|
|
|
|
|
|
|
(define (thunked-field-accessor-definition field)
|
|
|
|
|
;; Return the real accessor for FIELD, which is assumed to be a
|
|
|
|
|
;; thunked field.
|
|
|
|
|
(syntax-case field ()
|
|
|
|
|
((name get _ ...)
|
2015-01-19 17:21:47 -05:00
|
|
|
|
(with-syntax ((real-get (wrapped-field-accessor-name field)))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
#'(define-inlinable (get x)
|
|
|
|
|
;; The real value of that field is a thunk, so call it.
|
2019-03-22 09:02:00 -04:00
|
|
|
|
((real-get x) x))))))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
|
2015-01-19 17:21:47 -05:00
|
|
|
|
(define (delayed-field-accessor-definition field)
|
|
|
|
|
;; Return the real accessor for FIELD, which is assumed to be a
|
|
|
|
|
;; delayed field.
|
|
|
|
|
(syntax-case field ()
|
|
|
|
|
((name get _ ...)
|
|
|
|
|
(with-syntax ((real-get (wrapped-field-accessor-name field)))
|
|
|
|
|
#'(define-inlinable (get x)
|
|
|
|
|
;; The real value of that field is a promise, so force it.
|
|
|
|
|
(force (real-get x)))))))
|
|
|
|
|
|
2018-05-16 04:05:24 -04:00
|
|
|
|
(define (compute-abi-cookie field-specs)
|
|
|
|
|
;; Compute an "ABI cookie" for the given FIELD-SPECS. We use
|
|
|
|
|
;; 'string-hash' because that's a better hash function that 'hash' on a
|
|
|
|
|
;; list of symbols.
|
|
|
|
|
(syntax-case field-specs ()
|
|
|
|
|
(((field get properties ...) ...)
|
|
|
|
|
(string-hash (object->string
|
|
|
|
|
(syntax->datum #'((field properties ...) ...)))
|
2020-04-07 17:55:14 -04:00
|
|
|
|
(cond-expand
|
|
|
|
|
(guile-3 (target-most-positive-fixnum))
|
|
|
|
|
(else most-positive-fixnum))))))
|
2018-05-16 04:05:24 -04:00
|
|
|
|
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(syntax-case s ()
|
|
|
|
|
((_ type syntactic-ctor ctor pred
|
2019-03-29 17:40:55 -04:00
|
|
|
|
this-identifier
|
2015-06-11 16:26:18 -04:00
|
|
|
|
(field get properties ...) ...)
|
2019-03-29 17:40:55 -04:00
|
|
|
|
(identifier? #'this-identifier)
|
2015-06-11 16:26:18 -04:00
|
|
|
|
(let* ((field-spec #'((field get properties ...) ...))
|
2015-01-19 16:27:58 -05:00
|
|
|
|
(thunked (filter-map thunked-field? field-spec))
|
2015-01-19 17:21:47 -05:00
|
|
|
|
(delayed (filter-map delayed-field? field-spec))
|
2015-06-11 16:57:33 -04:00
|
|
|
|
(innate (filter-map innate-field? field-spec))
|
2015-01-19 16:27:58 -05:00
|
|
|
|
(defaults (filter-map field-default-value
|
2018-05-16 04:05:24 -04:00
|
|
|
|
#'((field properties ...) ...)))
|
2021-05-20 09:40:55 -04:00
|
|
|
|
(sanitizers (filter-map field-sanitizer
|
2023-04-28 15:19:03 -04:00
|
|
|
|
#'((field properties ...) ...)))
|
2018-05-16 04:05:24 -04:00
|
|
|
|
(cookie (compute-abi-cookie field-spec)))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(with-syntax (((field-spec* ...)
|
|
|
|
|
(map field-spec->srfi-9 field-spec))
|
2023-04-28 15:19:03 -04:00
|
|
|
|
((field-type ...)
|
|
|
|
|
(map (match-lambda
|
|
|
|
|
((? thunked-field?)
|
|
|
|
|
(datum->syntax s 'thunked))
|
|
|
|
|
((? delayed-field?)
|
|
|
|
|
(datum->syntax s 'delayed))
|
|
|
|
|
(else
|
|
|
|
|
(datum->syntax s 'normal)))
|
|
|
|
|
field-spec))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
((thunked-field-accessor ...)
|
|
|
|
|
(filter-map (lambda (field)
|
|
|
|
|
(and (thunked-field? field)
|
|
|
|
|
(thunked-field-accessor-definition
|
|
|
|
|
field)))
|
2015-01-19 17:21:47 -05:00
|
|
|
|
field-spec))
|
|
|
|
|
((delayed-field-accessor ...)
|
|
|
|
|
(filter-map (lambda (field)
|
|
|
|
|
(and (delayed-field? field)
|
|
|
|
|
(delayed-field-accessor-definition
|
|
|
|
|
field)))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
field-spec)))
|
|
|
|
|
#`(begin
|
2022-11-19 11:23:04 -05:00
|
|
|
|
(define-record-type #,(rtd-identifier #'type)
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(ctor field ...)
|
|
|
|
|
pred
|
|
|
|
|
field-spec* ...)
|
2022-11-19 11:23:04 -05:00
|
|
|
|
|
|
|
|
|
;; Rectify the vtable type name...
|
|
|
|
|
(set-struct-vtable-name! #,(rtd-identifier #'type) 'type)
|
|
|
|
|
(cond-expand
|
|
|
|
|
(guile-3
|
|
|
|
|
;; ... and the record type name.
|
|
|
|
|
(struct-set! #,(rtd-identifier #'type) vtable-offset-user
|
|
|
|
|
'type))
|
|
|
|
|
(else #f))
|
|
|
|
|
|
|
|
|
|
(define-syntax type
|
|
|
|
|
(lambda (s)
|
|
|
|
|
"This macro lets us query record type info at
|
|
|
|
|
macro-expansion time."
|
|
|
|
|
(syntax-case s (map-fields)
|
2023-04-28 15:19:01 -04:00
|
|
|
|
((_ (map-fields _ _) macro)
|
2023-04-28 15:19:03 -04:00
|
|
|
|
#'(macro ((field field-type) ...)))
|
2022-11-19 11:23:04 -05:00
|
|
|
|
(id
|
|
|
|
|
(identifier? #'id)
|
|
|
|
|
#'#,(rtd-identifier #'type)))))
|
|
|
|
|
|
2018-05-16 04:05:24 -04:00
|
|
|
|
(define #,(current-abi-identifier #'type)
|
|
|
|
|
#,cookie)
|
2019-03-29 17:40:55 -04:00
|
|
|
|
|
|
|
|
|
#,@(if (free-identifier=? #'this-identifier #'this-record)
|
|
|
|
|
#'()
|
|
|
|
|
#'((define-syntax-parameter this-identifier
|
|
|
|
|
(lambda (s)
|
|
|
|
|
"Return the record being defined. This macro may
|
|
|
|
|
only be used in the context of the definition of a thunked field."
|
|
|
|
|
(syntax-case s ()
|
|
|
|
|
(id
|
|
|
|
|
(identifier? #'id)
|
|
|
|
|
(syntax-violation 'this-identifier
|
|
|
|
|
"cannot be used outside \
|
|
|
|
|
of a record instantiation"
|
|
|
|
|
#'id)))))))
|
2015-06-11 17:28:13 -04:00
|
|
|
|
thunked-field-accessor ...
|
|
|
|
|
delayed-field-accessor ...
|
2015-06-11 15:37:49 -04:00
|
|
|
|
(make-syntactic-constructor type syntactic-ctor ctor
|
|
|
|
|
(field ...)
|
2018-05-16 04:05:24 -04:00
|
|
|
|
#:abi-cookie #,cookie
|
2015-06-11 15:37:49 -04:00
|
|
|
|
#:thunked #,thunked
|
2019-03-29 17:40:55 -04:00
|
|
|
|
#:this-identifier #'this-identifier
|
2015-06-11 15:37:49 -04:00
|
|
|
|
#:delayed #,delayed
|
2015-06-11 16:57:33 -04:00
|
|
|
|
#:innate #,innate
|
2021-05-20 09:40:55 -04:00
|
|
|
|
#:sanitizers #,sanitizers
|
2019-03-29 17:40:55 -04:00
|
|
|
|
#:defaults #,defaults)))))
|
|
|
|
|
((_ type syntactic-ctor ctor pred
|
|
|
|
|
(field get properties ...) ...)
|
|
|
|
|
;; When no 'this' identifier was specified, use 'this-record'.
|
|
|
|
|
#'(define-record-type* type syntactic-ctor ctor pred
|
|
|
|
|
this-record
|
|
|
|
|
(field get properties ...) ...)))))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
|
2013-07-10 12:04:08 -04:00
|
|
|
|
(define* (alist->record alist make keys
|
|
|
|
|
#:optional (multiple-value-keys '()))
|
|
|
|
|
"Apply MAKE to the values associated with KEYS in ALIST. Items in KEYS that
|
|
|
|
|
are also in MULTIPLE-VALUE-KEYS are considered to occur possibly multiple
|
|
|
|
|
times in ALIST, and thus their value is a list."
|
|
|
|
|
(let ((args (map (lambda (key)
|
|
|
|
|
(if (member key multiple-value-keys)
|
|
|
|
|
(filter-map (match-lambda
|
|
|
|
|
((k . v)
|
|
|
|
|
(and (equal? k key) v)))
|
|
|
|
|
alist)
|
|
|
|
|
(assoc-ref alist key)))
|
|
|
|
|
keys)))
|
2013-05-12 09:46:16 -04:00
|
|
|
|
(apply make args)))
|
|
|
|
|
|
|
|
|
|
(define (object->fields object fields port)
|
|
|
|
|
"Write OBJECT (typically a record) as a series of recutils-style fields to
|
|
|
|
|
PORT, according to FIELDS. FIELDS must be a list of field name/getter pairs."
|
|
|
|
|
(let loop ((fields fields))
|
|
|
|
|
(match fields
|
|
|
|
|
(()
|
|
|
|
|
object)
|
|
|
|
|
(((field . get) rest ...)
|
|
|
|
|
(format port "~a: ~a~%" field (get object))
|
|
|
|
|
(loop rest)))))
|
|
|
|
|
|
2015-01-08 19:07:57 -05:00
|
|
|
|
(define %recutils-field-charset
|
|
|
|
|
;; Valid characters starting a recutils field.
|
|
|
|
|
;; info "(recutils) Fields"
|
|
|
|
|
(char-set-union char-set:upper-case
|
|
|
|
|
char-set:lower-case
|
|
|
|
|
(char-set #\%)))
|
2013-07-10 12:26:46 -04:00
|
|
|
|
|
2013-07-10 10:54:17 -04:00
|
|
|
|
(define (recutils->alist port)
|
|
|
|
|
"Read a recutils-style record from PORT and return it as a list of key/value
|
|
|
|
|
pairs. Stop upon an empty line (after consuming it) or EOF."
|
|
|
|
|
(let loop ((line (read-line port))
|
|
|
|
|
(result '()))
|
2013-07-10 11:01:47 -04:00
|
|
|
|
(cond ((eof-object? line)
|
2013-07-10 10:54:17 -04:00
|
|
|
|
(reverse result))
|
2013-07-10 11:01:47 -04:00
|
|
|
|
((string-null? line)
|
|
|
|
|
(if (null? result)
|
|
|
|
|
(loop (read-line port) result) ; leading space: ignore it
|
|
|
|
|
(reverse result))) ; end-of-record marker
|
2013-07-10 10:54:17 -04:00
|
|
|
|
(else
|
2015-01-08 19:07:57 -05:00
|
|
|
|
;; Now check the first character of LINE, since that's what the
|
|
|
|
|
;; recutils manual says is enough.
|
|
|
|
|
(let ((first (string-ref line 0)))
|
|
|
|
|
(cond
|
|
|
|
|
((char-set-contains? %recutils-field-charset first)
|
|
|
|
|
(let* ((colon (string-index line #\:))
|
|
|
|
|
(field (string-take line colon))
|
|
|
|
|
(value (string-trim (string-drop line (+ 1 colon)))))
|
|
|
|
|
(loop (read-line port)
|
|
|
|
|
(alist-cons field value result))))
|
|
|
|
|
((eqv? first #\#) ;info "(recutils) Comments"
|
|
|
|
|
(loop (read-line port) result))
|
|
|
|
|
((eqv? first #\+) ;info "(recutils) Fields"
|
|
|
|
|
(let ((new-line (if (string-prefix? "+ " line)
|
|
|
|
|
(string-drop line 2)
|
|
|
|
|
(string-drop line 1))))
|
|
|
|
|
(match result
|
|
|
|
|
(((field . value) rest ...)
|
|
|
|
|
(loop (read-line port)
|
|
|
|
|
`((,field . ,(string-append value "\n" new-line))
|
|
|
|
|
,@rest))))))
|
|
|
|
|
(else
|
|
|
|
|
(error "unmatched line" line))))))))
|
2013-07-10 10:54:17 -04:00
|
|
|
|
|
2022-11-19 11:23:04 -05:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Pattern matching.
|
|
|
|
|
;;;
|
|
|
|
|
|
2023-04-28 15:19:03 -04:00
|
|
|
|
(define-syntax lookup-field+wrapper
|
2022-11-19 11:23:04 -05:00
|
|
|
|
(lambda (s)
|
2023-04-28 15:19:03 -04:00
|
|
|
|
"Look up FIELD in the given list and return both an expression that represents
|
|
|
|
|
its offset in the record and a procedure that wraps it to return its \"true\" value
|
|
|
|
|
(for instance, FORCE is returned in the case of a delayed field). RECORD is passed
|
|
|
|
|
to thunked values. Raise a syntax violation when the field is not found."
|
|
|
|
|
(syntax-case s (normal delayed thunked)
|
|
|
|
|
((_ record field offset ())
|
|
|
|
|
(syntax-violation 'match-record
|
|
|
|
|
"unknown record type field"
|
2023-06-04 03:47:36 -04:00
|
|
|
|
;; Attach the local source data to the field.
|
|
|
|
|
(datum->syntax #f (syntax->datum #'field) #:source s)))
|
2023-04-28 15:19:03 -04:00
|
|
|
|
((_ record field offset ((head normal) tail ...))
|
|
|
|
|
(free-identifier=? #'field #'head)
|
|
|
|
|
#'(values offset identity))
|
|
|
|
|
((_ record field offset ((head delayed) tail ...))
|
2022-11-19 11:23:04 -05:00
|
|
|
|
(free-identifier=? #'field #'head)
|
2023-04-28 15:19:03 -04:00
|
|
|
|
#'(values offset force))
|
|
|
|
|
((_ record field offset ((head thunked) tail ...))
|
|
|
|
|
(free-identifier=? #'field #'head)
|
|
|
|
|
#'(values offset (cut <> record)))
|
|
|
|
|
((_ record field offset (_ tail ...))
|
|
|
|
|
#'(lookup-field+wrapper record field
|
|
|
|
|
(+ 1 offset) (tail ...))))))
|
2022-11-19 11:23:04 -05:00
|
|
|
|
|
|
|
|
|
(define-syntax match-record-inner
|
|
|
|
|
(lambda (s)
|
|
|
|
|
(syntax-case s ()
|
2022-12-21 21:14:55 -05:00
|
|
|
|
((_ record type ((field variable) rest ...) body ...)
|
2023-04-28 15:19:03 -04:00
|
|
|
|
#'(let-syntax ((field-offset+wrapper
|
|
|
|
|
(syntax-rules ()
|
|
|
|
|
((_ f)
|
|
|
|
|
(lookup-field+wrapper record field 0 f)))))
|
2023-06-06 03:06:59 -04:00
|
|
|
|
(let*-values (((offset wrap)
|
|
|
|
|
(type (map-fields type match-record)
|
|
|
|
|
field-offset+wrapper))
|
|
|
|
|
((variable)
|
|
|
|
|
(wrap (struct-ref record offset))))
|
2022-11-19 11:23:04 -05:00
|
|
|
|
(match-record-inner record type (rest ...) body ...))))
|
2022-12-21 21:14:55 -05:00
|
|
|
|
((_ record type (field rest ...) body ...)
|
|
|
|
|
;; Redirect to the canonical form above.
|
|
|
|
|
#'(match-record-inner record type ((field field) rest ...) body ...))
|
2022-11-19 11:23:04 -05:00
|
|
|
|
((_ record type () body ...)
|
|
|
|
|
#'(begin body ...)))))
|
|
|
|
|
|
2017-10-25 14:44:54 -04:00
|
|
|
|
(define-syntax match-record
|
|
|
|
|
(syntax-rules ()
|
|
|
|
|
"Bind each FIELD of a RECORD of the given TYPE to it's FIELD name.
|
2022-11-19 11:23:04 -05:00
|
|
|
|
The order in which fields appear does not matter. A syntax error is raised if
|
2023-04-28 15:19:03 -04:00
|
|
|
|
an unknown field is queried."
|
2022-11-19 11:23:04 -05:00
|
|
|
|
((_ record type (fields ...) body ...)
|
2017-10-25 14:44:54 -04:00
|
|
|
|
(if (eq? (struct-vtable record) type)
|
2022-11-19 11:23:04 -05:00
|
|
|
|
(match-record-inner record type (fields ...) body ...)
|
|
|
|
|
(throw 'wrong-type-arg record)))))
|
2017-10-25 14:44:54 -04:00
|
|
|
|
|
2023-04-28 15:19:05 -04:00
|
|
|
|
(define-syntax match-record-lambda
|
|
|
|
|
(syntax-rules ()
|
|
|
|
|
"Return a procedure accepting a single record of the given TYPE for which each
|
|
|
|
|
FIELD will be bound to its FIELD name within the returned procedure. A syntax error
|
|
|
|
|
is raised if an unknown field is queried."
|
|
|
|
|
((_ type (field ...) body ...)
|
|
|
|
|
(lambda (record)
|
|
|
|
|
(if (eq? (struct-vtable record) type)
|
|
|
|
|
(match-record-inner record type (field ...) body ...)
|
|
|
|
|
(throw 'wrong-type-arg record))))))
|
|
|
|
|
|
2013-05-12 09:46:16 -04:00
|
|
|
|
;;; records.scm ends here
|