Skip to content

fix(node): carry full owner DID on ref-update wire event (#144)#145

Open
Gravirei wants to merge 15 commits into
Gitlawb:mainfrom
Gravirei:fix/issue-144-full-did-wire-slug
Open

fix(node): carry full owner DID on ref-update wire event (#144)#145
Gravirei wants to merge 15 commits into
Gitlawb:mainfrom
Gravirei:fix/issue-144-full-did-wire-slug

Conversation

@Gravirei

@Gravirei Gravirei commented Jul 2, 2026

Copy link
Copy Markdown
Contributor

Summary

Carry the full owner DID on the ref-update wire event so the feed gate can distinguish different DID methods that share the same trailing segment (e.g. did:web:host:alice vs did:gitlawb:alice).

Motivation & context

Refs #144

The ref-update wire slug was previously constructed from the last colon-segmented component of the owner DID ({owner_did.split(":").last}/{name}), which is lossy. Two owners with different DID methods but the same trailing segment would produce identical slugs, causing the feed gate to over-drop remote rows for anonymous callers.

This PR is the plumbing half of #144: it carries the full owner DID on the wire and stores it in the database so a follow-up can teach the feed gate to use did_matches for proper disambiguation. The gate itself (the second half of #144) will land in a separate change on top of this.

Kind of change

  • Bug fix
  • Feature
  • Security fix
  • Docs
  • Tests / CI
  • Refactor (no behavior change)
  • Breaking or protocol change (issue required first)

What changed

crates/gitlawb-node (gitlawb-node):

  • p2p/mod.rs: Added owner_did: Option<String> to RefUpdateEvent with #[serde(default)] for backward compat with older peers that do not include the field
  • db/mod.rs: Added owner_did: Option<String> to ReceivedRefUpdate, added migration v10 to add owner_did TEXT column and idx_ref_updates_owner index to the received_ref_updates table, updated all SELECT/INSERT queries to include the column
  • api/repos.rs: Set owner_did: Some(record.owner_did) when publishing ref-update events over gossip; added owner_did to the HTTP sync notify body
  • api/peers.rs: Added owner_did to NotifyRequest and wired it through to ReceivedRefUpdate

How a reviewer can verify

DATABASE_URL=postgresql://gitlawb:changeme@172.19.0.2:5432/gitlawb cargo test -p gitlawb-node
cargo fmt --check
cargo clippy --workspace --all-targets -- -D warnings

Before you request review

  • Scope is one logical change; no unrelated churn
  • cargo test --workspace passes locally (327/327)
  • New behavior is covered by tests (required for fixes)
  • cargo fmt --all and cargo clippy --workspace --all-targets -- -D warnings are clean
  • Commit titles use Conventional Commits (fix(...), test(...), style(...))
  • Docs / .env.example updated if behavior or config changed (or N/A)
  • Checked existing PRs so this is not a duplicate

Protocol & signing impact

This is a backward-compatible wire-format addition: the new owner_did field is #[serde(default)], so events from older peers deserialize without it (owner_did: None). The repo field format is unchanged. Migration v10 adds the column to existing databases; ADD COLUMN IF NOT EXISTS is idempotent for re-runs.

  • Touches DID / did:key, Ed25519 / RFC 9421 signatures, UCAN, ref certs, or P2P wire formats
  • Discussed in an issue before implementation
  • Backward-compatible with existing nodes and previously signed history

Summary by CodeRabbit

  • New Features

    • Ref-update notifications and event listings can now include an optional owner_did when available.
    • Peer-to-peer sync notifications now propagate the owner DID into published ref-update events.
    • GraphQL ref-updates and subscriptions expose owner_did as optional (omittable).
  • Bug Fixes

    • Owner DID is now stored and returned consistently across sync, peer updates, API event endpoints, and persisted ref-update history.
    • Backward compatibility is preserved when owner_did is missing or null in incoming notifications.

Gravirei added 9 commits June 30, 2026 19:52
Gitlawb#126)

The IPFS visibility gate used withheld_blob_oids (a deny-set enumerating
only reachable blobs), so a dangling/unreachable blob was absent from the
set and served in cleartext to anonymous callers. Flip to an allowed-set
(allowed_blob_set_for_caller) that enumerates reachable blobs the caller
may read: a dangling blob has no path, is never in the set, and 404s.
Move store::read_object before the allowed_blob_set_for_caller
spawn_blocking call so random-CID spray against repos with
path-scoped rules cannot trigger full-history git walks on
repos that don't carry the object.
…tion

✓ P3 blocker fixed: cargo fmt applied — the format gate will pass.

  ✓ P3 cleanup resolved: withheld_blob_oids is still used by replication code in repos.rs, so it stays.

  • P2 follow-up: Tree/commit disclosure tracked in Gitlawb#135 — out of scope here.
@github-actions github-actions Bot added the needs-tests Source changed without accompanying tests (advisory) label Jul 2, 2026
@github-actions

github-actions Bot commented Jul 2, 2026

Copy link
Copy Markdown
Contributor

Thanks for the contribution. A couple of things will help us review this faster:

  • This changes Rust source but no tests changed. Tests are required for fixes and strongly encouraged for features.

See CONTRIBUTING.md. Update the PR and these notes will clear automatically.

@coderabbitai

coderabbitai Bot commented Jul 2, 2026

Copy link
Copy Markdown

Review Change Stack

Note

Reviews paused

It looks like this branch is under active development. To avoid overwhelming you with review comments due to an influx of new commits, CodeRabbit has automatically paused this review. You can configure this behavior by changing the reviews.auto_review.auto_pause_after_reviewed_commits setting.

Use the following commands to manage reviews:

  • @coderabbitai resume to resume automatic reviews.
  • @coderabbitai review to trigger a single review.

Use the checkboxes below for quick actions:

  • ▶️ Resume reviews
  • 🔍 Trigger review

No actionable comments were generated in the recent review. 🎉

ℹ️ Recent review info
⚙️ Run configuration

Configuration used: defaults

Review profile: CHILL

Plan: Pro

Run ID: 966a016b-b4a9-4651-8bdb-30309a7ad1b4

📥 Commits

Reviewing files that changed from the base of the PR and between 59c26a7 and 63ab601.

📒 Files selected for processing (1)
  • crates/gitlawb-node/src/api/events.rs
✅ Files skipped from review due to trivial changes (1)
  • crates/gitlawb-node/src/api/events.rs

📝 Walkthrough

Walkthrough

Adds an optional owner_did field across ref-update request, event, persistence, GraphQL, and events API paths, and threads it through peer notifications, gossip handling, database queries, and endpoint tests.

Changes

Owner DID propagation

Layer / File(s) Summary
HTTP sync/notify and peer fan-out
crates/gitlawb-node/src/api/peers.rs, crates/gitlawb-node/src/api/repos.rs
Adds owner_did to sync notify requests, forwards it through peer notification payloads, and includes it in ref-update event publication and tests.
Gossip RefUpdateEvent owner_did propagation
crates/gitlawb-node/src/p2p/mod.rs, crates/gitlawb-node/src/state.rs
Adds owner_did to gossip ref-update events, persists it on receipt, and updates the broadcast payload contract.
ReceivedRefUpdate schema and DB storage
crates/gitlawb-node/src/db/mod.rs
Adds owner_did to the stored ref-update shape, migration, queries, and DB tests.
Events and GraphQL responses
crates/gitlawb-node/src/api/events.rs, crates/gitlawb-node/src/graphql/*.rs, crates/gitlawb-node/src/test_support.rs
Returns owner_did in ref-update events and GraphQL ref-update responses, updates schema mapping, and adds endpoint tests.

Estimated code review effort: 3 (Moderate) | ~25 minutes

Possibly related issues

Possibly related PRs

  • Gitlawb/node#39: Modifies the same /api/v1/sync/notify path and NotifyRequest/ReceivedRefUpdate flow.
  • Gitlawb/node#68: Touches the same git_receive_pack path in api/repos.rs and uses record.owner_did in adjacent logic.
  • Gitlawb/node#72: Changes the same peer notification payload-building path in api/repos.rs.

Suggested labels: sev:medium, subsystem:api

Suggested reviewers: jatmn, kevincodex1

🚥 Pre-merge checks | ✅ 5
✅ Passed checks (5 passed)
Check name Status Explanation
Title check ✅ Passed The title accurately summarizes the main change: carrying the full owner DID on ref-update wire events.
Description check ✅ Passed The description follows the template well, covering summary, motivation, changes, verification, and protocol impact.
Docstring Coverage ✅ Passed No functions found in the changed files to evaluate docstring coverage. Skipping docstring coverage check.
Linked Issues check ✅ Passed Check skipped because no linked issues were found for this pull request.
Out of Scope Changes check ✅ Passed Check skipped because no linked issues were found for this pull request.
✨ Finishing Touches
🧪 Generate unit tests (beta)
  • Create PR with unit tests

Thanks for using CodeRabbit! It's free for OSS, and your support helps us grow. If you like it, consider giving us a shout-out.

❤️ Share

Comment @coderabbitai help to get the list of available commands.

@beardthelion beardthelion added crate:node gitlawb-node — the serving node and REST API kind:bug Defect fix — wrong or unsafe behavior subsystem:peers Peer announce, discovery, and registry labels Jul 2, 2026
@github-actions github-actions Bot removed the needs-tests Source changed without accompanying tests (advisory) label Jul 2, 2026
@Gravirei Gravirei marked this pull request as ready for review July 2, 2026 03:16

@coderabbitai coderabbitai Bot left a comment

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

🧹 Nitpick comments (2)
crates/gitlawb-node/src/test_support.rs (1)

1969-2043: 📐 Maintainability & Code Quality | 🔵 Trivial | 💤 Low value

Consider a shared builder for ReceivedRefUpdate test fixtures.

Both tests construct near-identical ReceivedRefUpdate literals inline. A shared helper (similar to the update(...) fixture already used in db/mod.rs tests) would reduce duplication and make future field additions (like owner_did) less error-prone to keep in sync across test files.

🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@crates/gitlawb-node/src/test_support.rs` around lines 1969 - 2043, The
`events_returns_inserted_ref_updates` and `events_limit_respects_limit_param`
tests duplicate the same `crate::db::ReceivedRefUpdate` setup inline, so add a
shared test fixture/builder for `ReceivedRefUpdate` in `test_support.rs` and use
it in both cases. Model it after the existing `update(...)` helper used in the
db tests, and make sure the helper accepts overrides for fields like `repo`,
`owner_did`, `new_sha`, and timestamps so future schema changes stay consistent
across tests.
crates/gitlawb-node/src/db/mod.rs (1)

3919-4077: 📐 Maintainability & Code Quality | 🔵 Trivial | ⚡ Quick win

Assert owner_did in the filtered-by-repo test too.

list_ref_updates_filtered_by_repo (lines 4027-4058) inserts records with distinct owner_did values but only asserts id/count, not owner_did, unlike the sibling list_repo_ref_updates_filters_by_repo test which does check it. Adding that assertion would close a small gap in coverage for the exact field this PR introduces.

♻️ Proposed test strengthening
         let filtered = db
             .list_ref_updates_filtered(Some("ownerA/proj"), 100)
             .await
             .unwrap();
         assert_eq!(filtered.len(), 1);
         assert_eq!(filtered[0].id, "u5");
+        assert_eq!(filtered[0].owner_did.as_deref(), Some("did:key:zA"));
🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@crates/gitlawb-node/src/db/mod.rs` around lines 3919 - 4077, The
list_ref_updates_filtered_by_repo test is missing coverage for the new owner_did
behavior. Update ref_update_db_tests::list_ref_updates_filtered_by_repo to
assert the returned row’s owner_did matches the inserted value, similar to
list_repo_ref_updates_filters_by_repo, so the filtered path verifies both repo
filtering and owner_did preservation.
🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Inline comments:
In `@crates/gitlawb-node/src/test_support.rs`:
- Around line 1969-2007: The ref-update events payload returned by the events
API is still missing owner_did, so the current test only verifies persistence in
Postgres. Update the ref-update feed serialization in the events response path
to include owner_did, using the relevant event model/handler that backs
events_router and GET /api/v1/events/ref-updates, and then extend
events_returns_inserted_ref_updates to assert events[0]["owner_did"] matches the
inserted value.

---

Nitpick comments:
In `@crates/gitlawb-node/src/db/mod.rs`:
- Around line 3919-4077: The list_ref_updates_filtered_by_repo test is missing
coverage for the new owner_did behavior. Update
ref_update_db_tests::list_ref_updates_filtered_by_repo to assert the returned
row’s owner_did matches the inserted value, similar to
list_repo_ref_updates_filters_by_repo, so the filtered path verifies both repo
filtering and owner_did preservation.

In `@crates/gitlawb-node/src/test_support.rs`:
- Around line 1969-2043: The `events_returns_inserted_ref_updates` and
`events_limit_respects_limit_param` tests duplicate the same
`crate::db::ReceivedRefUpdate` setup inline, so add a shared test
fixture/builder for `ReceivedRefUpdate` in `test_support.rs` and use it in both
cases. Model it after the existing `update(...)` helper used in the db tests,
and make sure the helper accepts overrides for fields like `repo`, `owner_did`,
`new_sha`, and timestamps so future schema changes stay consistent across tests.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: defaults

Review profile: CHILL

Plan: Pro

Run ID: 6d0fa77a-014d-4a04-af46-9ef99e20fe27

📥 Commits

Reviewing files that changed from the base of the PR and between 466a550 and b988780.

📒 Files selected for processing (5)
  • crates/gitlawb-node/src/api/peers.rs
  • crates/gitlawb-node/src/api/repos.rs
  • crates/gitlawb-node/src/db/mod.rs
  • crates/gitlawb-node/src/p2p/mod.rs
  • crates/gitlawb-node/src/test_support.rs

Comment thread crates/gitlawb-node/src/test_support.rs

@beardthelion beardthelion left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The write-side plumbing is right: the full DID comes from the local repo record, it's threaded through both the gossip and HTTP-notify paths, and the #[serde(default)] / nullable column keeps older peers compatible. CodeRabbit's feed-serialization ask is genuinely addressed too. Two things block merge, and the first is a hard one: the schema change breaks every existing node on upgrade, and the PR closes #144 without fixing #144's symptom.

Findings

  • [P1] Ship owner_did as a new migration version, not appended to v1
    crates/gitlawb-node/src/db/mod.rs:542
    The ALTER TABLE ... ADD COLUMN owner_did and idx_ref_updates_owner were added inside Migration { version: 1 }, but run_migrations skips any version already recorded in schema_migrations, and the catalogue comment right above the array says new schema must be a new version (v2, v3, ...). Every existing node has v1 recorded, so on upgrade the column is never created; insert_ref_update (binds $12) and all three list_* SELECTs then reference a missing column and error, which stops ref-update ingestion on both the gossip and notify paths and 500s the events feed. Fresh databases pass because they run v1 from scratch, which is why CI is green. I reproduced it against the real migration runner: an existing node (v1..v9 recorded, no owner_did) upgraded to this build leaves the column absent and the insert/select fail; moving both statements into a new Migration { version: 10 } makes the same upgrade round-trip cleanly.

  • [P2] Don't close #144 on this PR; nothing consumes owner_did yet
    crates/gitlawb-node/src/api/events.rs:38
    #144 has two halves: carry the full owner DID (done here) and make the feed gate use it to stop the over-drop. Nothing in this diff reads owner_did for a visibility decision; it's stored and echoed in the feed JSON only, and idx_ref_updates_owner is an index no query uses. The gate that over-drops isn't in this branch, so merging this auto-closes #144 while the reported symptom persists. Keep the plumbing, but change "Closes #144" to "Refs #144" and land the gate consumption (the did:key-aware match #144 describes) in the follow-up that sits on top of the gate.

  • [P3] Populate owner_did on the other read surfaces, or hold the JSON change
    crates/gitlawb-node/src/graphql/query.rs:61, crates/gitlawb-node/src/api/events.rs:117
    The field lands on the global REST feed but the GraphQL ref_updates resolver drops it (RefUpdateType has no such field) and the repo-scoped list_repo_events gossip block omits it. No consumer reads it yet so there's no impact today, but the asymmetry is the same drift the duplicated four-copy SELECT column list invites, and it becomes a trap once something depends on the field. Either populate all three surfaces or defer the JSON exposure until the gate needs it. While here, add a migration upgrade-path test (seed schema_migrations at v1, run migrations, assert the column round-trips) so the v1-append class can't recur.

The direction is right and the receive/publish wiring is sound; once the migration moves to its own version this is close.

@Gravirei Gravirei requested a review from beardthelion July 2, 2026 05:27
@beardthelion beardthelion dismissed their stale review July 2, 2026 13:36

P1 migration break resolved on 72ccb28; re-reviewed the new head, posting a fresh review.

@beardthelion beardthelion left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Re-reviewed on 72ccb28d. The P1 migration break is resolved and I verified it by execution: the owner_did DDL now lives in a new Migration{version:10}, and on a simulated existing node (schema_migrations at v1..v9, received_ref_updates without the column) migrate() adds the column and index and records v10. Putting the DDL back inside the already-applied v1 makes that same check fail (the column is never re-added), so the guard is load-bearing. Wire compat (#[serde(default)] Option<String>) and the DB plumbing are sound, and "Refs #144" is correct. Four items to fold in.

Findings

  • [P2] Drop the premature idx_ref_updates_owner index, or defer it to the gate PR
    crates/gitlawb-node/src/db/mod.rs:833
    Nothing reads received_ref_updates by owner_did yet: every reference is a projection, no WHERE/JOIN/ORDER. Migrations run inside a transaction so CREATE INDEX CONCURRENTLY isn't available here (mod.rs:426), which means this index builds under a write-blocking lock at startup on exactly the populated nodes this PR targets, for no current benefit. Ship the column now and add the index in #143 next to the query that reads it.

  • [P2] Expose owner_did on the repo-scoped events feed
    crates/gitlawb-node/src/api/events.rs:116
    list_ref_updates returns owner_did now, but the gossip_events block in list_repo_events omits it, so GET /api/v1/repos/{owner}/{repo}/events silently drops the field. Add "owner_did": u.owner_did to that json! block to match line 38.

  • [P2] Carry owner_did through the GraphQL query and subscription
    crates/gitlawb-node/src/graphql/types.rs:52
    RefUpdateType and RefUpdateBroadcast (state.rs:12) have no owner_did, so the ref_updates query (query.rs:61) and the subscription drop it while REST exposes it. Add owner_did: Option<String> to both structs and populate it at query.rs, subscription.rs, and the broadcast send in repos.rs.

  • [P2] Add an existing-node upgrade test for v10
    crates/gitlawb-node/src/db/mod.rs:3200
    migration_v10_creates_owner_did_column runs only a fresh v1..v10 chain, which can't catch DDL appended to an already-applied version. Seed schema_migrations with v1..v9 and a received_ref_updates without the column, then migrate() and assert the column, index, and v10 row appear. I ran exactly this against the head and it passes; the same body fails when the DDL sits back in v1, so it is the missing regression guard, not a live bug.

One forward note for #143, not a blocker here: the wire owner_did is attacker-controlled (the peer signature authenticates the relaying node, not this field) and is stored verbatim. Every authorization path today reads the trusted record.owner_did from the local repo row, so nothing consumes the wire value yet, but the gate must cross-check it against a trusted source and must not fall back to the lossy last-segment / did_matches matcher the full DID exists to avoid.

@Gravirei migration fix verified, thanks. The above are plumbing-consistency and deploy-window items.

- Drop idx_ref_updates_owner from Migration v10; no query reads
  owner_did yet and CREATE INDEX (non-CONCURRENT) inside a transaction
  takes a write-blocking lock on populated nodes for zero benefit.
  Defer the index to Gitlawb#143 alongside the query that uses it.

- Expose owner_did on repo-scoped events feed: add the field to the
  gossip_events json! block in list_repo_events so GET
  /api/v1/repos/{owner}/{repo}/events is consistent with
  GET /api/v1/events/ref-updates.

- Carry owner_did through GraphQL: add owner_did: Option<String> to
  RefUpdateBroadcast (state.rs) and RefUpdateType (types.rs), populate
  it from the DB row in query.rs, from the broadcast in subscription.rs,
  and from the trusted local record.owner_did in the broadcast send in
  repos.rs.

- Add migration_v10_existing_node_upgrade test: seeds a v1..v9 state,
  strips the column and version row, re-runs migrate(), and asserts the
  column and v10 row appear. This is the regression guard: it fails if
  the DDL is accidentally moved back into v1.

Refs Gitlawb#144

@coderabbitai coderabbitai Bot left a comment

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

Caution

Some comments are outside the diff and can’t be posted inline due to platform limitations.

⚠️ Outside diff range comments (2)
crates/gitlawb-node/src/db/mod.rs (2)

3207-3218: 📐 Maintainability & Code Quality | 🟡 Minor | ⚡ Quick win

Assert the column is actually nullable.

The tests fetch is_nullable but only assert the name/type. Since older peers can omit owner_did, please assert "YES" so a future NOT NULL change is caught.

Proposed fix
         assert_eq!(col.0, "owner_did");
         assert_eq!(col.1, "text");
+        assert_eq!(col.2, "YES");

Apply the same assertion in both migration tests.

Also applies to: 3274-3284

🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@crates/gitlawb-node/src/db/mod.rs` around lines 3207 - 3218, The migration
tests for received_ref_updates currently verify owner_did exists and is text,
but they ignore the fetched is_nullable value. Update both migration test blocks
that query information_schema.columns in db::mod to assert the third tuple field
from sqlx::query_as is "YES" alongside the existing name/type checks, so changes
to nullable behavior are caught.

2103-2107: 🗄️ Data Integrity & Integration | 🟠 Major | ⚡ Quick win

Preserve later owner_did enrichment on duplicate ref-update IDs.

With ON CONFLICT(id) DO NOTHING, a mixed-version network can store the older None payload first and ignore a later duplicate carrying the full owner DID, leaving the feed unable to disambiguate that event.

Proposed fix
              (id, node_did, pusher_did, repo, ref_name, old_sha, new_sha, timestamp,
               cert_id, received_at, from_peer, owner_did)
              VALUES ($1,$2,$3,$4,$5,$6,$7,$8,$9,$10,$11,$12)
-             ON CONFLICT(id) DO NOTHING",
+             ON CONFLICT(id) DO UPDATE
+             SET owner_did = COALESCE(received_ref_updates.owner_did, EXCLUDED.owner_did)
+             WHERE received_ref_updates.owner_did IS NULL
+               AND EXCLUDED.owner_did IS NOT NULL",
🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@crates/gitlawb-node/src/db/mod.rs` around lines 2103 - 2107, The INSERT into
received_ref_updates currently uses ON CONFLICT(id) DO NOTHING, which prevents a
later duplicate from enriching an earlier row with owner_did. Update the upsert
logic in the received ref-update insert path so duplicates with the same id can
fill in a missing owner_did while still avoiding overwriting existing non-null
data. Refer to the received_ref_updates insert statement and the surrounding db
write function that persists ref updates.
🧹 Nitpick comments (1)
crates/gitlawb-node/src/db/mod.rs (1)

4096-4129: 📐 Maintainability & Code Quality | 🔵 Trivial | ⚡ Quick win

Add a same-slug owner collision regression.

This test uses different repo values, so it does not cover the PR’s core case: two updates with the same trailing repo slug but different full owner_did values. Add a row pair like alice/myrepo + did:web:host:alice and alice/myrepo + did:gitlawb:alice.

🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@crates/gitlawb-node/src/db/mod.rs` around lines 4096 - 4129, The current
`list_repo_ref_updates_filters_by_repo` test only verifies filtering by full
repo name, not the same-slug owner collision case. Update this test in `db` to
insert two ref updates with the same trailing repo slug but different full owner
identifiers, using the existing `insert_ref_update`, `update`, and
`list_repo_ref_updates` helpers, and assert the query returns the correct row
for each full repo owner path. Keep the original repo-filter assertions if
helpful, but add coverage for the `owner_did` collision scenario the PR is
targeting.
🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Inline comments:
In `@crates/gitlawb-node/src/api/events.rs`:
- Line 130: The event payload in list_repo_events is only populating owner_did
for gossipsub events, so local cert events in the same feed remain inconsistent.
Update the local event mapping in events.rs to also set owner_did from
record.owner_did, keeping the repo-scoped payload shape consistent with the
existing gossipsub path and preserving consumers that dedup or gate on the full
owner DID.

---

Outside diff comments:
In `@crates/gitlawb-node/src/db/mod.rs`:
- Around line 3207-3218: The migration tests for received_ref_updates currently
verify owner_did exists and is text, but they ignore the fetched is_nullable
value. Update both migration test blocks that query information_schema.columns
in db::mod to assert the third tuple field from sqlx::query_as is "YES"
alongside the existing name/type checks, so changes to nullable behavior are
caught.
- Around line 2103-2107: The INSERT into received_ref_updates currently uses ON
CONFLICT(id) DO NOTHING, which prevents a later duplicate from enriching an
earlier row with owner_did. Update the upsert logic in the received ref-update
insert path so duplicates with the same id can fill in a missing owner_did while
still avoiding overwriting existing non-null data. Refer to the
received_ref_updates insert statement and the surrounding db write function that
persists ref updates.

---

Nitpick comments:
In `@crates/gitlawb-node/src/db/mod.rs`:
- Around line 4096-4129: The current `list_repo_ref_updates_filters_by_repo`
test only verifies filtering by full repo name, not the same-slug owner
collision case. Update this test in `db` to insert two ref updates with the same
trailing repo slug but different full owner identifiers, using the existing
`insert_ref_update`, `update`, and `list_repo_ref_updates` helpers, and assert
the query returns the correct row for each full repo owner path. Keep the
original repo-filter assertions if helpful, but add coverage for the `owner_did`
collision scenario the PR is targeting.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: defaults

Review profile: CHILL

Plan: Pro

Run ID: 9f70fb07-7129-4c96-9e77-7b2943106a56

📥 Commits

Reviewing files that changed from the base of the PR and between 72ccb28 and 59c26a7.

📒 Files selected for processing (7)
  • crates/gitlawb-node/src/api/events.rs
  • crates/gitlawb-node/src/api/repos.rs
  • crates/gitlawb-node/src/db/mod.rs
  • crates/gitlawb-node/src/graphql/query.rs
  • crates/gitlawb-node/src/graphql/subscription.rs
  • crates/gitlawb-node/src/graphql/types.rs
  • crates/gitlawb-node/src/state.rs
🚧 Files skipped from review as they are similar to previous changes (1)
  • crates/gitlawb-node/src/api/repos.rs

Comment thread crates/gitlawb-node/src/api/events.rs
Local cert events in the repo-scoped feed now include owner_did sourced
from the trusted local record, matching the field already present on
gossipsub events. Keeps the payload consistent for consumers that
gate or dedup by full owner DID.

Refs Gitlawb#144
@Gravirei Gravirei requested a review from beardthelion July 2, 2026 16:20

@beardthelion beardthelion left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Re-reviewed on 63ab6015. All four items from my last pass are in, CodeRabbit's local-cert-events fix landed, and I verified the security- and deploy-relevant paths by execution:

  • Migration, both ways: on a simulated existing node (v1..v9 recorded, no owner_did column), migrate() adds the column and records v10; moving that DDL back into the already-applied v1 makes migration_v10_existing_node_upgrade fail (RowNotFound) while the fresh-DB test still passes, so the upgrade guard is load-bearing. A pre-upgrade row (owner_did NULL, since the ALTER has no default) round-trips through the feed as JSON null, no 500. The index is correctly deferred to #143.

  • Feed provenance, executed: the repo-scoped feed's local block carries the trusted record.owner_did, and the gossip block echoes the stored wire value verbatim, which I confirmed with a spoofed owner_did surfacing unchanged (display-only, nothing filters on it). All four received_ref_updates SELECTs project owner_did, so the shared row mapper never hits a missing column. The global feed and GraphQL (query, subscription, broadcast) carry the field through as a passthrough.

  • [P3] The stored received_ref_updates.owner_did is attacker-controlled (the peer signature authenticates the relaying node, not this field, and notify_sync accepts it unsigned by default) and is echoed verbatim, as the spoof test shows. It is display-only and nothing gates on it, so not a break. A short comment at the two ingest sites (p2p/mod.rs, api/peers.rs:398) marking the column untrusted would keep #143's DID-aware feed gate from consuming it as trusted; that gate must reconcile it against the repo slug or a verified cert.

@Gravirei the P2s and the migration all check out, thanks. @kevincodex1 this is ready.

@kevincodex1 kevincodex1 left a comment

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

looks good to me!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

crate:node gitlawb-node — the serving node and REST API kind:bug Defect fix — wrong or unsafe behavior subsystem:peers Peer announce, discovery, and registry

Projects

None yet

Development

Successfully merging this pull request may close these issues.

3 participants