artdag: job-as-post-object projection (post.sx) + 12 tests
Some checks failed
Test, Build, and Deploy / test-build-deploy (push) Failing after 47s

lib/artdag/post.sx — the artdag-side projection for 'a job is a type of post' (per the
host loop). job->post-object: {:type artdag/job :id <output content-id> :wire <dag->wire>},
post-id = content-id = natural AP object id. post-object-verify binds the id to the payload
(record ids recompute + post id present), rejecting tampered params/bogus ids. String
transport for the feed/SXTP body; post-run lets a peer decode->run->result, content-address
cache-hitting. Activity wrapping stays host-side. post 12/12, total 225/225.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-06-28 18:38:34 +00:00
parent 550d0db5a5
commit 0b7b3b9efb
6 changed files with 208 additions and 9 deletions

View File

@@ -13,7 +13,7 @@ if [ ! -x "$SX_SERVER" ]; then
exit 1 exit 1
fi fi
SUITES=(dag analyze plan execute optimize fed cost serialize stats fault maude-optimize schedule) SUITES=(dag analyze plan execute optimize fed cost serialize stats fault post maude-optimize schedule)
OUT_JSON="lib/artdag/scoreboard.json" OUT_JSON="lib/artdag/scoreboard.json"
OUT_MD="lib/artdag/scoreboard.md" OUT_MD="lib/artdag/scoreboard.md"
@@ -96,6 +96,7 @@ ${MK_LOADS}
(load "lib/artdag/serialize.sx") (load "lib/artdag/serialize.sx")
(load "lib/artdag/stats.sx") (load "lib/artdag/stats.sx")
(load "lib/artdag/fault.sx") (load "lib/artdag/fault.sx")
(load "lib/artdag/post.sx")
(load "lib/artdag/api.sx") (load "lib/artdag/api.sx")
${BRIDGE_LOAD} ${BRIDGE_LOAD}
${SCHED_LOAD} ${SCHED_LOAD}

68
lib/artdag/post.sx Normal file
View File

@@ -0,0 +1,68 @@
; lib/artdag/post.sx — project an artdag job to/from a feed "post object", so a job
; can ride as the :object of a feed activity ({:actor :verb :object :at :tags}) per the
; host loop. A post object is content-addressed and self-verifying:
; {:type "artdag/job" :id <content-id of the output node> :wire <dag->wire>}
; The :id IS the post/object id (the stable structural digest = natural AP object id);
; the :wire is the self-describing, write/read-safe payload from serialize.sx whose
; records each carry their own content-id. The dag<->feed-activity wrapping (actor/verb/
; at/tags) stays on the host/feed side; this file is only the job<->object projection.
; Depends on dag.sx + serialize.sx (and execute.sx for post-run).
(define artdag/post-type "artdag/job")
; a job = a dag + the output node (by author name) the post is "about".
(define artdag/job->post-object (fn (dag output-name) {:id (artdag/dag-id dag output-name) :type artdag/post-type :wire (artdag/dag->wire dag)}))
(define
artdag/post-object?
(fn
(x)
(and
(= (type-of x) "dict")
(= (get x :type) artdag/post-type)
(has-key? x :id)
(has-key? x :wire))))
(define artdag/post-object-id (fn (post) (get post :id)))
(define artdag/post-object-wire (fn (post) (get post :wire)))
; integrity: the payload's records each verify (id == recomputed content-id) AND the
; claimed post id is actually produced by the job (present among the wire records).
(define
artdag/post-object-verify
(fn
(post)
(and
(artdag/post-object? post)
(artdag/wire-verify (get post :wire))
(artdag/member?
(get post :id)
(map (fn (rec) (nth rec 0)) (get post :wire))))))
; decode the payload back into a runnable dag (pure; verify separately, mirroring
; serialize.sx's wire->dag / wire-verify split).
(define
artdag/post-object->job
(fn (post) (artdag/wire->dag (get post :wire))))
; ---- string transport (drop into a feed activity / SXTP body) ----
(define
artdag/job->post-string
(fn
(dag output-name)
(write-to-string (artdag/job->post-object dag output-name))))
(define artdag/post-string->object (fn (s) (read (open-input-string s))))
; ---- run a received post: decode -> run -> result at the post id ----
; the peer recomputes the job (content-addressed, so a warm cache hits everything it
; already has). Returns the result of the output node the post is about.
(define
artdag/post-run
(fn
(post runner cache)
(artdag/result-of
(artdag/run (artdag/post-object->job post) runner cache)
(artdag/post-object-id post))))

View File

@@ -10,10 +10,11 @@
"serialize": {"pass": 13, "fail": 0}, "serialize": {"pass": 13, "fail": 0},
"stats": {"pass": 12, "fail": 0}, "stats": {"pass": 12, "fail": 0},
"fault": {"pass": 14, "fail": 0}, "fault": {"pass": 14, "fail": 0},
"post": {"pass": 12, "fail": 0},
"maude-optimize": {"pass": 40, "fail": 0}, "maude-optimize": {"pass": 40, "fail": 0},
"schedule": {"pass": 15, "fail": 0} "schedule": {"pass": 15, "fail": 0}
}, },
"total_pass": 213, "total_pass": 225,
"total_fail": 0, "total_fail": 0,
"total": 213 "total": 225
} }

