3.2 Solution: Objects

This solution shows how numbers are interpreted as objects and messages to these numbers might include symbols such as +. Consider extending this solution with some of the following:
  • recognize stuck states for expressions such as (send n + (object ...)), (send (object ...) + n), or (send o m v) where o does not have an entry point labeled m.

  • a clone operation for objects

  • an update operation for objects that adds a new method

  • and the inclusion of fields.

  #lang racket
   
  ;; a model of simple object programming (no updater, no prototype, no clone)
   
  (require redex (only-in "common.rkt" in))
   
  ;; -----------------------------------------------------------------------------
  ;; syntax
   
  (define-language Object
    (e ::= n y (object (m (x) e) ...) (send e m e))
    (y ::= x this)
    (n ::= natural)
    (m ::= variable-not-otherwise-mentioned)
    (x ::= variable-not-otherwise-mentioned))
   
  ;; -----------------------------------------------------------------------------
  ;; examples
   
  (define help
    (term (object [help (x) x])))
   
  (define p-good
    (term
     (send
      (object [get(x) this]
              [set(x) x])
      set
      ,help)))
   
  (define p-8
    (term
     (send (object [get(x) this] [set(x) (send x + 3)]) set 5)))
   
  (module+ test
    (test-equal (redex-match? Object e help) #true)
    (test-equal (redex-match? Object e p-good) #true))
   
  ;; -----------------------------------------------------------------------------
  ;; scope
   
  ;; (=α e_1 e_2) determines whether e_1 and e_2 are α equivalent
  (module+ test
    (test-equal (term ( (object (help (x) x)) (object (help (y) y)) )) #true)
    (test-equal (term ( (object (help (x) x)) (object (main (y) y)) )) #false))
   
  (define-metafunction Object
     : any any -> boolean
    [( any_1 any_2) ,(equal? (term (sd any_1)) (term (sd any_2)))])
   
  ;; a Racket definition for use in Racket positions 
  (define (=α/racket x y) (term ( ,x ,y)))
   
  ;; (sd e) computes the static distance version of e
  (define-extended-language SD Object
    (e ::= .... (K n))
    (n ::= natural))
   
  (define SD? (redex-match? SD e))
   
  (module+ test
    (define sd1 (term (K 1)))
    (define sd2 (term 1))
    
    (test-equal (SD? sd1) #true))
   
  (define-metafunction SD
    sd : any -> any
    [(sd any_1) (sd/a any_1 ())])
   
  (module+ test
    (define help-sd
      (term (object [help () (K 0)])))
    (define p-good-sd
      (term
       (send
        (object [get(x) this]
                [set(x) (K 0)])
        set
        ,help-sd)))
    (test-equal (term (sd/a x ())) (term x))
    (test-equal (term (sd/a x (y z x))) (term (K 2)))
    (test-equal (term (sd ,help)) help-sd))
   
  (define-metafunction SD
    sd/a : any (x ...) -> any
    ;; bound variable 
    [(sd/a x (x_1 ... x x_2 ...))
     (K n_rib)
     (where n_rib ,(length (term (x_1 ...))))
     (where #false (in x (x_1 ...)))]
    ;; free variable 
    [(sd/a x (x_1 ...)) x]
    [(sd/a (object (m (x) any_1) ...) (any_rest ...))
     (object (m () (sd/a any_1 (x any_rest ...))) ...)]
    [(sd/a (send any_fun m any_arg) (any_rib ...))
     (send (sd/a any_fun (any_rib ...)) m (sd/a any_arg (any_rib ...)))]
    [(sd/a any (x_1 ...)) any])
   
  ;; -----------------------------------------------------------------------------
  ;; substitution
   
  ;; (subst ([e x] ...) e_*) substitutes e ... for x ... in e_* (hygienically)
  (module+ test
    (test-equal (term (subst ([1 x][2 y]) x)) 1)
    (test-equal (term (subst ([1 x][2 y]) y)) 2)
    (test-equal (term (subst ([1 x][2 y]) z)) (term z))
    (test-equal
     (term (subst ([1 x][2 y]) (object (m (z) x))))
     (term (object (m (z) 1)))
     #:equiv =α/racket)
    (test-equal
     (term (subst ([1 x][2 y]) (object (m (z) (object (n (x) x))))))
     (term (object (m (z) (object (n (x) x)))))
     #:equiv =α/racket)
    (test-equal
     (term (subst ([1 x][2 y]) (object (m (z) (object (n (y) x))))))
     (term (object (m (z) (object (n (y) 1)))))
     #:equiv =α/racket)
    (test-equal
     (term (subst ([(object (x) y) x][2 y]) (object (m (z) (object (n (y) x))))))
     (term (object (m (z) (object (n (y1) (object (x) y))))))
     #:equiv =α/racket)
    
    (test-equal
     (term
      (subst ([(object (help (x) x)) x]
              [(object (get (x) this) (set (x) x)) this])
             x))
     help
     #:equiv =α/racket))
   
  (define-metafunction Object
    subst : ((any y) ...) any -> any
    [(subst [(any_1 y_1) ... (any_x x) (any_2 y_2) ...] x) any_x]
    [(subst [(any_1 y_1) ... ] x) x]
    [(subst [(any_1 y_1) ... ] (object (m (x) any_m) ...))
     (object
      (m (y_new) (subst ((any_1 y_1) ...) (subst-raw ((y_new x) ...) any_m))) ...)
     (where (y_new ...) (fresh-in any_m ... any_1 ... (x ...)))]
    [(subst [(any_1 y_1) ... ] (any ...)) ((subst [(any_1 y_1) ... ] any) ...)]
    [(subst [(any_1 y_1) ... ] any_*) any_*])
   
  (define-metafunction Object
    subst-raw : ((y y) ...) any -> any
    [(subst-raw ((y_n1 y_o1) ... (y_new x) (y_n2 y_o2) ...) x) y_new]
    [(subst-raw ((y_n1 y_o1) ... ) x) x]
    [(subst-raw ((y_n1 y_o1) ... ) (object (m (x) any_m) ...))
     (object (m (x) (subst-raw ((y_n1 y_o1) ... ) any_m)) ...)]
    [(subst-raw [(any_1 y_1) ... ] (any ...))
     ((subst-raw [(any_1 y_1) ... ] any) ...)]
    [(subst-raw [(any_1 y_1) ... ] any_*) any_*])
   
  ;; (fresh-in any ... (x ...)) generates a sequence of variables
  ;; like x ... not in any ...
  (define-metafunction Object
    fresh-in : any ... (x ...) -> (x ...)
    [(fresh-in any ... (x ...))
     ,(variables-not-in (term (any ...)) (term (x ...)))])
   
  ;; -----------------------------------------------------------------------------
  ;; the object calculus (standard reduction)
   
  (define-extended-language Object-calculus Object
    (v ::= n (object (m (x) e) ...))
    (E ::= hole (send E m e) (send v m E)))
   
  (module+ test
    #;
    (traces -->obj p-good)
    (test-->> -->obj #:equiv =α/racket p-good help)
    (test-->> -->obj #:equiv =α/racket p-8 8))
   
  (define -->obj 
    (reduction-relation
     Object-calculus
     (--> (in-hole E (send (name THIS
                                 (object (m_left (x_left) e_left) ...
                                         (m (x) e)
                                         (m_right (x_right) e_right) ...))
                           m
                           v))
          (in-hole E (subst ([v x][THIS this]) e))
          send)
     (--> (in-hole E (send n_1 + n_2))
          (in-hole E ,(+ (term n_1) (term n_2)))
          add)))
   
  ;; -----------------------------------------------------------------------------
  (module+ test
    (test-results))
   
   
   
  ;;; ------------------------------------------------------------
  ;;; common.rkt starts here
   
  #lang racket
   
  ;; basic definitions for the Redex Summer School 2015
   
  (provide
   ;; Language 
   Lambda
   
   ;; Any -> Boolean
   ;; is the given value in the expression language? 
   lambda?
   
   ;; x (x ...) -> Boolean
   ;; (in x (x_1 ...)) determines whether x occurs in x_1 ...
   in
   
   ;; Any Any -> Boolean
   ;; (=α/racket e_1 e_2) determines whether e_1 is α-equivalent to e_2
   ;; e_1, e_2 are in Lambda or extensions of Lambda that 
   ;; do not introduce binding constructs beyond lambda 
   =α/racket
   
   ;; ((Lambda x) ...) Lambda -> Lambda
   ;; (subs ((e_1 x_1) ...) e) substitures e_1 for x_1 ... in e
   ;; e_1, ... e are in Lambda or extensions of Lambda that 
   ;; do not introduce binding constructs beyond lambda 
   subst)
   
  ;; -----------------------------------------------------------------------------
  (require redex)
   
  (define-language Lambda
    (e ::=
       x 
       (lambda (x_!_ ...) e)
       (e e ...))
    (x ::= variable-not-otherwise-mentioned))
   
  (define lambda? (redex-match? Lambda e))
   
  (module+ test
    (define e1 (term y))
    (define e2 (term (lambda (y) y)))
    (define e3 (term (lambda (x y) y)))
    (define e4 (term (,e2 e3)))
   
    (test-equal (lambda? e1) #true)
    (test-equal (lambda? e2) #true)
    (test-equal (lambda? e3) #true)
    (test-equal (lambda? e4) #true)
   
    (define eb1 (term (lambda (x x) y)))
    (define eb2 (term (lambda (x y) 3)))
   
    (test-equal (lambda? eb1) #false)
    (test-equal (lambda? eb2) #false))
   
  ;; -----------------------------------------------------------------------------
  ;; (in x x_1 ...) is x a member of (x_1 ...)?
   
  (module+ test
    (test-equal (term (in x (y z x y z))) #true)
    (test-equal (term (in x ())) #false)
    (test-equal (term (in x (y z w))) #false))
   
  (define-metafunction Lambda
    in : x (x ...) -> boolean
    [(in x (x_1 ... x x_2 ...)) #true]
    [(in x (x_1 ...)) #false])
   
  ;; -----------------------------------------------------------------------------
  ;; (=α e_1 e_2) determines whether e_1 and e_2 are α equivalent
   
  (module+ test
    (test-equal (term ( (lambda (x) x) (lambda (y) y))) #true)
    (test-equal (term ( (lambda (x) (x 1)) (lambda (y) (y 1)))) #true)
    (test-equal (term ( (lambda (x) x) (lambda (y) z))) #false))
   
  (define-metafunction Lambda
     : any any -> boolean
    [( any_1 any_2) ,(equal? (term (sd any_1)) (term (sd any_2)))])
   
  ;; a Racket definition for use in Racket positions 
  (define (=α/racket x y) (term ( ,x ,y)))
   
  ;; (sd e) computes the static distance version of e
  (define-extended-language SD Lambda
    (e ::= .... (K n))
    (n ::= natural))
   
  (define SD? (redex-match? SD e))
   
  (module+ test
    (define sd1 (term (K 1)))
    (define sd2 (term 1))
   
    (test-equal (SD? sd1) #true))
   
  (define-metafunction SD
    sd : any -> any
    [(sd any_1) (sd/a any_1 ())])
   
  (module+ test
    (test-equal (term (sd/a x ())) (term x))
    (test-equal (term (sd/a x ((y) (z) (x)))) (term (K 2 0)))
    (test-equal (term (sd/a ((lambda (x) x) (lambda (y) y)) ()))
                (term ((lambda () (K 0 0)) (lambda () (K 0 0)))))
    (test-equal (term (sd/a (lambda (x) (x (lambda (y) y))) ()))
                (term (lambda () ((K 0 0) (lambda () (K 0 0))))))
    (test-equal (term (sd/a (lambda (z x) (x (lambda (y) z))) ()))
                (term (lambda () ((K 0 1) (lambda () (K 1 0)))))))
   
  (define-metafunction SD
    sd/a : any ((x ...) ...) -> any
    [(sd/a x ((x_1 ...) ... (x_0 ... x x_2 ...) (x_3 ...) ...))
     ;; bound variable 
     (K n_rib n_pos)
     (where n_rib ,(length (term ((x_1 ...) ...))))
     (where n_pos ,(length (term (x_0 ...))))
     (where #false (in x (x_1 ... ...)))]
    [(sd/a (lambda (x ...) any_1) (any_rest ...))
     (lambda () (sd/a any_1 ((x ...) any_rest ...)))]
    [(sd/a (any_fun any_arg ...) (any_rib ...))
     ((sd/a any_fun (any_rib ...)) (sd/a any_arg (any_rib ...)) ...)]
    [(sd/a any_1 any)
     ;; free variable, constant, etc 
     any_1])
   
   
  ;; -----------------------------------------------------------------------------
  ;; (subst ([e x] ...) e_*) substitutes e ... for x ... in e_* (hygienically)
   
  (module+ test
    (test-equal (term (subst ([1 x][2 y]) x)) 1)
    (test-equal (term (subst ([1 x][2 y]) y)) 2)
    (test-equal (term (subst ([1 x][2 y]) z)) (term z))
    (test-equal (term (subst ([1 x][2 y]) (lambda (z w) (x y))))
                (term (lambda (z w) (1 2))))
    (test-equal (term (subst ([1 x][2 y]) (lambda (z w) (lambda (x) (x y)))))
                (term (lambda (z w) (lambda (x) (x 2))))
                #:equiv =α/racket)
    (test-equal (term (subst ((2 x)) ((lambda (x) (1 x)) x)))
                (term ((lambda (x) (1 x)) 2))
                #:equiv =α/racket)
    (test-equal (term (subst (((lambda (x) y) x)) (lambda (y) x)))
                (term (lambda (y1) (lambda (x) y)))
                #:equiv =α/racket))
   
  (define-metafunction Lambda
    subst : ((any x) ...) any -> any
    [(subst [(any_1 x_1) ... (any_x x) (any_2 x_2) ...] x) any_x]
    [(subst [(any_1 x_1) ... ] x) x]
    [(subst [(any_1 x_1) ... ] (lambda (x ...) any_body))
     (lambda (x_new ...)
       (subst ((any_1 x_1) ...)
              (subst-raw ((x_new x) ...) any_body)))
     (where  (x_new ...)  ,(variables-not-in (term (any_body any_1 ...)) (term (x ...)))) ]
    [(subst [(any_1 x_1) ... ] (any ...)) ((subst [(any_1 x_1) ... ] any) ...)]
    [(subst [(any_1 x_1) ... ] any_*) any_*])
   
  (define-metafunction Lambda
    subst-raw : ((x x) ...) any -> any
    [(subst-raw ((x_n1 x_o1) ... (x_new x) (x_n2 x_o2) ...) x) x_new]
    [(subst-raw ((x_n1 x_o1) ... ) x) x]
    [(subst-raw ((x_n1 x_o1) ... ) (lambda (x ...) any))
     (lambda (x ...) (subst-raw ((x_n1 x_o1) ... ) any))]
    [(subst-raw [(any_1 x_1) ... ] (any ...))
     ((subst-raw [(any_1 x_1) ... ] any) ...)]
    [(subst-raw [(any_1 x_1) ... ] any_*) any_*])
   
  ;; -----------------------------------------------------------------------------
  (module+ test
    (test-results))