2.6 Pattern-based Updating
(require ocular-patdown/update) | package: ocular-patdown |
syntax
(update target-expr clause ...)
clause = [pat body ...+]
Like match for performing immutable updates. Also useful for creating composed optics.
Patterns act like trees of optic compositions (see optic-compose) with variables as leaves, binding composed optics.
> (struct posn [x y] #:transparent)
> (update (posn 1 2) [(struct-field posn x) (set! x 3)]) (posn 3 2)
> (update (list (posn 1 2) (posn 3 4)) [(list (struct-field posn x) p) (modify! x -) (set! p (posn 5 6))]) (list (posn -1 2) (posn 5 6))
> (update (list (posn 1 2) (posn 3 4) (posn 5 6)) [(list-of (struct-field posn x)) (modify! x -)]) (list (posn -1 2) (posn -3 4) (posn -5 6))
> (update (list 1 2 3) [(list a b) (error "boom")] [(list a b c) (set! c 4)]) '(1 2 4)
> (update (list 1 2) [(list a b) (get a)]) 1
> (update (list 1 2) [(list a b) (set! a (+ 4 (get b)))]) '(6 2)
2.6.1 Patterns
The grammar for patterns is as follows:
pat | = | _ | ||
| | id | |||
| | (and pat ...) | |||
| | (cons pat pat) | |||
| | (list pat ...) | |||
| | (list-of pat) | |||
| | (struct-field struct-id id pat) | |||
| | (struct-field struct-id id) | |||
| | (iso expr expr expr pat) | |||
| | (optic expr expr pat) |
More details on the different patterns:
syntax
syntax
(and pat ...)
syntax
(cons car-pat cdr-pat)
> (update (cons 1 2) [(cons a b) (set! a 3)]) '(3 . 2)
> (update (list #t #f) [(cons a (cons b _)) (set! b 'true)]) '(#t true)
syntax
(list pat ...)
> (update (list 1 2 3 4) [(list-of n) (modify! n sqr)]) '(1 4 9 16)
> (update '((1 2 3) (4 5 6) () (7)) [(list-of (list-of n)) (modify! n sqr)]) '((1 4 9) (16 25 36) () (49))
> (update (list (posn 1 2) (posn 3 4)) [(list-of (struct-field posn x)) (modify! x -)]) (list (posn -1 2) (posn -3 4))
The second form is shorthand for (struct-field struct-name field-name field-name).
Cannot be used on fields from a struct’s super type.
> (update (posn 1 2) [(struct-field posn x x-value) (set! x-value 3)]) (posn 3 2)
> (update (posn 1 2) [(struct-field posn x) (set! x 3)]) (posn 3 2)
> (struct posn3 posn [z] #:transparent)
> (update (posn3 3 4 5) [(struct-field posn3 z) (set! z 9)]) (posn3 3 4 9)
> (update (posn3 3 4 5) [(struct-field posn x) (set! x 9)]) (posn 9 4)
Naively trying to use a super type’s struct field to perform an update on an instance of the subtype will yield an instance of the super type.
> (update 'foo [(iso symbol? symbol->string string->symbol str) (modify! str string-upcase)]) 'FOO
2.6.2 Getter and Updater Utilities
get, optic-set!, and modify! are just ordinary procedures that operate on optics. The only thing that is special about them is that they know about the current target of an update.
parameter
(current-update-target target) → void? target : any/c
> (update (list 1 2) [(list a b) (current-update-target)]) '(1 2)
> (update (list 1 2) [(list a b) (set! a 4) (current-update-target)]) '(4 2)
> (update (cons 1 2) [(cons a _) (get a)]) 1
> (parameterize ([current-update-target (cons 1 2)]) (get car-lens)) 1
procedure
(optic-set! optic value) → any/c
optic : lens? value : any/c
Using set! on a variable bound by update will use optic-set!, so you can just use set! instead.
> (update (cons 1 2) [(cons a _) (optic-set! a #t)]) '(#t . 2)
> (update (cons 1 2) [(cons a _) (set! a #t)]) '(#t . 2)
> (current-update-target (cons 1 2)) > (optic-set! car-lens -1) '(-1 . 2)
> (current-update-target) '(-1 . 2)
> (update (cons 1 2) [(cons a _) (modify! a sub1)]) '(0 . 2)
> (current-update-target (cons 1 2)) > (modify! cdr-lens sqr) '(1 . 4)
> (current-update-target) '(1 . 4)
> (update (list 1 2 3) [(list-of a) (fold a + 0)]) 6
> (parameterize ([current-update-target (list 1 2 3)]) (fold list-traversal cons '())) '(3 2 1)
2.6.3 Extending Update
syntax
(define-update-syntax id transformer-expr)
(define-update-syntax (id arg-id ...) body-expr ...+)
transformer-expr : (-> syntax? syntax?)
(define-update-syntax id (lambda (arg-id ...) (begin body-expr ...)))
> (define-update-syntax posn (syntax-rules () [(posn x-pat y-pat) (and (struct-field posn x x-pat) (struct-field posn y y-pat))]))
> (update (posn 1 2) [(posn a b) (set! a 4) (modify! b -)]) (posn 4 -2)