Files
rose-ash/spec/harness.sx
giles 6f52cff0d9 W14: fix C22/K104 — harness logs IO before invoking the mock (+3 pins)
The interceptor appended the IO-log entry only after the mock returned, so
a throwing mock left no entry and error-path tests falsely reported "never
invoked" through assert-io-called/count (hosts.md C22, core.md K104).

spec/harness.sx make-interceptor now appends {:args :result nil :op}
BEFORE invoking the mock and updates :result in place via dict-set! on
return. This is W14-owned test infrastructure (PLAN.md W14 approach item
4), not a semantics edit.

Pins: suite gate-C22-throwing-mock-logged (throwing mock leaves an entry
with pending result; happy path updates the result; mixed throwing +
successful sequence counts all calls). Harness self-suite (15 tests) and
test-relate-picker (the only other harness consumer) verified green;
285/0 on the pins run.

Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
2026-07-04 02:07:24 +00:00

236 lines
6.9 KiB
Plaintext

;; Assert condition is truthy, error with message
(define-library
(sx harness)
(export
assert
assert=
default-platform
make-harness
harness-reset!
harness-log
harness-get
harness-set!
make-interceptor
install-interceptors
io-calls
io-call-count
io-call-nth
io-call-args
io-call-result
assert-io-called
assert-no-io
assert-io-count
assert-io-args
assert-io-result
assert-state)
(begin
(define
assert
(fn
(condition msg)
(when (not condition) (error (or msg "Assertion failed")))))
(define
assert=
(fn
(actual expected msg)
(when
(not (= actual expected))
(error (or msg (str "Expected " expected ", got " actual))))))
(define default-platform {:get-cookie (fn (name) nil) :local-storage-get (fn (key) nil) :emit-dom (fn (op &rest args) nil) :request-arg (fn (name) nil) :append-child (fn (parent child) nil) :remove-class (fn (el cls) nil) :action (fn (service name &rest args) {:ok true}) :abort (fn (code) nil) :now (fn () 0) :get-element (fn (id) nil) :add-class (fn (el cls) nil) :query (fn (service name &rest args) (list)) :fetch (fn (url &rest opts) {:ok true :body "" :status 200}) :remove-child (fn (parent child) nil) :set-text (fn (el text) nil) :set-attr (fn (el name val) nil) :config (fn (key) nil) :request-path (fn () "/") :create-element (fn (tag) nil) :url-for (fn (endpoint &rest args) "/mock-url") :set-cookie (fn (name val &rest opts) nil) :local-storage-set (fn (key val) nil) :sleep (fn (ms) nil) :frag (fn (service comp &rest args) "") :app-url (fn (service &rest path) "/mock-app-url") :csrf-token (fn () "test-csrf-token") :current-user (fn () nil)})
(define
make-harness
:effects ()
(fn
(&key platform)
(let
((merged (if (nil? platform) default-platform (merge default-platform platform))))
{:state {:dom nil :storage {} :cookies {}} :platform merged :log (list)})))
(define
harness-reset!
:effects ()
(fn
(session)
(dict-set! session "log" (list))
(dict-set! session "state" {:dom nil :storage {} :cookies {}})
session))
(define
harness-log
:effects ()
(fn
(session &key op)
(let
((log (get session "log")))
(if
(nil? op)
log
(filter (fn (entry) (= (get entry "op") op)) log)))))
(define
harness-get
:effects ()
(fn (session key) (get (get session "state") key)))
(define
harness-set!
:effects ()
(fn
(session key value)
(dict-set! (get session "state") key value)
nil))
(define
make-interceptor
:effects ()
(fn
(session op-name mock-fn)
(fn
(&rest args)
(let
((entry {:op op-name :result nil :args args}) (log (get session "log")))
(append! log entry)
(let
((result (if (empty? args) (mock-fn) (if (= 1 (len args)) (mock-fn (first args)) (if (= 2 (len args)) (mock-fn (first args) (nth args 1)) (if (= 3 (len args)) (mock-fn (first args) (nth args 1) (nth args 2)) (apply mock-fn args)))))))
(dict-set! entry "result" result)
result)))))
(define
install-interceptors
:effects ()
(fn
(session env)
(for-each
(fn
(key)
(let
((mock-fn (get (get session "platform") key))
(interceptor (make-interceptor session key mock-fn)))
(env-bind! env key interceptor)))
(keys (get session "platform")))
env))
(define
io-calls
:effects ()
(fn
(session op-name)
(filter
(fn (entry) (= (get entry "op") op-name))
(get session "log"))))
(define
io-call-count
:effects ()
(fn (session op-name) (len (io-calls session op-name))))
(define
io-call-nth
:effects ()
(fn
(session op-name n)
(let
((calls (io-calls session op-name)))
(if (< n (len calls)) (nth calls n) nil))))
(define
io-call-args
:effects ()
(fn
(session op-name n)
(let
((call (io-call-nth session op-name n)))
(if (nil? call) nil (get call "args")))))
(define
io-call-result
:effects ()
(fn
(session op-name n)
(let
((call (io-call-nth session op-name n)))
(if (nil? call) nil (get call "result")))))
(define
assert-io-called
:effects ()
(fn
(session op-name)
(assert
(> (io-call-count session op-name) 0)
(str "Expected IO operation " op-name " to be called but it was not"))))
(define
assert-no-io
:effects ()
(fn
(session op-name)
(assert
(= (io-call-count session op-name) 0)
(str
"Expected IO operation "
op-name
" not to be called but it was called "
(io-call-count session op-name)
" time(s)"))))
(define
assert-io-count
:effects ()
(fn
(session op-name expected)
(let
((actual (io-call-count session op-name)))
(assert
(= actual expected)
(str
"Expected "
op-name
" to be called "
expected
" time(s) but was called "
actual
" time(s)")))))
(define
assert-io-args
:effects ()
(fn
(session op-name n expected-args)
(let
((actual (io-call-args session op-name n)))
(assert
(equal? actual expected-args)
(str
"Expected call "
n
" to "
op-name
" with args "
(str expected-args)
" but got "
(str actual))))))
(define
assert-io-result
:effects ()
(fn
(session op-name n expected)
(let
((actual (io-call-result session op-name n)))
(assert
(equal? actual expected)
(str
"Expected call "
n
" to "
op-name
" to return "
(str expected)
" but got "
(str actual))))))
(define
assert-state
:effects ()
(fn
(session key expected)
(let
((actual (harness-get session key)))
(assert
(equal? actual expected)
(str
"Expected state "
key
" to be "
(str expected)
" but got "
(str actual)))))))) ;; end define-library
;; Re-export to global namespace for backward compatibility
(import (sx harness))