Some checks failed
Test, Build, and Deploy / test-build-deploy (push) Has been cancelled
183 lines
6.0 KiB
Plaintext
183 lines
6.0 KiB
Plaintext
;; Smalltalk runtime tests — class table, type→class mapping, instances.
|
|
;;
|
|
;; Reuses helpers (st-test, st-deep=?) from tokenize.sx. Counters reset
|
|
;; here so this file's summary covers runtime tests only.
|
|
|
|
(set! st-test-pass 0)
|
|
(set! st-test-fail 0)
|
|
(set! st-test-fails (list))
|
|
|
|
;; Fresh hierarchy for every test file.
|
|
(st-bootstrap-classes!)
|
|
|
|
;; ── 1. Bootstrap installed expected classes ──
|
|
(st-test "Object exists" (st-class-exists? "Object") true)
|
|
(st-test "Behavior exists" (st-class-exists? "Behavior") true)
|
|
(st-test "Metaclass exists" (st-class-exists? "Metaclass") true)
|
|
(st-test "True/False/UndefinedObject"
|
|
(and
|
|
(st-class-exists? "True")
|
|
(st-class-exists? "False")
|
|
(st-class-exists? "UndefinedObject"))
|
|
true)
|
|
(st-test "SmallInteger / Float / Symbol exist"
|
|
(and
|
|
(st-class-exists? "SmallInteger")
|
|
(st-class-exists? "Float")
|
|
(st-class-exists? "Symbol"))
|
|
true)
|
|
(st-test "BlockClosure exists" (st-class-exists? "BlockClosure") true)
|
|
|
|
;; ── 2. Superclass chain ──
|
|
(st-test "Object has no superclass" (st-class-superclass "Object") nil)
|
|
(st-test "Behavior super = Object" (st-class-superclass "Behavior") "Object")
|
|
(st-test "True super = Boolean" (st-class-superclass "True") "Boolean")
|
|
(st-test "Symbol super = String" (st-class-superclass "Symbol") "String")
|
|
(st-test
|
|
"String chain"
|
|
(st-class-chain "String")
|
|
(list "String" "ArrayedCollection" "SequenceableCollection" "Collection" "Object"))
|
|
(st-test
|
|
"SmallInteger chain"
|
|
(st-class-chain "SmallInteger")
|
|
(list "SmallInteger" "Integer" "Number" "Magnitude" "Object"))
|
|
|
|
;; ── 3. inherits-from? ──
|
|
(st-test "True inherits from Boolean" (st-class-inherits-from? "True" "Boolean") true)
|
|
(st-test "True inherits from Object" (st-class-inherits-from? "True" "Object") true)
|
|
(st-test "True inherits from True" (st-class-inherits-from? "True" "True") true)
|
|
(st-test
|
|
"True does not inherit from Number"
|
|
(st-class-inherits-from? "True" "Number")
|
|
false)
|
|
(st-test
|
|
"Object does not inherit from Number"
|
|
(st-class-inherits-from? "Object" "Number")
|
|
false)
|
|
|
|
;; ── 4. type→class mapping ──
|
|
(st-test "class-of nil" (st-class-of nil) "UndefinedObject")
|
|
(st-test "class-of true" (st-class-of true) "True")
|
|
(st-test "class-of false" (st-class-of false) "False")
|
|
(st-test "class-of int" (st-class-of 42) "SmallInteger")
|
|
(st-test "class-of zero" (st-class-of 0) "SmallInteger")
|
|
(st-test "class-of negative int" (st-class-of -3) "SmallInteger")
|
|
(st-test "class-of float" (st-class-of 3.14) "Float")
|
|
(st-test "class-of string" (st-class-of "hi") "String")
|
|
(st-test "class-of symbol" (st-class-of (quote foo)) "Symbol")
|
|
(st-test "class-of list" (st-class-of (list 1 2)) "Array")
|
|
(st-test "class-of empty list" (st-class-of (list)) "Array")
|
|
(st-test "class-of lambda" (st-class-of (fn (x) x)) "BlockClosure")
|
|
(st-test "class-of dict" (st-class-of {:a 1}) "Dictionary")
|
|
|
|
;; ── 5. User class definition ──
|
|
(st-class-define! "Account" "Object" (list "balance" "owner"))
|
|
(st-class-define! "SavingsAccount" "Account" (list "rate"))
|
|
|
|
(st-test "Account exists" (st-class-exists? "Account") true)
|
|
(st-test "Account super = Object" (st-class-superclass "Account") "Object")
|
|
(st-test
|
|
"SavingsAccount chain"
|
|
(st-class-chain "SavingsAccount")
|
|
(list "SavingsAccount" "Account" "Object"))
|
|
(st-test
|
|
"SavingsAccount own ivars"
|
|
(get (st-class-get "SavingsAccount") :ivars)
|
|
(list "rate"))
|
|
(st-test
|
|
"SavingsAccount inherited+own ivars"
|
|
(st-class-all-ivars "SavingsAccount")
|
|
(list "balance" "owner" "rate"))
|
|
|
|
;; ── 6. Instance construction ──
|
|
(define a1 (st-make-instance "Account"))
|
|
(st-test "instance is st-instance" (st-instance? a1) true)
|
|
(st-test "instance class" (get a1 :class) "Account")
|
|
(st-test "instance ivars start nil" (st-iv-get a1 "balance") nil)
|
|
(st-test
|
|
"instance has all expected ivars"
|
|
(sort (keys (get a1 :ivars)))
|
|
(sort (list "balance" "owner")))
|
|
(define a2 (st-iv-set! a1 "balance" 100))
|
|
(st-test "iv-set! returns updated copy" (st-iv-get a2 "balance") 100)
|
|
(st-test "iv-set! does not mutate original" (st-iv-get a1 "balance") nil)
|
|
(st-test "class-of instance" (st-class-of a1) "Account")
|
|
|
|
(define s1 (st-make-instance "SavingsAccount"))
|
|
(st-test
|
|
"subclass instance has all inherited ivars"
|
|
(sort (keys (get s1 :ivars)))
|
|
(sort (list "balance" "owner" "rate")))
|
|
|
|
;; ── 7. Method install + lookup ──
|
|
(st-class-add-method!
|
|
"Account"
|
|
"balance"
|
|
(st-parse-method "balance ^ balance"))
|
|
(st-class-add-method!
|
|
"Account"
|
|
"deposit:"
|
|
(st-parse-method "deposit: amount balance := balance + amount. ^ self"))
|
|
|
|
(st-test
|
|
"method registered"
|
|
(has-key? (get (st-class-get "Account") :methods) "balance")
|
|
true)
|
|
|
|
(st-test
|
|
"method lookup direct"
|
|
(= (st-method-lookup "Account" "balance" false) nil)
|
|
false)
|
|
|
|
(st-test
|
|
"method lookup walks superclass"
|
|
(= (st-method-lookup "SavingsAccount" "deposit:" false) nil)
|
|
false)
|
|
|
|
(st-test
|
|
"method lookup unknown selector"
|
|
(st-method-lookup "Account" "frobnicate" false)
|
|
nil)
|
|
|
|
(st-test
|
|
"method lookup records defining class"
|
|
(get (st-method-lookup "SavingsAccount" "balance" false) :defining-class)
|
|
"Account")
|
|
|
|
;; SavingsAccount overrides deposit:
|
|
(st-class-add-method!
|
|
"SavingsAccount"
|
|
"deposit:"
|
|
(st-parse-method "deposit: amount ^ super deposit: amount + 1"))
|
|
|
|
(st-test
|
|
"subclass override picked first"
|
|
(get (st-method-lookup "SavingsAccount" "deposit:" false) :defining-class)
|
|
"SavingsAccount")
|
|
|
|
(st-test
|
|
"Account still finds its own deposit:"
|
|
(get (st-method-lookup "Account" "deposit:" false) :defining-class)
|
|
"Account")
|
|
|
|
;; ── 8. Class-side methods ──
|
|
(st-class-add-class-method!
|
|
"Account"
|
|
"new"
|
|
(st-parse-method "new ^ super new"))
|
|
(st-test
|
|
"class-side lookup"
|
|
(= (st-method-lookup "Account" "new" true) nil)
|
|
false)
|
|
(st-test
|
|
"instance-side does not find class method"
|
|
(st-method-lookup "Account" "new" false)
|
|
nil)
|
|
|
|
;; ── 9. Re-bootstrap resets table ──
|
|
(st-bootstrap-classes!)
|
|
(st-test "after re-bootstrap Account gone" (st-class-exists? "Account") false)
|
|
(st-test "after re-bootstrap Object stays" (st-class-exists? "Object") true)
|
|
|
|
(list st-test-pass st-test-fail)
|