View File

@@ -14,6 +14,7 @@ _Generated by `lib/artdag/conformance.sh`_
| serialize | 13 | 0 | 13 | | serialize | 13 | 0 | 13 |
| stats | 12 | 0 | 12 | | stats | 12 | 0 | 12 |
| fault | 14 | 0 | 14 | | fault | 14 | 0 | 14 |
| post | 12 | 0 | 12 |
| maude-optimize | 40 | 0 | 40 | | maude-optimize | 40 | 0 | 40 |
| schedule | 15 | 0 | 15 | | schedule | 15 | 0 | 15 |
| **Total** | **213** | **0** | **213** | | **Total** | **225** | **0** | **225** |

111
lib/artdag/tests/post.sx Normal file
View File

@@ -0,0 +1,111 @@
; Forward direction — artdag job as a feed "post object" (per the host loop).
; A job projects to a content-addressed, self-verifying object suitable as a feed
; activity :object; a peer decodes, verifies and runs it to the same result.
(define po-runner (artdag/op-table-runner {:blur (fn (params inputs) (+ (first inputs) (get params :radius))) :src (fn (params inputs) 0) :over (fn (params inputs) (+ (nth inputs 0) (nth inputs 1)))}))
(define
po-job
(artdag/build
(list
(list "s" "src" (list) {})
(list "b" "blur" (list "s") {:radius 2})
(list "c" "blur" (list "s") {:radius 3})
(list "out" "over" (list "b" "c") {} true))))
(define po-out-id (artdag/dag-id po-job "out"))
(define po-post (artdag/job->post-object po-job "out"))
(artdag-test
"post: is a well-formed post object"
(artdag/post-object? po-post)
true)
(artdag-test "post: type tag is artdag/job" (get po-post :type) "artdag/job")
(artdag-test
"post: post id is the output node's content-id"
(artdag/post-object-id po-post)
po-out-id)
(artdag-test
"post: payload is the whole dag (one record per node)"
(len (artdag/post-object-wire po-post))
(artdag/node-count po-job))
(artdag-test
"post: verifies (ids intact, output present)"
(artdag/post-object-verify po-post)
true)
; ---- round-trip: decode reconstructs the job by content-id ----
(define po-job2 (artdag/post-object->job po-post))
(artdag-test
"post: decoded job contains the output node by content-id"
(artdag/member? po-out-id (keys (artdag/dag-nodes po-job2)))
true)
(artdag-test
"post: decoded job has the same node count"
(artdag/node-count po-job2)
(artdag/node-count po-job))
; ---- string transport (feed activity / SXTP body) ----
(define po-str (artdag/job->post-string po-job "out"))
(define po-post2 (artdag/post-string->object po-str))
(artdag-test
"post: survives string transport (id preserved)"
(artdag/post-object-id po-post2)
po-out-id)
(artdag-test
"post: transported post still verifies"
(artdag/post-object-verify po-post2)
true)
; ---- a peer runs the received post to the same result ----
(define
po-local-result
(artdag/result-of (artdag/run po-job po-runner (persist/open)) po-out-id))
(define po-peer-result (artdag/post-run po-post2 po-runner (persist/open)))
(artdag-test
"post: peer runs the received job to the same result"
(= po-peer-result po-local-result)
true)
; ---- tamper detection: mutate a param under a stale id ----
(define
po-tampered
(assoc
po-post
:wire (map
(fn
(rec)
(if
(= (nth rec 1) "blur")
(list
(nth rec 0)
(nth rec 1)
(nth rec 2)
{:radius 99}
(nth rec 4))
rec))
(artdag/post-object-wire po-post))))
(artdag-test
"post: tampered payload fails verification"
(artdag/post-object-verify po-tampered)
false)
; ---- an id not produced by the job fails verification ----
(artdag-test
"post: post id absent from payload fails verification"
(artdag/post-object-verify (assoc po-post :id "node:bogus"))
false)

