Files
rose-ash/lib/smalltalk/tests/runtime.sx
giles 52523606a8
Some checks failed
Test, Build, and Deploy / test-build-deploy (push) Has been cancelled
smalltalk: class table + bootstrap hierarchy + 54 tests
2026-04-25 01:34:59 +00:00

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)