View File

@@ -31,7 +31,7 @@ edges.
## Status (rolling) ## Status (rolling)
`bash lib/artdag/conformance.sh`**213/213** (12 suites: dag, analyze, plan, execute, optimize, fed, cost, serialize, stats, fault, maude-optimize, schedule) `bash lib/artdag/conformance.sh`**225/225** (13 suites: dag, analyze, plan, execute, optimize, fed, cost, serialize, stats, fault, post, maude-optimize, schedule)
Base roadmap (Phases 16) COMPLETE + Phase 7 (maude rule-based optimization) COMPLETE Base roadmap (Phases 16) COMPLETE + Phase 7 (maude rule-based optimization) COMPLETE
(only optional miniKanren scheduling remains). Now hardening only. (only optional miniKanren scheduling remains). Now hardening only.
@@ -117,10 +117,14 @@ loop's `GET/POST /feed`. The engine already has the primitives; the wrapping liv
- **federation already mirrors the feed** — `fed-export`/`fed-import` are trust-gated with - **federation already mirrors the feed** — `fed-export`/`fed-import` are trust-gated with
provenance; a re-posted job dedupes/cache-hits by global content-id. provenance; a re-posted job dedupes/cache-hits by global content-id.
Boundary: host loop owns the `dag ⇄ feed-object` adapter; `lib/artdag` stays the engine. Boundary: host loop owns the activity wrapping (actor/verb/at/tags); `lib/artdag` owns the
Candidate artdag-side affordance (only if wanted here): a thin `job->post-object` / job⇄object projection. **BUILT — `lib/artdag/post.sx`** (post suite 12/12): a post object is
`post-object->job` projection so host never reaches into wire internals. Not yet built — `{:type "artdag/job" :id <output content-id> :wire <dag->wire>}``job->post-object`,
flagged, not scheduled. `post-object->job`, `post-object-verify` (wire ids intact + the post id is produced by the
job; rejects tampered params and bogus ids), `job->post-string`/`post-string->object` for the
feed/SXTP body, and `post-run` (a peer decodes → runs → result at the post id, content-address
cache-hitting what it already has). The host loop drops the object into a feed activity's
`:object`; post-id = content-id = the AP object id.
## Phase 1 — DAG model + content addressing ## Phase 1 — DAG model + content addressing
@@ -237,6 +241,19 @@ be an op token.
## Progress log ## Progress log
- **2026-06-28 Forward — job as a feed post object** (post suite 12/12, total 225/225).
`lib/artdag/post.sx`: the artdag-side projection for "a job is a type of post" (per the
host loop). `job->post-object dag output-name``{:type "artdag/job" :id <content-id of
output> :wire <dag->wire>}` — the post/object id IS the output node's content-id (= the
natural AP object id), the body is serialize.sx's self-describing wire. `post-object-verify`
binds the claimed id to the payload (every record's id recomputes + the post id is present
among them) — rejects a param tampered under a stale id and a bogus post id.
`job->post-string`/`post-string->object` carry it in a feed activity / SXTP body;
`post-object->job` decodes; `post-run post runner cache` lets a peer decode → run → read the
result at the post id (content-addressed, so a warm peer cache-hits what it already holds).
The activity wrapper (actor/verb/at/tags) stays on the host/feed side — this is only
job⇄object. Reuses serialize.sx + execute.sx; no new substrate.
- **2026-06-28 Phase 3/7 — miniKanren CLP(FD) scheduler** (schedule suite 15/15, total - **2026-06-28 Phase 3/7 — miniKanren CLP(FD) scheduler** (schedule suite 15/15, total
213/213). `lib/artdag/schedule.sx` on `lib/minikanren` (read-only substrate): each node 213/213). `lib/artdag/schedule.sx` on `lib/minikanren` (read-only substrate): each node
gets a slot var in `[1..max-slots]`, every edge `(input->node)` imposes `fd-lt gets a slot var in `[1..max-slots]`, every edge `(input->node)` imposes `fd-lt